emby.go 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. package emby_helper
  2. import (
  3. "fmt"
  4. "github.com/allanpk716/ChineseSubFinder/internal/common"
  5. "github.com/allanpk716/ChineseSubFinder/internal/pkg/log_helper"
  6. "github.com/allanpk716/ChineseSubFinder/internal/types/emby"
  7. "github.com/go-resty/resty/v2"
  8. "github.com/panjf2000/ants/v2"
  9. "golang.org/x/net/context"
  10. "sync"
  11. "time"
  12. )
  13. type EmbyApi struct {
  14. embyConfig emby.EmbyConfig
  15. threads int
  16. timeOut time.Duration
  17. }
  18. func NewEmbyHelper(embyConfig emby.EmbyConfig) *EmbyApi {
  19. em := EmbyApi{}
  20. em.embyConfig = embyConfig
  21. if em.embyConfig.LimitCount < common.EmbyApiGetItemsLimitMin ||
  22. em.embyConfig.LimitCount > common.EmbyApiGetItemsLimitMax {
  23. em.embyConfig.LimitCount = common.EmbyApiGetItemsLimitMin
  24. }
  25. em.threads = 6
  26. em.timeOut = 30 * time.Second
  27. return &em
  28. }
  29. // RefreshRecentlyVideoInfo 字幕下载完毕一次,就可以触发一次这个。并发 6 线程去刷新
  30. func (em EmbyApi) RefreshRecentlyVideoInfo() error {
  31. items, err := em.GetRecentlyItems()
  32. if err != nil {
  33. return err
  34. }
  35. updateFunc := func(i interface{}) error {
  36. tmpId := i.(string)
  37. return em.UpdateVideoSubList(tmpId)
  38. }
  39. p, err := ants.NewPoolWithFunc(em.threads, func(inData interface{}) {
  40. data := inData.(InputData)
  41. defer data.Wg.Done()
  42. ctx, cancel := context.WithTimeout(context.Background(), em.timeOut)
  43. defer cancel()
  44. done := make(chan error, 1)
  45. panicChan := make(chan interface{}, 1)
  46. go func() {
  47. defer func() {
  48. if p := recover(); p != nil {
  49. panicChan <- p
  50. }
  51. }()
  52. done <- updateFunc(data.Id)
  53. }()
  54. select {
  55. case err = <-done:
  56. if err != nil {
  57. log_helper.GetLogger().Errorln("RefreshRecentlyVideoInfo.NewPoolWithFunc got error", err)
  58. }
  59. return
  60. case p := <-panicChan:
  61. log_helper.GetLogger().Errorln("RefreshRecentlyVideoInfo.NewPoolWithFunc got panic", p)
  62. case <-ctx.Done():
  63. log_helper.GetLogger().Errorln("RefreshRecentlyVideoInfo.NewPoolWithFunc got time out", ctx.Err())
  64. return
  65. }
  66. })
  67. if err != nil {
  68. return err
  69. }
  70. defer p.Release()
  71. wg := sync.WaitGroup{}
  72. for _, item := range items.Items {
  73. wg.Add(1)
  74. err = p.Invoke(InputData{Id: item.Id, Wg: &wg})
  75. if err != nil {
  76. log_helper.GetLogger().Errorln("RefreshRecentlyVideoInfo ants.Invoke", err)
  77. }
  78. }
  79. wg.Wait()
  80. return nil
  81. }
  82. // GetRecentlyItems 获取近期的视频,在 API 调试界面 -- ItemsService
  83. func (em EmbyApi) GetRecentlyItems() (emby.EmbyRecentlyItems, error) {
  84. var recItems emby.EmbyRecentlyItems
  85. recItems.Items = make([]emby.EmbyRecentlyItem, 0)
  86. var recItemMap = make(map[string]emby.EmbyRecentlyItem)
  87. var recItemExsitMap = make(map[string]emby.EmbyRecentlyItem)
  88. var err error
  89. if em.embyConfig.SkipWatched == false {
  90. // 默认是不指定某一个User的视频列表
  91. _, err = em.getNewClient().R().
  92. SetQueryParams(map[string]string{
  93. "api_key": em.embyConfig.ApiKey,
  94. "IsUnaired": "false",
  95. "Limit": fmt.Sprintf("%d", em.embyConfig.LimitCount),
  96. "Recursive": "true",
  97. "SortOrder": "Descending",
  98. "IncludeItemTypes": "Episode,Movie",
  99. "Filters": "IsNotFolder",
  100. "SortBy": "DateCreated",
  101. }).
  102. SetResult(&recItems).
  103. Get(em.embyConfig.Url + "/emby/Items")
  104. } else {
  105. var userIds emby.EmbyUsers
  106. userIds, err = em.GetUserIdList()
  107. if err != nil {
  108. return emby.EmbyRecentlyItems{}, err
  109. }
  110. for _, item := range userIds.Items {
  111. var tmpRecItems emby.EmbyRecentlyItems
  112. // 获取指定用户的视频列表
  113. _, err = em.getNewClient().R().
  114. SetQueryParams(map[string]string{
  115. "api_key": em.embyConfig.ApiKey,
  116. "IsUnaired": "false",
  117. "Limit": fmt.Sprintf("%d", em.embyConfig.LimitCount),
  118. "Recursive": "true",
  119. "SortOrder": "Descending",
  120. "IncludeItemTypes": "Episode,Movie",
  121. "Filters": "IsNotFolder",
  122. "SortBy": "DateCreated",
  123. }).
  124. SetResult(&tmpRecItems).
  125. Get(em.embyConfig.Url + "/emby/Users/" + item.Id + "/Items")
  126. if err != nil {
  127. return emby.EmbyRecentlyItems{}, err
  128. }
  129. // 相同的视频项目,需要判断是否已经看过了,看过的需要排除
  130. // 项目是否相同可以通过 Id 判断
  131. for _, recentlyItem := range tmpRecItems.Items {
  132. // 这个视频是否已经插入过了,可能会进行删除
  133. _, bFound := recItemMap[recentlyItem.Id]
  134. if bFound == false {
  135. // map 中不存在
  136. // 如果没有播放过,则插入
  137. if recentlyItem.UserData.Played == false {
  138. recItemMap[recentlyItem.Id] = recentlyItem
  139. }
  140. } else {
  141. // map 中存在
  142. // 既然存在,则可以理解为其他人是没有看过的,但是,如果当前的用户看过了,那么就要删除这一条
  143. if recentlyItem.UserData.Played == true {
  144. // 先记录下来,然后再删除这一条
  145. recItemExsitMap[recentlyItem.Id] = recentlyItem
  146. }
  147. }
  148. recItemMap[recentlyItem.Id] = recentlyItem
  149. }
  150. }
  151. for id := range recItemExsitMap {
  152. delete(recItemMap, id)
  153. }
  154. for _, item := range recItemMap {
  155. recItems.Items = append(recItems.Items, item)
  156. }
  157. recItems.TotalRecordCount = len(recItemMap)
  158. }
  159. return recItems, nil
  160. }
  161. // GetUserIdList 获取所有的 UserId
  162. func (em EmbyApi) GetUserIdList() (emby.EmbyUsers, error) {
  163. var recItems emby.EmbyUsers
  164. _, err := em.getNewClient().R().
  165. SetQueryParams(map[string]string{
  166. "api_key": em.embyConfig.ApiKey,
  167. }).
  168. SetResult(&recItems).
  169. Get(em.embyConfig.Url + "/emby/Users/Query")
  170. if err != nil {
  171. return emby.EmbyUsers{}, err
  172. }
  173. return recItems, nil
  174. }
  175. // GetItemAncestors 获取父级信息,在 API 调试界面 -- LibraryService
  176. func (em EmbyApi) GetItemAncestors(id string) ([]emby.EmbyItemsAncestors, error) {
  177. var recItems []emby.EmbyItemsAncestors
  178. _, err := em.getNewClient().R().
  179. SetQueryParams(map[string]string{
  180. "api_key": em.embyConfig.ApiKey,
  181. }).
  182. SetResult(&recItems).
  183. Get(em.embyConfig.Url + "/emby/Items/" + id + "/Ancestors")
  184. if err != nil {
  185. return nil, err
  186. }
  187. return recItems, nil
  188. }
  189. // GetItemVideoInfo 在 API 调试界面 -- UserLibraryService
  190. func (em EmbyApi) GetItemVideoInfo(id string) (emby.EmbyVideoInfo, error) {
  191. var recItem emby.EmbyVideoInfo
  192. _, err := em.getNewClient().R().
  193. SetQueryParams(map[string]string{
  194. "api_key": em.embyConfig.ApiKey,
  195. }).
  196. SetResult(&recItem).
  197. Get(em.embyConfig.Url + "/emby/LiveTv/Programs/" + id)
  198. if err != nil {
  199. return emby.EmbyVideoInfo{}, err
  200. }
  201. return recItem, nil
  202. }
  203. // GetItemVideoInfoByUserId 可以拿到这个视频的选择字幕Index,配合 GetItemVideoInfo 使用。 在 API 调试界面 -- UserLibraryService
  204. func (em EmbyApi) GetItemVideoInfoByUserId(userId, videoId string) (emby.EmbyVideoInfoByUserId, error) {
  205. var recItem emby.EmbyVideoInfoByUserId
  206. _, err := em.getNewClient().R().
  207. SetQueryParams(map[string]string{
  208. "api_key": em.embyConfig.ApiKey,
  209. }).
  210. SetResult(&recItem).
  211. Get(em.embyConfig.Url + "/emby/Users/" + userId + "/Items/" + videoId)
  212. if err != nil {
  213. return emby.EmbyVideoInfoByUserId{}, err
  214. }
  215. return recItem, nil
  216. }
  217. // UpdateVideoSubList 更新字幕列表, 在 API 调试界面 -- ItemRefreshService
  218. func (em EmbyApi) UpdateVideoSubList(id string) error {
  219. _, err := em.getNewClient().R().
  220. SetQueryParams(map[string]string{
  221. "api_key": em.embyConfig.ApiKey,
  222. }).
  223. Post(em.embyConfig.Url + "/emby/Items/" + id + "/Refresh")
  224. if err != nil {
  225. return err
  226. }
  227. return nil
  228. }
  229. // GetSubFileData 下载字幕 subExt -> .ass or .srt , 在 API 调试界面 -- SubtitleService
  230. func (em EmbyApi) GetSubFileData(videoId, mediaSourceId, subIndex, subExt string) (string, error) {
  231. response, err := em.getNewClient().R().
  232. Get(em.embyConfig.Url + "/emby/Videos/" + videoId + "/" + mediaSourceId + "/Subtitles/" + subIndex + "/Stream" + subExt)
  233. if err != nil {
  234. return "", err
  235. }
  236. return response.String(), nil
  237. }
  238. func (em EmbyApi) getNewClient() *resty.Client {
  239. tmpClient := resty.New()
  240. tmpClient.RemoveProxy()
  241. tmpClient.SetTimeout(em.timeOut)
  242. return tmpClient
  243. }
  244. type InputData struct {
  245. Id string
  246. Wg *sync.WaitGroup
  247. }