aboutsummaryrefslogtreecommitdiff
path: root/anime/anime.go
blob: 5d207ea1b45b7f2aa39fa14fdad54a179cef57c6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/* mal2go - MyAnimeList V2 API wrapper for Go
 * Copyright (C) 2022  Vidhu Kant Sharma <vidhukant@protonmail.ch>

 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.

 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.

 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <https://www.gnu.org/licenses/>. */

package anime

import (
	"encoding/json"
	"errors"
	"fmt"
	"log"
	"math"
	"strconv"
)

const BASE_URL string = "https://api.myanimelist.net/v2/anime"

// Each anime has its own ID on MAL
func GetAnimeById(token string, animeId int, fields []string) (Anime, error) {
  var anime Anime

  // Check if given fields are valid
  for _, j := range(fields) {
    if !isValidField(j) {
      return anime, errors.New(fmt.Sprintf("GetAnimeById: Invalid field specified: \"%s\"", j))
    }
  }

  // default fields to use when none are specified
  defaultFields := []string{
    "id", "title", "main_picture", 
    "alternative_titles", "start_date", 
    "end_date", "synopsis", "mean", "rank", 
    "popularity", "num_list_users", 
    "num_scoring_users", "nsfw", "created_at", 
    "updated_at", "media_type", "status", 
    "genres", "my_list_status", "num_episodes", 
    "start_season", "broadcast", "source", 
    "average_episode_duration", "rating", 
    "pictures", "background", "related_anime", 
    "related_manga", "recommendations", 
    "studios", "statistics",
  }

  if cap(fields) == 0 {
    fields = defaultFields
    log.Println("GetAnimeById: WARN: No fields specified, using all default fields to get data")
  }

  endpoint, _ := urlGenerator(
    BASE_URL + "/" + strconv.Itoa(animeId),
    []string{"fields"},
    /* it seems to still return all fields from the API. 
     * this might be an issue with MAL itself
     * TODO: look into this */
    [][]string{fields},
    true,
  )

  data := requestHandler(token, endpoint)
  json.Unmarshal([]byte(data), &anime)

  return anime, nil
}

// Ranking is a list of anime sorted by their rank
func GetAnimeRanking(token string, rankingType string, limit int, offset int) (AnimeRanking, error) {
  var animeRanking AnimeRanking

  // if limit exceeds what MAL supports
  if limit > 500 {
    return animeRanking, errors.New(fmt.Sprintf("GetAnimeRanking: Limit too high(%d). Max limit is 500", limit))
  } else if offset > 499 {
    return animeRanking, errors.New(fmt.Sprintf("GetAnimeRanking: Offset too high(%d). Max offset for mal2go is 499", offset))
  }

  // if ranking type is invalid
  if !isValidRankingType(rankingType) {
    return animeRanking, errors.New(fmt.Sprintf("GetAnimeRanking: Invalid ranking type specified: \"%s\"", rankingType))
  }

  endpoint, _ := urlGenerator(
    BASE_URL + "/ranking",
    []string{"ranking_type", "limit", "offset"},
    [][]string{{rankingType}, {strconv.Itoa(limit)}, {strconv.Itoa(offset)}},
    true,
  )

  // gets data from API and stores it in a struct
  var rankingData RawRanking
  data := requestHandler(token, endpoint)
  json.Unmarshal([]byte(data), &rankingData)

  // Adding all the animes in ranking list to a slice
  var animeRankingTitles []AnimeRankingTitle
  for _, element := range rankingData.Data {
    animeRankingTitles = append(
      animeRankingTitles,
      AnimeRankingTitle {
        Anime:   element.Anime, 
        RankNum: element.Ranking.Rank,
      },
    )
  }

  // Finally, create the AnimeRanking object
  animeRanking = AnimeRanking {
    Titles: animeRankingTitles,
    Paging: ListPaging {
      NextPage: rankingData.Paging.NextPage,
      PrevPage: rankingData.Paging.PrevPage,
    },
  }

  return animeRanking, nil
}