Compare commits
21 Commits
Author | SHA1 | Date | |
---|---|---|---|
946420bcc3 | |||
8257268df5 | |||
97ebf34be0 | |||
f381b5421e | |||
|
9cfd5c4274 | ||
|
b0d51ad15e | ||
|
0a8b11e154 | ||
|
cbafaa5781 | ||
|
7c2af867b0 | ||
|
b86a80b001 | ||
|
f890d7f0ee | ||
|
31dc49eae4 | ||
|
267a891e4d | ||
|
bba80e3005 | ||
|
8330101154 | ||
|
8997e6ae1a | ||
|
91c2214082 | ||
|
b175b02b87 | ||
|
b6b5814588 | ||
|
81797f66ca | ||
3e5695a802 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,2 +1,3 @@
|
|||||||
config.json
|
config.json
|
||||||
housekeeper
|
housekeeper
|
||||||
|
.vscode
|
21
cmd/main.go
21
cmd/main.go
@@ -37,6 +37,8 @@ func main() {
|
|||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
libraryType := locator.GetLibraryType(cfg, sectionID)
|
||||||
|
|
||||||
ids, titles := locator.GetTitles(cfg, sectionID, days)
|
ids, titles := locator.GetTitles(cfg, sectionID, days)
|
||||||
|
|
||||||
if check {
|
if check {
|
||||||
@@ -47,11 +49,20 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if delete {
|
if delete {
|
||||||
files := eraser.LookupFileLocation(cfg, ids)
|
if libraryType == "movie" {
|
||||||
err = eraser.DeleteMedia(delete, files)
|
files := eraser.LookupMovieFileLocation(cfg, ids)
|
||||||
if err != nil {
|
err = eraser.DeleteFiles(delete, files)
|
||||||
log.Println(err)
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
} else if libraryType == "show" {
|
||||||
|
// files := eraser.LookupTVFileLocation(cfg, ids)
|
||||||
|
sonarrIDs := locator.GetSonarrIDs(cfg, titles)
|
||||||
|
eraser.DeleteSeriesFromSonarr(cfg, sonarrIDs)
|
||||||
|
// err = eraser.DeleteFiles(delete, files)
|
||||||
|
// if err != nil {
|
||||||
|
// log.Println(err)
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -11,28 +11,38 @@ import (
|
|||||||
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/config"
|
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type error interface {
|
||||||
|
Error() string
|
||||||
|
}
|
||||||
|
|
||||||
// TelegramPost will send a message to a specific ChatID in Telegram containing the list of items to be cleaned with this cleaner.
|
// TelegramPost will send a message to a specific ChatID in Telegram containing the list of items to be cleaned with this cleaner.
|
||||||
func TelegramPost(config config.Config, titles []string) error {
|
func TelegramPost(config config.Config, titles []string) error {
|
||||||
url := "https://api.telegram.org/bot" + config.TelegramToken + "/sendMessage"
|
var err error
|
||||||
|
if len(titles) != 0 {
|
||||||
|
url := "https://api.telegram.org/bot" + config.TelegramToken + "/sendMessage"
|
||||||
|
|
||||||
values := map[string]string{"chat_id": config.TelegramChatID, "text": "The following items are to be removed from " + config.ServerName + " in 24 hours. Please go to Plex and start the title to keep it on " + config.ServerName + ". You do not need to keep watching, just hit play and load a few seconds.\n\n" + fmt.Sprintf("%v", strings.Join(titles, "\n")), "disable_notifications": "true"}
|
values := map[string]string{"chat_id": config.TelegramChatID, "text": "The following items are to be removed from " + config.ServerName + " in 24 hours. Please go to Plex and start the title to keep it on " + config.ServerName + ". You do not need to keep watching, just hit play and load a few seconds.\n\n" + fmt.Sprintf("%v", strings.Join(titles, "\n")), "disable_notifications": "true"}
|
||||||
|
|
||||||
jsonValue, _ := json.Marshal(values)
|
jsonValue, _ := json.Marshal(values)
|
||||||
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonValue))
|
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonValue))
|
||||||
req.Header.Set("X-Custom-Header", "Housekeeper")
|
req.Header.Set("X-Custom-Header", "Housekeeper")
|
||||||
req.Header.Set("Content-Type", "application/json")
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
client := &http.Client{}
|
client := &http.Client{}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
fmt.Println("response Status:", resp.Status)
|
||||||
|
fmt.Println("response Headers:", resp.Header)
|
||||||
|
body, _ := ioutil.ReadAll(resp.Body)
|
||||||
|
fmt.Println("response Body:", string(body))
|
||||||
|
|
||||||
|
} else {
|
||||||
|
fmt.Println("There are no titles, therefore no message to send!")
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
fmt.Println("response Status:", resp.Status)
|
|
||||||
fmt.Println("response Headers:", resp.Header)
|
|
||||||
body, _ := ioutil.ReadAll(resp.Body)
|
|
||||||
fmt.Println("response Body:", string(body))
|
|
||||||
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@@ -18,6 +18,8 @@ type Config struct {
|
|||||||
TelegramToken string `json:"telegramToken"`
|
TelegramToken string `json:"telegramToken"`
|
||||||
TelegramChatID string `json:"telegramChatID"`
|
TelegramChatID string `json:"telegramChatID"`
|
||||||
ServerName string `json:"serverName"`
|
ServerName string `json:"serverName"`
|
||||||
|
SonarrContext string `json:"sonarrContext"`
|
||||||
|
SonarrAPIKey string `json:"sonarrAPIKey"`
|
||||||
}
|
}
|
||||||
|
|
||||||
//GetConfig gets the configuration values for the api using the file in the supplied configPath.
|
//GetConfig gets the configuration values for the api using the file in the supplied configPath.
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
package eraser
|
package eraser
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
@@ -13,17 +14,14 @@ import (
|
|||||||
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/model"
|
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
// LookupFileLocation will gather a list of Information based on IDs returned by locator.GetTitles
|
// LookupMovieFileLocation will gather a list of Information based on IDs returned by locator.GetTitles
|
||||||
func LookupFileLocation(config config.Config, ids []int) []string {
|
func LookupMovieFileLocation(config config.Config, ids []int) []string {
|
||||||
fileList := make([]string, 0)
|
fileList := make([]string, 0)
|
||||||
|
|
||||||
for _, i := range ids {
|
for _, i := range ids {
|
||||||
plexURL := fmt.Sprintf("%s:%d%s%d%s%s", config.PlexHost, config.PlexPort, "/library/metadata/", i, "/?X-Plex-Token=", config.PlexToken)
|
plexURL := fmt.Sprintf("%s:%d%s%d%s%s", config.PlexHost, config.PlexPort, "/library/metadata/", i, "/?X-Plex-Token=", config.PlexToken)
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, plexURL, nil)
|
req, err := http.NewRequest(http.MethodGet, plexURL, nil)
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
httpClient := http.Client{}
|
httpClient := http.Client{}
|
||||||
req.Header.Set("User-Agent", "Housekeeper")
|
req.Header.Set("User-Agent", "Housekeeper")
|
||||||
@@ -41,15 +39,95 @@ func LookupFileLocation(config config.Config, ids []int) []string {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
plexModel := model.XMLPlexAPI{}
|
plexModel := model.XMLPlexMovieAPI{}
|
||||||
xml.Unmarshal(body, &plexModel)
|
xml.Unmarshal(body, &plexModel)
|
||||||
fileList = append(fileList, filepath.Dir(plexModel.Video.Media.Part.File))
|
fileList = append(fileList, filepath.Dir(plexModel.Video.Media.Part.File))
|
||||||
}
|
}
|
||||||
return fileList
|
return fileList
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteMedia will actually perform the deletion.
|
// LookupTVFileLocation will gather a list of Information based on IDs returned by locator.GetTitles
|
||||||
func DeleteMedia(delete bool, files []string) error {
|
func LookupTVFileLocation(config config.Config, ids []int) []string {
|
||||||
|
fileList := make([]string, 0)
|
||||||
|
m := make(map[string]bool)
|
||||||
|
results := make([]string, 0)
|
||||||
|
|
||||||
|
for _, i := range ids {
|
||||||
|
plexURL := fmt.Sprintf("%s:%d%s%d%s%s", config.PlexHost, config.PlexPort, "/library/metadata/", i, "/allLeaves/?X-Plex-Token=", config.PlexToken)
|
||||||
|
|
||||||
|
req, err := http.NewRequest(http.MethodGet, plexURL, nil)
|
||||||
|
|
||||||
|
httpClient := http.Client{}
|
||||||
|
req.Header.Set("User-Agent", "Housekeeper")
|
||||||
|
|
||||||
|
res, getErr := httpClient.Do(req)
|
||||||
|
if getErr != nil {
|
||||||
|
log.Fatal(getErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, readErr := ioutil.ReadAll(res.Body)
|
||||||
|
if readErr != nil {
|
||||||
|
log.Fatal(readErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
plexModel := model.XMLPlexTVAPI{}
|
||||||
|
xml.Unmarshal(body, &plexModel)
|
||||||
|
|
||||||
|
plexTV := plexModel.Video
|
||||||
|
|
||||||
|
for _, i := range plexTV {
|
||||||
|
fileList = append(fileList, filepath.Dir(i.Media.Part.File))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, r := range fileList {
|
||||||
|
if _, ok := m[r]; !ok {
|
||||||
|
m[r] = true
|
||||||
|
results = append(results, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
return results
|
||||||
|
}
|
||||||
|
|
||||||
|
func DeleteSeriesFromSonarr(config config.Config, ids []int) {
|
||||||
|
for _, i := range ids {
|
||||||
|
sonarrURL := fmt.Sprintf("%s%s%s%d%s%s", config.BaseURL, config.SonarrContext, "/api/series/", i, "/?deleteFiles=true&apikey=", config.SonarrAPIKey)
|
||||||
|
req, err := http.NewRequest(http.MethodDelete, sonarrURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient := http.Client{}
|
||||||
|
req.Header.Set("User-Agent", "Housekeeper")
|
||||||
|
|
||||||
|
res, getErr := httpClient.Do(req)
|
||||||
|
if getErr != nil {
|
||||||
|
log.Fatal(getErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, readErr := ioutil.ReadAll(res.Body)
|
||||||
|
if readErr != nil {
|
||||||
|
log.Fatal(readErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteModel := model.SonarrResponse{}
|
||||||
|
jsonErr := json.Unmarshal(body, &deleteModel)
|
||||||
|
if jsonErr != nil {
|
||||||
|
log.Fatal(jsonErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// if strings.Contains("does not exist", deleteModel.Message) {
|
||||||
|
// log.Printf("The following ID does not exist: %v", i)
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteFiles will actually perform the deletion.
|
||||||
|
func DeleteFiles(delete bool, files []string) error {
|
||||||
var err error
|
var err error
|
||||||
if delete {
|
if delete {
|
||||||
for _, i := range files {
|
for _, i := range files {
|
||||||
|
@@ -2,6 +2,7 @@ package locator
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"encoding/xml"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
@@ -14,6 +15,44 @@ import (
|
|||||||
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/util"
|
"git.linuxrocker.com/mattburchett/Housekeeper/pkg/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// GetLibraryType checks to see what type the library is.
|
||||||
|
func GetLibraryType(config config.Config, sectionID int) string {
|
||||||
|
typeURL := fmt.Sprintf("%s:%d%s%d%s%s", config.PlexHost, config.PlexPort, "/library/sections/", sectionID, "/?X-Plex-Token=", config.PlexToken)
|
||||||
|
|
||||||
|
req, err := http.NewRequest(http.MethodGet, typeURL, nil)
|
||||||
|
httpClient := http.Client{}
|
||||||
|
req.Header.Set("User-Agent", "Housekeeper")
|
||||||
|
|
||||||
|
res, getErr := httpClient.Do(req)
|
||||||
|
if getErr != nil {
|
||||||
|
log.Fatal(getErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, readErr := ioutil.ReadAll(res.Body)
|
||||||
|
if readErr != nil {
|
||||||
|
log.Fatal(readErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
typeModel := model.XMLPlexLibraryType{}
|
||||||
|
xml.Unmarshal(body, &typeModel)
|
||||||
|
|
||||||
|
var libraryType string
|
||||||
|
|
||||||
|
if typeModel.Thumb == "/:/resources/movie.png" {
|
||||||
|
libraryType = "movie"
|
||||||
|
} else if typeModel.Thumb == "/:/resources/show.png" {
|
||||||
|
libraryType = "show"
|
||||||
|
} else {
|
||||||
|
log.Fatal("Unsupported library type found. This app only supports movies and shows.")
|
||||||
|
}
|
||||||
|
|
||||||
|
return libraryType
|
||||||
|
}
|
||||||
|
|
||||||
// GetCount will gather a count of media in a specific library, required for GetTitles.
|
// GetCount will gather a count of media in a specific library, required for GetTitles.
|
||||||
func GetCount(config config.Config, sectionID int) int {
|
func GetCount(config config.Config, sectionID int) int {
|
||||||
countURL := fmt.Sprintf("%s%s%s%s%s%d", config.BaseURL, config.PlexPyContext, "/api/v2?apikey=", config.PlexPyAPIKey, "&cmd=get_library§ion_id=", sectionID)
|
countURL := fmt.Sprintf("%s%s%s%s%s%d", config.BaseURL, config.PlexPyContext, "/api/v2?apikey=", config.PlexPyAPIKey, "&cmd=get_library§ion_id=", sectionID)
|
||||||
@@ -105,3 +144,42 @@ func GetTitles(config config.Config, sectionID int, days int) ([]int, []string)
|
|||||||
sort.Strings(titles)
|
sort.Strings(titles)
|
||||||
return ids, titles
|
return ids, titles
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetSonarrIDs gets the IDs to delete from the title list in PlexPy.
|
||||||
|
func GetSonarrIDs(config config.Config, titles []string) []int {
|
||||||
|
ids := make([]int, 0)
|
||||||
|
sonarrURL := fmt.Sprintf("%s%s%s%s", config.BaseURL, config.SonarrContext, "/api/series?apikey=", config.SonarrAPIKey)
|
||||||
|
req, err := http.NewRequest(http.MethodGet, sonarrURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient := http.Client{}
|
||||||
|
req.Header.Set("User-Agent", "Housekeeper")
|
||||||
|
|
||||||
|
res, getErr := httpClient.Do(req)
|
||||||
|
if getErr != nil {
|
||||||
|
log.Fatal(getErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, readErr := ioutil.ReadAll(res.Body)
|
||||||
|
if readErr != nil {
|
||||||
|
log.Fatal(readErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
sonarrModel := model.SonarrSeries{}
|
||||||
|
jsonErr := json.Unmarshal(body, &sonarrModel)
|
||||||
|
if jsonErr != nil {
|
||||||
|
log.Fatal(jsonErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, r := range sonarrModel {
|
||||||
|
for _, s := range titles {
|
||||||
|
if r.Title == s {
|
||||||
|
ids = append(ids, r.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ids
|
||||||
|
}
|
||||||
|
237
pkg/model/eraser_model.go
Normal file
237
pkg/model/eraser_model.go
Normal file
@@ -0,0 +1,237 @@
|
|||||||
|
package model
|
||||||
|
|
||||||
|
import "encoding/xml"
|
||||||
|
|
||||||
|
// XMLPlexMovieAPI - This is the XML version of the Library.
|
||||||
|
type XMLPlexMovieAPI struct {
|
||||||
|
XMLName xml.Name `xml:"MediaContainer"`
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
Size string `xml:"size,attr"`
|
||||||
|
AllowSync string `xml:"allowSync,attr"`
|
||||||
|
Identifier string `xml:"identifier,attr"`
|
||||||
|
LibrarySectionID string `xml:"librarySectionID,attr"`
|
||||||
|
LibrarySectionTitle string `xml:"librarySectionTitle,attr"`
|
||||||
|
LibrarySectionUUID string `xml:"librarySectionUUID,attr"`
|
||||||
|
MediaTagPrefix string `xml:"mediaTagPrefix,attr"`
|
||||||
|
MediaTagVersion string `xml:"mediaTagVersion,attr"`
|
||||||
|
Video struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
RatingKey string `xml:"ratingKey,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
GUID string `xml:"guid,attr"`
|
||||||
|
LibrarySectionTitle string `xml:"librarySectionTitle,attr"`
|
||||||
|
LibrarySectionID string `xml:"librarySectionID,attr"`
|
||||||
|
LibrarySectionKey string `xml:"librarySectionKey,attr"`
|
||||||
|
Studio string `xml:"studio,attr"`
|
||||||
|
Type string `xml:"type,attr"`
|
||||||
|
Title string `xml:"title,attr"`
|
||||||
|
ContentRating string `xml:"contentRating,attr"`
|
||||||
|
Summary string `xml:"summary,attr"`
|
||||||
|
Rating string `xml:"rating,attr"`
|
||||||
|
AudienceRating string `xml:"audienceRating,attr"`
|
||||||
|
Year string `xml:"year,attr"`
|
||||||
|
Tagline string `xml:"tagline,attr"`
|
||||||
|
Thumb string `xml:"thumb,attr"`
|
||||||
|
Art string `xml:"art,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
OriginallyAvailableAt string `xml:"originallyAvailableAt,attr"`
|
||||||
|
AddedAt string `xml:"addedAt,attr"`
|
||||||
|
UpdatedAt string `xml:"updatedAt,attr"`
|
||||||
|
AudienceRatingImage string `xml:"audienceRatingImage,attr"`
|
||||||
|
ChapterSource string `xml:"chapterSource,attr"`
|
||||||
|
PrimaryExtraKey string `xml:"primaryExtraKey,attr"`
|
||||||
|
RatingImage string `xml:"ratingImage,attr"`
|
||||||
|
Media struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
VideoResolution string `xml:"videoResolution,attr"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
Bitrate string `xml:"bitrate,attr"`
|
||||||
|
Width string `xml:"width,attr"`
|
||||||
|
Height string `xml:"height,attr"`
|
||||||
|
AspectRatio string `xml:"aspectRatio,attr"`
|
||||||
|
AudioChannels string `xml:"audioChannels,attr"`
|
||||||
|
AudioCodec string `xml:"audioCodec,attr"`
|
||||||
|
VideoCodec string `xml:"videoCodec,attr"`
|
||||||
|
Container string `xml:"container,attr"`
|
||||||
|
VideoFrameRate string `xml:"videoFrameRate,attr"`
|
||||||
|
AudioProfile string `xml:"audioProfile,attr"`
|
||||||
|
VideoProfile string `xml:"videoProfile,attr"`
|
||||||
|
Part struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
File string `xml:"file,attr"`
|
||||||
|
Size string `xml:"size,attr"`
|
||||||
|
AudioProfile string `xml:"audioProfile,attr"`
|
||||||
|
Container string `xml:"container,attr"`
|
||||||
|
VideoProfile string `xml:"videoProfile,attr"`
|
||||||
|
Stream []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
StreamType string `xml:"streamType,attr"`
|
||||||
|
Default string `xml:"default,attr"`
|
||||||
|
Codec string `xml:"codec,attr"`
|
||||||
|
Index string `xml:"index,attr"`
|
||||||
|
Bitrate string `xml:"bitrate,attr"`
|
||||||
|
Language string `xml:"language,attr"`
|
||||||
|
LanguageCode string `xml:"languageCode,attr"`
|
||||||
|
BitDepth string `xml:"bitDepth,attr"`
|
||||||
|
ChromaLocation string `xml:"chromaLocation,attr"`
|
||||||
|
ChromaSubsampling string `xml:"chromaSubsampling,attr"`
|
||||||
|
FrameRate string `xml:"frameRate,attr"`
|
||||||
|
HasScalingMatrix string `xml:"hasScalingMatrix,attr"`
|
||||||
|
Height string `xml:"height,attr"`
|
||||||
|
Level string `xml:"level,attr"`
|
||||||
|
Profile string `xml:"profile,attr"`
|
||||||
|
RefFrames string `xml:"refFrames,attr"`
|
||||||
|
ScanType string `xml:"scanType,attr"`
|
||||||
|
Title string `xml:"title,attr"`
|
||||||
|
Width string `xml:"width,attr"`
|
||||||
|
DisplayTitle string `xml:"displayTitle,attr"`
|
||||||
|
Selected string `xml:"selected,attr"`
|
||||||
|
Channels string `xml:"channels,attr"`
|
||||||
|
AudioChannelLayout string `xml:"audioChannelLayout,attr"`
|
||||||
|
SamplingRate string `xml:"samplingRate,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
} `xml:"Stream"`
|
||||||
|
} `xml:"Part"`
|
||||||
|
} `xml:"Media"`
|
||||||
|
Genre []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Genre"`
|
||||||
|
Director struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Director"`
|
||||||
|
Writer []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Writer"`
|
||||||
|
Producer []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Producer"`
|
||||||
|
Country struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Country"`
|
||||||
|
Role []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Filter string `xml:"filter,attr"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
Role string `xml:"role,attr"`
|
||||||
|
Thumb string `xml:"thumb,attr"`
|
||||||
|
} `xml:"Role"`
|
||||||
|
} `xml:"Video"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// XMLPlexTVAPI - This is the XML version of the Library.
|
||||||
|
type XMLPlexTVAPI struct {
|
||||||
|
XMLName xml.Name `xml:"MediaContainer"`
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
Size string `xml:"size,attr"`
|
||||||
|
AllowSync string `xml:"allowSync,attr"`
|
||||||
|
Art string `xml:"art,attr"`
|
||||||
|
Banner string `xml:"banner,attr"`
|
||||||
|
Identifier string `xml:"identifier,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
LibrarySectionID string `xml:"librarySectionID,attr"`
|
||||||
|
LibrarySectionTitle string `xml:"librarySectionTitle,attr"`
|
||||||
|
LibrarySectionUUID string `xml:"librarySectionUUID,attr"`
|
||||||
|
MediaTagPrefix string `xml:"mediaTagPrefix,attr"`
|
||||||
|
MediaTagVersion string `xml:"mediaTagVersion,attr"`
|
||||||
|
MixedParents string `xml:"mixedParents,attr"`
|
||||||
|
Nocache string `xml:"nocache,attr"`
|
||||||
|
ParentIndex string `xml:"parentIndex,attr"`
|
||||||
|
ParentTitle string `xml:"parentTitle,attr"`
|
||||||
|
ParentYear string `xml:"parentYear,attr"`
|
||||||
|
Theme string `xml:"theme,attr"`
|
||||||
|
Title1 string `xml:"title1,attr"`
|
||||||
|
Title2 string `xml:"title2,attr"`
|
||||||
|
ViewGroup string `xml:"viewGroup,attr"`
|
||||||
|
ViewMode string `xml:"viewMode,attr"`
|
||||||
|
Video []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
RatingKey string `xml:"ratingKey,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
ParentRatingKey string `xml:"parentRatingKey,attr"`
|
||||||
|
GrandparentRatingKey string `xml:"grandparentRatingKey,attr"`
|
||||||
|
Studio string `xml:"studio,attr"`
|
||||||
|
Type string `xml:"type,attr"`
|
||||||
|
Title string `xml:"title,attr"`
|
||||||
|
GrandparentKey string `xml:"grandparentKey,attr"`
|
||||||
|
ParentKey string `xml:"parentKey,attr"`
|
||||||
|
GrandparentTitle string `xml:"grandparentTitle,attr"`
|
||||||
|
ParentTitle string `xml:"parentTitle,attr"`
|
||||||
|
ContentRating string `xml:"contentRating,attr"`
|
||||||
|
Summary string `xml:"summary,attr"`
|
||||||
|
Index string `xml:"index,attr"`
|
||||||
|
ParentIndex string `xml:"parentIndex,attr"`
|
||||||
|
Rating string `xml:"rating,attr"`
|
||||||
|
Year string `xml:"year,attr"`
|
||||||
|
Thumb string `xml:"thumb,attr"`
|
||||||
|
Art string `xml:"art,attr"`
|
||||||
|
ParentThumb string `xml:"parentThumb,attr"`
|
||||||
|
GrandparentThumb string `xml:"grandparentThumb,attr"`
|
||||||
|
GrandparentArt string `xml:"grandparentArt,attr"`
|
||||||
|
GrandparentTheme string `xml:"grandparentTheme,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
OriginallyAvailableAt string `xml:"originallyAvailableAt,attr"`
|
||||||
|
AddedAt string `xml:"addedAt,attr"`
|
||||||
|
UpdatedAt string `xml:"updatedAt,attr"`
|
||||||
|
TitleSort string `xml:"titleSort,attr"`
|
||||||
|
Media struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
VideoResolution string `xml:"videoResolution,attr"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
Bitrate string `xml:"bitrate,attr"`
|
||||||
|
Width string `xml:"width,attr"`
|
||||||
|
Height string `xml:"height,attr"`
|
||||||
|
AspectRatio string `xml:"aspectRatio,attr"`
|
||||||
|
AudioChannels string `xml:"audioChannels,attr"`
|
||||||
|
AudioCodec string `xml:"audioCodec,attr"`
|
||||||
|
VideoCodec string `xml:"videoCodec,attr"`
|
||||||
|
Container string `xml:"container,attr"`
|
||||||
|
VideoFrameRate string `xml:"videoFrameRate,attr"`
|
||||||
|
VideoProfile string `xml:"videoProfile,attr"`
|
||||||
|
Part struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
ID string `xml:"id,attr"`
|
||||||
|
Key string `xml:"key,attr"`
|
||||||
|
Duration string `xml:"duration,attr"`
|
||||||
|
File string `xml:"file,attr"`
|
||||||
|
Size string `xml:"size,attr"`
|
||||||
|
Container string `xml:"container,attr"`
|
||||||
|
VideoProfile string `xml:"videoProfile,attr"`
|
||||||
|
} `xml:"Part"`
|
||||||
|
} `xml:"Media"`
|
||||||
|
Director struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Director"`
|
||||||
|
Writer []struct {
|
||||||
|
Text string `xml:",chardata"`
|
||||||
|
Tag string `xml:"tag,attr"`
|
||||||
|
} `xml:"Writer"`
|
||||||
|
} `xml:"Video"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SonarrResponse struct {
|
||||||
|
Message string `json:"message"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
}
|
@@ -1,6 +1,9 @@
|
|||||||
package model
|
package model
|
||||||
|
|
||||||
import "encoding/xml"
|
import (
|
||||||
|
"encoding/xml"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
// PlexPyLibraryInfo will gather all the library related info. We really just need the count from this...
|
// PlexPyLibraryInfo will gather all the library related info. We really just need the count from this...
|
||||||
type PlexPyLibraryInfo struct {
|
type PlexPyLibraryInfo struct {
|
||||||
@@ -63,139 +66,91 @@ type PlexPyMediaInfo struct {
|
|||||||
} `json:"response"`
|
} `json:"response"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// XMLPlexAPI - This is the XML version of the struct below it.
|
type XMLPlexLibraryType struct {
|
||||||
type XMLPlexAPI struct {
|
XMLName xml.Name `xml:"MediaContainer"`
|
||||||
XMLName xml.Name `xml:"MediaContainer"`
|
Text string `xml:",chardata"`
|
||||||
Text string `xml:",chardata"`
|
Size string `xml:"size,attr"`
|
||||||
Size string `xml:"size,attr"`
|
AllowSync string `xml:"allowSync,attr"`
|
||||||
AllowSync string `xml:"allowSync,attr"`
|
Art string `xml:"art,attr"`
|
||||||
Identifier string `xml:"identifier,attr"`
|
Content string `xml:"content,attr"`
|
||||||
LibrarySectionID string `xml:"librarySectionID,attr"`
|
Identifier string `xml:"identifier,attr"`
|
||||||
LibrarySectionTitle string `xml:"librarySectionTitle,attr"`
|
LibrarySectionID string `xml:"librarySectionID,attr"`
|
||||||
LibrarySectionUUID string `xml:"librarySectionUUID,attr"`
|
MediaTagPrefix string `xml:"mediaTagPrefix,attr"`
|
||||||
MediaTagPrefix string `xml:"mediaTagPrefix,attr"`
|
MediaTagVersion string `xml:"mediaTagVersion,attr"`
|
||||||
MediaTagVersion string `xml:"mediaTagVersion,attr"`
|
Nocache string `xml:"nocache,attr"`
|
||||||
Video struct {
|
Thumb string `xml:"thumb,attr"`
|
||||||
Text string `xml:",chardata"`
|
Title1 string `xml:"title1,attr"`
|
||||||
RatingKey string `xml:"ratingKey,attr"`
|
ViewGroup string `xml:"viewGroup,attr"`
|
||||||
Key string `xml:"key,attr"`
|
ViewMode string `xml:"viewMode,attr"`
|
||||||
GUID string `xml:"guid,attr"`
|
Directory []struct {
|
||||||
LibrarySectionTitle string `xml:"librarySectionTitle,attr"`
|
Text string `xml:",chardata"`
|
||||||
LibrarySectionID string `xml:"librarySectionID,attr"`
|
Key string `xml:"key,attr"`
|
||||||
LibrarySectionKey string `xml:"librarySectionKey,attr"`
|
Title string `xml:"title,attr"`
|
||||||
Studio string `xml:"studio,attr"`
|
Secondary string `xml:"secondary,attr"`
|
||||||
Type string `xml:"type,attr"`
|
Prompt string `xml:"prompt,attr"`
|
||||||
Title string `xml:"title,attr"`
|
Search string `xml:"search,attr"`
|
||||||
ContentRating string `xml:"contentRating,attr"`
|
} `xml:"Directory"`
|
||||||
Summary string `xml:"summary,attr"`
|
}
|
||||||
Rating string `xml:"rating,attr"`
|
|
||||||
AudienceRating string `xml:"audienceRating,attr"`
|
// SonarrSeries type takes all the data from Sonarr and places it in a struct
|
||||||
Year string `xml:"year,attr"`
|
type SonarrSeries []struct {
|
||||||
Tagline string `xml:"tagline,attr"`
|
Title string `json:"title"`
|
||||||
Thumb string `xml:"thumb,attr"`
|
AlternateTitles []struct {
|
||||||
Art string `xml:"art,attr"`
|
Title string `json:"title"`
|
||||||
Duration string `xml:"duration,attr"`
|
SeasonNumber int `json:"seasonNumber"`
|
||||||
OriginallyAvailableAt string `xml:"originallyAvailableAt,attr"`
|
} `json:"alternateTitles"`
|
||||||
AddedAt string `xml:"addedAt,attr"`
|
SortTitle string `json:"sortTitle"`
|
||||||
UpdatedAt string `xml:"updatedAt,attr"`
|
SeasonCount int `json:"seasonCount"`
|
||||||
AudienceRatingImage string `xml:"audienceRatingImage,attr"`
|
TotalEpisodeCount int `json:"totalEpisodeCount"`
|
||||||
ChapterSource string `xml:"chapterSource,attr"`
|
EpisodeCount int `json:"episodeCount"`
|
||||||
PrimaryExtraKey string `xml:"primaryExtraKey,attr"`
|
EpisodeFileCount int `json:"episodeFileCount"`
|
||||||
RatingImage string `xml:"ratingImage,attr"`
|
SizeOnDisk int64 `json:"sizeOnDisk"`
|
||||||
Media struct {
|
Status string `json:"status"`
|
||||||
Text string `xml:",chardata"`
|
Overview string `json:"overview"`
|
||||||
VideoResolution string `xml:"videoResolution,attr"`
|
PreviousAiring time.Time `json:"previousAiring"`
|
||||||
ID string `xml:"id,attr"`
|
Network string `json:"network"`
|
||||||
Duration string `xml:"duration,attr"`
|
AirTime string `json:"airTime,omitempty"`
|
||||||
Bitrate string `xml:"bitrate,attr"`
|
Images []struct {
|
||||||
Width string `xml:"width,attr"`
|
CoverType string `json:"coverType"`
|
||||||
Height string `xml:"height,attr"`
|
URL string `json:"url"`
|
||||||
AspectRatio string `xml:"aspectRatio,attr"`
|
} `json:"images"`
|
||||||
AudioChannels string `xml:"audioChannels,attr"`
|
Seasons []struct {
|
||||||
AudioCodec string `xml:"audioCodec,attr"`
|
SeasonNumber int `json:"seasonNumber"`
|
||||||
VideoCodec string `xml:"videoCodec,attr"`
|
Monitored bool `json:"monitored"`
|
||||||
Container string `xml:"container,attr"`
|
Statistics struct {
|
||||||
VideoFrameRate string `xml:"videoFrameRate,attr"`
|
PreviousAiring time.Time `json:"previousAiring"`
|
||||||
AudioProfile string `xml:"audioProfile,attr"`
|
EpisodeFileCount int `json:"episodeFileCount"`
|
||||||
VideoProfile string `xml:"videoProfile,attr"`
|
EpisodeCount int `json:"episodeCount"`
|
||||||
Part struct {
|
TotalEpisodeCount int `json:"totalEpisodeCount"`
|
||||||
Text string `xml:",chardata"`
|
SizeOnDisk int64 `json:"sizeOnDisk"`
|
||||||
ID string `xml:"id,attr"`
|
PercentOfEpisodes float64 `json:"percentOfEpisodes"`
|
||||||
Key string `xml:"key,attr"`
|
} `json:"statistics"`
|
||||||
Duration string `xml:"duration,attr"`
|
} `json:"seasons"`
|
||||||
File string `xml:"file,attr"`
|
Year int `json:"year"`
|
||||||
Size string `xml:"size,attr"`
|
Path string `json:"path"`
|
||||||
AudioProfile string `xml:"audioProfile,attr"`
|
ProfileID int `json:"profileId"`
|
||||||
Container string `xml:"container,attr"`
|
SeasonFolder bool `json:"seasonFolder"`
|
||||||
VideoProfile string `xml:"videoProfile,attr"`
|
Monitored bool `json:"monitored"`
|
||||||
Stream []struct {
|
UseSceneNumbering bool `json:"useSceneNumbering"`
|
||||||
Text string `xml:",chardata"`
|
Runtime int `json:"runtime"`
|
||||||
ID string `xml:"id,attr"`
|
TvdbID int `json:"tvdbId"`
|
||||||
StreamType string `xml:"streamType,attr"`
|
TvRageID int `json:"tvRageId"`
|
||||||
Default string `xml:"default,attr"`
|
TvMazeID int `json:"tvMazeId"`
|
||||||
Codec string `xml:"codec,attr"`
|
FirstAired time.Time `json:"firstAired"`
|
||||||
Index string `xml:"index,attr"`
|
LastInfoSync time.Time `json:"lastInfoSync"`
|
||||||
Bitrate string `xml:"bitrate,attr"`
|
SeriesType string `json:"seriesType"`
|
||||||
Language string `xml:"language,attr"`
|
CleanTitle string `json:"cleanTitle"`
|
||||||
LanguageCode string `xml:"languageCode,attr"`
|
ImdbID string `json:"imdbId,omitempty"`
|
||||||
BitDepth string `xml:"bitDepth,attr"`
|
TitleSlug string `json:"titleSlug"`
|
||||||
ChromaLocation string `xml:"chromaLocation,attr"`
|
Certification string `json:"certification,omitempty"`
|
||||||
ChromaSubsampling string `xml:"chromaSubsampling,attr"`
|
Genres []string `json:"genres"`
|
||||||
FrameRate string `xml:"frameRate,attr"`
|
Tags []interface{} `json:"tags"`
|
||||||
HasScalingMatrix string `xml:"hasScalingMatrix,attr"`
|
Added time.Time `json:"added"`
|
||||||
Height string `xml:"height,attr"`
|
Ratings struct {
|
||||||
Level string `xml:"level,attr"`
|
Votes int `json:"votes"`
|
||||||
Profile string `xml:"profile,attr"`
|
Value float64 `json:"value"`
|
||||||
RefFrames string `xml:"refFrames,attr"`
|
} `json:"ratings"`
|
||||||
ScanType string `xml:"scanType,attr"`
|
QualityProfileID int `json:"qualityProfileId"`
|
||||||
Title string `xml:"title,attr"`
|
ID int `json:"id"`
|
||||||
Width string `xml:"width,attr"`
|
NextAiring time.Time `json:"nextAiring,omitempty"`
|
||||||
DisplayTitle string `xml:"displayTitle,attr"`
|
|
||||||
Selected string `xml:"selected,attr"`
|
|
||||||
Channels string `xml:"channels,attr"`
|
|
||||||
AudioChannelLayout string `xml:"audioChannelLayout,attr"`
|
|
||||||
SamplingRate string `xml:"samplingRate,attr"`
|
|
||||||
Key string `xml:"key,attr"`
|
|
||||||
} `xml:"Stream"`
|
|
||||||
} `xml:"Part"`
|
|
||||||
} `xml:"Media"`
|
|
||||||
Genre []struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
} `xml:"Genre"`
|
|
||||||
Director struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
} `xml:"Director"`
|
|
||||||
Writer []struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
} `xml:"Writer"`
|
|
||||||
Producer []struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
} `xml:"Producer"`
|
|
||||||
Country struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
} `xml:"Country"`
|
|
||||||
Role []struct {
|
|
||||||
Text string `xml:",chardata"`
|
|
||||||
ID string `xml:"id,attr"`
|
|
||||||
Filter string `xml:"filter,attr"`
|
|
||||||
Tag string `xml:"tag,attr"`
|
|
||||||
Role string `xml:"role,attr"`
|
|
||||||
Thumb string `xml:"thumb,attr"`
|
|
||||||
} `xml:"Role"`
|
|
||||||
} `xml:"Video"`
|
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user