MusicKit and App Clips

Sponsored
RevenueCat logo
Relax, you can roll back your mobile release

No one is immune from shipping critical bugs to production, but Runway helps you limit the amount of havoc that can cause.

Last week I published an article about how to create an App Clip for your app where I mentioned that App Clips have a set of limitations such as the amount of memory they can use and the amount of permissions they can request.

While I was working on the NowPlaying App Clip, I faced one of these limitations and I thought it would be interesting to share my experience with you and how I got around it.

The problem

The NowPlaying App Clip, which is still in the early stages of development, shows information about a given song using the same layout and data as the main app.

To find the song from an ID, such as an ISRC (International Standard Recording Code), NowPlaying uses MusicKit, which is an Apple framework to interact with Apple Music. Unfortunately for us, we couldn’t reuse our logic from the main app as is, as MusicKit does not work in App Clips.

Before going into the details about why it’s not possible to use MusicKit in an App Clip, let’s set some context about how I set up the App Clip and what I tried to get it working.

Setting up MusicKit for the App Clip

The first thing you need to do to be able to use MusicKit in your app is to create an app identifier with the same bundle identifier as your App Clip in your Developer Account and enable the MusicKit App Service:

Start by creating a new media id and enabling MusicKit

Adding keys to the Info.plist

As MusicKit requires authorization from the user and will present an alert, you need to add the ‘Privacy - Media Library Usage Description’ (NSAppleMusicUsageDescription) key with a description of why you need access to the user’s media library to your target’s Info.plist file.

Info.plist
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
	<key>NSAppleMusicUsageDescription</key>
	<string>Need access to your music library</string>
</dict>
</plist>

Requesting authorization

Now that you have added the services to the App Clip identifier in App Store Connect and the NSAppleMusicUsageDescription key to the Info.plist file, you can go ahead and request authorization from the user in your App Clip’s code:

ContentView.swift
import SwiftUI
import MusicKit

struct ContentView: View {
    var body: some View {
        VStack {
            Image(systemName: "globe")
                .imageScale(.large)
                .foregroundStyle(.tint)
            Text("Hello, world!")
        }
        .padding()
        .task {
            guard await isAppAuthorised() else {
                return
            }
        }
    }

    private func isAppAuthorised() async -> Bool {
        guard MusicAuthorization.currentStatus != .authorized else {
            return true
        }

        let response = await MusicAuthorization.request()
        return response == .authorized
    }
}

It doesn’t work 😭

If you run your App Clip target now, you will see that, regardless of what you try, the permission status will always be .denied and the request method will never present the authorization alert to the user.

I have not been able to find any documentation about this but, after asking a few people about this issue, I found out that it’s not possible to grant MusicKit permissions in an App Clip.

What I tried

I was keen on getting the App Clip working so I decided to persevere and try a few different things to see if I could work around the permissions issue.

❌ Catalogue requests

As the App Clip only needs to find a song by its ISRC (International Standard Recording Code) directly from the Apple Music catalogue without the need to access the user’s media library, I thought that I could potentially use this part of the MusicKit framework without requesting authorization:

ContentView.swift
import SwiftUI
import MusicKit

struct ContentView: View {
    var body: some View {
        VStack {
            Image(systemName: "globe")
                .imageScale(.large)
                .foregroundStyle(.tint)
            Text("Hello, world!")
        }
        .padding()
        .task {
            do {
                let isrc = "QM72823682"
                let songRequest = MusicCatalogResourceRequest<Song>(matching: \.isrc, equalTo: isrc)
                let songResponse = try await songRequest.response()
                print(songResponse.items.first)
            } catch {
                print(error.localizedDescription)
            }
        }
    }
}

Despite my high hopes, this didn’t work either. As soon as I ran the App Clip, I got the following error:

Permission denied

Failed to fetch current country code because the music authorization status is set to .denied. This is recoverable by guiding your user to the privacy settings, so they can grant your app access to Apple Music.

❌ Making a request to the Apple Music API using MusicKit

I wasn’t ready to give up at this point and I decided to request the information from the Apple Music API directly. If you have ever tried to use this API, you will know that authenticating with it manually can certainly be cumbersome. Thankfully, MusicKit offers a way of making authenticated network requests to the Apple Music API with very little effort:

ContentView.swift
import SwiftUI
import MusicKit

struct ContentView: View {
    var body: some View {
        VStack {
            Image(systemName: "globe")
                .imageScale(.large)
                .foregroundStyle(.tint)
            Text("Hello, world!")
        }
        .padding()
        .task {
            do {
                let isrc = "QM72823682"
                let urlString = "https: //api.music.apple.com/v1/catalog/us/songs?filter[isrc]=\(isrc)"
                let url = URL(string: urlString)!
                let songRequest = MusicDataRequest(urlRequest: .init(url: url))
                let response = try await songRequest.response()
                let items = try JSONDecoder().decode (MusicItemCollection<Song>.self, from: response.data)
                print(items.first)
            } catch {
                print(error.localizedDescription)
            }
        }
    }
}

Unfortunately, this approach didn’t work either. Despite this request only needing a developer token to work, I still got the same permission error as before:

Permission denied

Failed to request tokens because the music authorization status is set to .denied. This is recoverable by guiding your user to the privacy settings, so they can grant your app access to Apple Music.

✅ Making a request to the Apple Music API using URLSession

Finally, I opted to make the request to the Apple Music API myself using URLSession and manual authentication. For this specific request, as it doesn’t need any user permissions, I only had to get a developer token and add it to the request’s headers.

MusicKit does most of the heavy lifting of creating a developer token for you and allows you to retrieve it with a single line of code:

let developerToken = try? await MusicDataRequest.tokenProvider.developerToken(options: .ignoreCache)

However, and as you might be suspecting by now, this doesn’t work in an App Clip either. Unfortunately, when you try to get the developer token, you get the following error saying permission is denied 😭:

Failed retrieving developer token: Error Domain=ICError Code=-7010 "Failed to get listener endpoint for cloud service status monitor." UserInfo={NSDebugDescription=Failed to get listener endpoint for cloud service status monitor., NSUnderlyingError=0x600000c754d0 {Error Domain=NSCocoaErrorDomain Code=4097 "connection to service named com.apple.itunescloudd.xpc" UserInfo={NSDebugDescription=connection to service named com.apple.itunescloudd.xpc}}}. Throwing .permissionDenied.

Generating a developer token

With my options very limited at this point, and as I wasn’t quite ready to give up yet, I decided to go ahead and generate a developer token manually.

Contrary to what you might think, this process is fairly complex and, over the next few sections, I will do my best to walk you through the process.

Creating a Media ID

The first thing you need to do is to create a new media identifier in your Developer Account by going to the ‘Identifiers’ section, clicking on the + button, selecting ‘Media IDs’ from the list and clicking on ‘Continue:

Create a new media ID by going to identifiers and pressing the + button.

Next, you need to select ‘MusicKit’ from the list, give the Media ID an identifier and a description and click on ‘Continue’:

Give the ID a name and a description

Finally, register the new Media ID:

Confirm the choice by clicking on the Register button

Creating a new key

Once you have done this, head over to the ‘Keys’ section in the Apple Developer Portal and click on the ’+’ button:

Create a new key from the Keys section by tapping on the + button

Give your key a name and enable ‘Media Services’ from the list:

Configure the key with a name and with the media services option checked

Tap on ‘Configure’ next to the ‘Media Services’ row, select the Media ID you created earlier and click ‘Save’:

Configure the media services option by assigning it to a media id

Finally, click on ‘Continue’ and then finish the process by downloading the key. You must keep this key somewhere safe as you won’t be able to download it again.

You will also need to get the key id from the key’s details page:

retrieve the key id from the key's details page

Creating a JWT token

Now that you have a media id and a key, you need to turn them into a JWT token that you can use to authenticate with the Apple Music API.

I decided to do this using Swift with the SwiftJWT library, a Swift package that helps you create and verify JWT tokens.

You can find more information about the specifics of the JWT token’s contents in Apple’s documentation, but to make a compatible JWT token from the id and key you created earlier, you can use the following code:

import SwiftJWT

struct JWTClaims: Claims {
    let iss: String
    let iat: Date?
    let exp: Date?
}

func generateToken(privateKey: String) throws -> String {
    let header = Header(kid: "key-id")
    let claims = JWTClaims(iss: "asc-team", iat: Date(), exp: Date() + 60 * 60 * 12)
    var jwt = JWT(header: header, claims: claims)
    let keyData = key.data(using: .utf8)!

    return try jwt.sign(using: .es256(privateKey: privateKey.data(using: .utf8)!))
}

🌟 Full credit for the code snippet above goes to this awesome YouTube video by Get Swifty, who does an amazing job at explaining how to make a JWT token for the Apple Music API.

Note that the privateKey parameter in the method above is the contents of the key you downloaded earlier. You can view and copy the key’s contents by opening it in a text editor of your choice.

I would recommend keeping this key safe and making the JWT token short-lived to increase security. In the example above, I have set the token to expire after 12 hours.

You can always verify that the token is valid by extracting the claims from the raw JWT string before making a request using SwiftJWT:

import SwiftJWT

func isTokenValid(tokenString: String) -> Bool {
    guard let expiry = try? JWT<JWTClaims>(jwtString: token).claims.exp else {
        return false
    }

    return expiry > Date()
}

Decorating the request

Now that you have a JWT token, you are ready to make a request to the Apple Music API and use the response to display information in your App Clip.

All you have to do now is write some URLSession code that makes a request to the Apple Music API and decorate it with the JWT token you created earlier:

ContentView.swift
import SwiftUI
import MusicKit

struct ContentView: View {
    var body: some View {
        VStack {
            Image(systemName: "globe")
                .imageScale(.large)
                .foregroundStyle(.tint)
            Text("Hello, world!")
        }
        .padding()
        .task {
            do {
                let isrc = "QM72823682"
                let urlString = "https://api.music.apple.com/v1/catalog/us/songs?filter[isrc]=QM7282368269"
                let url = URL(string: urlString)!
                let session = URLSession.shared
                let (data, _) = try await session.data(for: appleMusicRequest(for: url))
                let decoder = JSONDecoder()
                // A struct conforming to `Decodable`
                let songResponse = try decoder.decode(SongResponse.self, from: data)
                print(items.first)
            } catch {
                print(error.localizedDescription)
                return nil
            }
        }
    }

    private func appleMusicRequest(for url: URL) -> URLRequest {
        let keyToken = "🙈"
        var request = URLRequest(url: url)
        request.httpMethod = "GET"
        request.setValue("application/json", forHTTPHeaderField: "Content-Type")
        request.setValue("Bearer \(keyToken)", forHTTPHeaderField: "Authorization")

        return request
    }
}