4 // Given a track, attempts to fetch the lyrics from lyricswiki
7 // URL of the API endpoint to use
8 private let apiURL = "https://lyrics.wikia.com/api.php?action=lyrics&func=getSong&fmt=realjson"
10 // Method used to call the API
11 private let apiMethod = "GET"
13 // Regular expxression used to find the lyrics in the lyricswiki HTML
14 private let lyricsMatcher = "class='lyricbox'>(.+)<div"
16 // The track we'll be looking for
17 private let track: Track
19 // Fetches the lyrics and returns if found
24 // Encode the track artist and name and finish building the API call URL
26 if let artist = track.artist.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) {
27 if let name: String = track.name.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) {
28 if let url = URL(string: "\(apiURL)&artist=\(artist)&song=\(name)") {
30 // We'll lock until the async call is finished
32 var requestFinished = false
33 let asyncLock = NSCondition()
36 // Call the API and unlock when you're done
38 fetchLyricsFromAPI(withURL: url, completionHandler: {lyricsResult -> Void in
40 requestFinished = true
44 while !requestFinished {
55 // Initializes with a track
56 init(withTrack targetTrack: Track) {
61 // Fetch the lyrics URL from the API, triggers the request to fetch the
63 private func fetchLyricsFromAPI(withURL url: URL, completionHandler: @escaping (String?) -> Void) {
65 var apiRequest = URLRequest(url: url)
66 apiRequest.httpMethod = "GET"
68 let task = URLSession.shared.dataTask(with: apiRequest, completionHandler: {data, _, _ -> Void in
70 // If the response is parseable JSON, and has a url, we'll look for
71 // the lyrics in there
74 if let jsonResponse = try? JSONSerialization.jsonObject(with: data) {
75 if let jsonResponse = jsonResponse as? [String: Any] {
76 if let lyricsUrlString = jsonResponse["url"] as? String {
77 if let lyricsUrl = URL(string: lyricsUrlString) {
79 // At this point we have a valid wiki url
80 self.fetchLyricsFromPage(withURL: lyricsUrl, completionHandler: completionHandler)
88 completionHandler(nil)
93 // Fetch the lyrics from the page and send it to the parser
94 private func fetchLyricsFromPage(withURL url: URL, completionHandler: @escaping (String?) -> Void) {
96 var pageRequest = URLRequest(url: url)
97 pageRequest.httpMethod = "GET"
99 let task = URLSession.shared.dataTask(with: pageRequest, completionHandler: {data, _, _ -> Void in
101 // If the response is parseable JSON, and has a url, we'll look for
102 // the lyrics in there
105 if let htmlBody = String(data: data, encoding: String.Encoding.utf8) {
106 self.parseHtmlBody(htmlBody, completionHandler: completionHandler)
111 completionHandler(nil)
116 // Parses the wiki to find the lyrics, decodes the lyrics object
117 private func parseHtmlBody(_ body: String, completionHandler: @escaping (String?) -> Void) {
119 // Look for the lyrics lightbox
121 if let regex = try? NSRegularExpression(pattern: lyricsMatcher) {
122 let matches = regex.matches(in: body, range: NSRange(location: 0, length: body.count))
124 for match in matches {
126 let nsBody = body as NSString
127 let range = match.range(at: 1)
128 let encodedLyrics = nsBody.substring(with: range)
130 let decodedLyrics = decodeLyrics(encodedLyrics)
132 completionHandler(decodedLyrics)
137 completionHandler(nil)
140 // Escapes the HTML entities
141 private func decodeLyrics(_ lyrics: String) -> String {
143 let unescapedLyrics = lyrics.htmlUnescape()
144 return unescapedLyrics.replacingOccurrences(of: "<br />", with: "\n")