KMM Networking layer

     

Clouds

This is the third in a multi-part series on Kotlin Multiplatform Mobile. The source code for the final project is available here. This entry will cover configuring Ktor and executing networking requests for the mobile iOS and Android apps. Unlike the tutorial, I’m not using the SpaceX API, which provides public access to information about SpaceX rocket launches. Instead I’m using the Army’s ODIN API for a reboot of my WEG iOS and Android applications. The ODIN API provides in-depth information about a wide array of military equipment.

With the Data Transfer Objects (DTOs) already created in the previous sections, we can jump straight into the API implementation below:

class Api {
    private val httpClient = HttpClient {
        install(JsonFeature) {
            val json = Json { ignoreUnknownKeys = true }
            serializer = KotlinxSerializer(json)
        }
    }

    suspend fun getEquipment(): SearchResults {
        val equipmentURl = API_URL + "?format=json&" +
                "action=query&" +
                "list=searchG2&" +
                "srimages=1&" +
                "srsearch=incategory:Land&" +
                "srlimit=20" +
                "&sroffset=0"
        return httpClient.get(equipmentURl)
    }

    companion object {
        const val BASE_URL = "https://odin.tradoc.army.mil"
        const val API_URL = "$BASE_URL/mediawiki/api.php"
    }
}

The API class contains a private Ktor HttpClient. Instead of accessing the client directly, our client apps will invoke networking functions and receive results from the API. If you’re wondering where all those url query parameters came from, it wasn’t easy. Since the API documentation for ODIN is rather obtuse, I found it easier to open the Chrome DevTools and just click around the ODIN website, making notes of the requests that my web client was making to the server. Then I’d input them into Postman and tweak the parameters to figure out exactly what worked and what didn’t.

The getEquipment() function invocation was briefly touched on in the KMM Database layer blog entry. Here it is again for reference:

class EquipmentSDK(databaseDriverFactory: DatabaseDriverFactory) {
    private val db = Database(databaseDriverFactory)
    private val api = Api()

    @Throws(Exception::class)
    suspend fun getEquipment(forceReload: Boolean): List<SearchResult>? {
        val cachedEquipment = db.getAllResults()
        return if (cachedEquipment.isNotEmpty() && !forceReload) {
            cachedEquipment
        } else {
            api.getEquipment().asList().also { results ->
                results?.let {
                    db.clearDatabase()
                    db.insertSearchResults(it)
                }
            }
        }
    }
}

This EquipmentSDK class creates a repository pattern that our client apps can use to abstract away caching. In this way they don’t need to worry about when to fetch fresh data from the server or to use cached data, the shared KMM SDK handles it all for them.

That concludes the API section of the KMM series. The next KMM entry will cover creating our Jetpack Compose views. Before we get to that though we’ll take a break from the unmitigated stream of technical discourse for a critique of stoicism.

Photo by Michael & Diane Weidner on Unsplash

comments powered by Disqus