Kine (Kotlin I/O Networking Extensions)

A simple and powerful HTTP networking library for Kotlin/Android.

Features

  • [x] Supports HTTP GET/POST/PUT/DELETE/HEAD/PATCH
  • [x] Supports synchronous and asynchronous requests
  • [x] Download file with progress
  • [x] Get Image as bitmap from url
  • [x] MultiPart Upload
  • [x] Response caching in disk
  • [x] OkHttp cache control full support
  • [x] Supports OkHttp with ability to write your own Httpclient
  • [x] Cancel any in-flight request
  • [x] Request timeout with retry policy
  • [x] App Wide Configuration using Kime with support for common Headers , base Url for all request
  • [x] Log manager with option to turn logging off for per request and App wide
  • [x] Supports response deserialization into POJO/POKO with Gson or Moshi
  • [x] Supports reactive programming via RxJava 2.x / RxJava 3.x
  • [x] Supports kotlin coroutines
  • [x] Supports live data
  • [x] Supports ability to handle any type of response with custom converter option

Upcoming Planned Features

  • Support for jackson parser
  • Bitmap caching and auto previous request cancellation for ImageLoading
  • New interceptor api for manipulating request pre execute

Installation

Library is available on jcenter

Gradle

repositories {
    jcenter()
}

dependencies {
    compile 'com.kine:kine:1.0.0.0' //for JVM
    compile 'com.kine:kine-json:2.8.6.0' //for json support(do not use this dependency in android project)
    compile 'com.kine:kine-android:1.0.0.0' //for Android
    compile 'com.kine:kine-rxjava2:2.2.19.0' //for RxJava2 support
    compile 'com.kine:kine-rxjava3:3.0.6.0' //for RxJava3 support
    compile 'com.kine:kine-coroutine:1.3.8.0' //for Coroutine support
    compile 'com.kine:kine-livedata:1.3.8.0' //for LiveData support
    compile 'com.kine:kine-gson:2.8.6.0' //for Gson support
    compile 'com.kine:kine-moshi:1.9.3.0' //for Moshi support
    compile 'com.kine:kine-okhttp:4.8.1.0' //for OkHttp support
    compile 'com.kine:kine-okhttplegacy:3.12.12.0' //for Legacy OkHttp support(API 9)
    compile 'com.kine:kine-imageloader:1.0.0.0' //for Imageloading support
}
Note: Kine will autodetect if android,okhttp,gson or moshi dependency are there and set appropriate client, converter or main thread callback you don't need to specify them,if both moshi and gson are there, kine will add moshi first(converter are always called by add order so if you have both dependency and trying to parse it using gson it won't work). To override this behaviour you can set them using Kine.builder() as shown below

Usage

Kine Configuration

Kine.Builder()
            .headers(hashMapOf()) // common headers for all app requests
            .client(OkHttpKineClient()) // common client to use for all requests
            .converter(GsonConverter()) // gson converter for parsing response
            .connectionChecker(SimpleConnectionChecker(context!!)) // a simple connection checker for no internet
            .baseUrl(ConfigUtils.dummyBaseUrl) // a base url for all requests
            .logLevel(LogLevel.ERROR) // logs to display according to level
            .disableAllLogs(true) // disable all logging for all requests
            .build()
Note: Options provided with individual KineRequest will always take priority over Kine global configuration except for headers , headers will always be added to individual request headers specified by the user

Kine requests can be made with KineRequest class or using one of the String or other extension methods.
If you specify a callback the call is asynchronous, if you don't its synchronous.
Exception will be thrown in synchronous (you need to catch them).
Exception will be delivered to onError callback in asynchronous except for user error like not specifying a converter and expecting a parsed response.

Get Json

"https://example/api/test".httpGet().responseAs(JSONObject::class.java,{ response->
               val response =  response.body
           }, { e ->
               e.printStackTrace()
           })
// for sync
val response =  "https://example/api/test".httpGet().responseAs(JSONObject::class.java)

Get String

"https://example/api/test".httpGet().responseAs(String::class.java,{ response->
               val response =  response.body
           }, { e ->
               e.printStackTrace()
           })
// for sync
val response =  "https://example/api/test".httpGet().responseAs(String::class.java)

Get JsonArray

"https://example/api/test".httpGet().responseAs(JSONArray::class.java,{ response->
               val list =  Gson().fromJsonArray<Post>(response.body.toString())
           }, { e ->
               e.printStackTrace()
           })
// for sync
val response =  "https://jsonplaceholder.typicode.com/posts".httpGet().responseAs(JSONArray::class.java)

Gson (requires kine-gson dependency) / Moshi (requires kine-moshi dependency)

"https://example/api/test".httpGet().responseAs(User::class.java,{ response->
               val user:User =  response.body
           }, { e ->
               e.printStackTrace()
           })
// for sync
val response =  "https://example/api/test".httpGet().responseAs(User::class.java)

RxJava2 (requires kine-rxjava2 dependency) / RxJava3 (requires kine-rxjava3 dependency)

Single

"https://example/api/test".httpGet().toSingle(clazz)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(object : SingleObserver<KineResponse<T>?> {
                override fun onSubscribe(d: Disposable) {
                }
                override fun onSuccess(response: KineResponse<T>) {
                    val response = response.body.toString()
                }
                override fun onError(e: Throwable) {
                    e.printStackTrace()
                }
            })

Flowable

"https://example/api/test".httpGet().toFlowable(clazz)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(object : FlowableSubscriber<KineResponse<T>> {

                override fun onError(e: Throwable) {
                    e.printStackTrace()
                }
                override fun onSubscribe(s: Subscription) {
                    s.request(1)
                }
                override fun onNext(response: KineResponse<T>) {
                    val response = response.body.toString()
                }
                override fun onComplete() {
                }
            })

Observable

 "https://example/api/test".httpGet().toObservable(clazz)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(object : Observer<KineResponse<T>?> {
                override fun onError(e: Throwable) {
                    e.printStackTrace()
                }
                override fun onNext(response: KineResponse<T>) {
                    val response = response.body.toString()
                }
                override fun onComplete() {
                }
                override fun onSubscribe(d: Disposable) {
                }
            })

Kotlin Coroutine (requires kine-coroutine dependency)

 GlobalScope.launch(Dispatchers.Main) {
            val response = "https://example/api/test".httpGet().responseAsCoroutine(clazz)
        }

Download File

"https://example/api/test/files/test10Mb.db".downloadTo(
            File(Environment.getExternalStorageDirectory(),"test.db"),{downloaded,total->
// note the progress listener is not called on main thread it is always called on background thread 
// for async request and calling thread on sync request
                activity?.runOnUiThread {
                    val progress = "progress ${((downloaded*100)/total)}"
                }
            }, { response ->
                val savedPath = response.body?.path
            }, { e ->
                e.printStackTrace()
            })

Image Loading (requires kine-imageloader dependency)

Load Bitmap From Url

  "https://example/api/test/files/abc.png".loadBitmapResponseFromUrl( { response ->
            imageView!!.setImageBitmap(response.body)
        }, { e -> e.printStackTrace() })

Load Image from Url to ImageView

  imageView.loadImage("https://example/api/test/files/abc.png",placeHolderResId)

Cancelling request

Application

 Kine.cancelAllRequests() // cancels all request with clients set with Kine
 Kine.cancelAllRequests(tag) // cancels all request with that tag with clients 
 // set with Kine(note passing null will cancel all request)

Individual Request

  val client = OkHttpKineClient()
  "url".httpGet().client(client).responseAs(User::class.java)
  client.cancelAllRequests() // behaves same as above Kine cancel methods

MultiPart Image Upload

   KineRequest.upload("").addMultiPartParam("name","test",null).addMultiPartFileParam("tset",
   File("sdcard/test.apk"),null).setUploadListener { bytesWritten, total ->  }.responseAs(User::class.java,{ response->
   val response = response.body
     },{e->
           e.printStackTrace()
     })

Requirements

For Android

OkHttp 4x

  • Min SDK 21+(for OkHttpClient you can write your own client for supporting Api Version below it)
  • Java 8+

OkHttp 3x (legacy)

  • Min SDK 9+(for OkHttpClient you can write your own client for supporting Api Version below it)
  • Java 7+

For Kotlin/Java

  • Kotlin 1.4/Java 8+

GitHub