Bimba.git

ref: 47d75f47e5348dc513907e5d3972a5c2fd716029

app/src/main/java/ml/adamsprogs/bimba/api/Api.kt


  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
package ml.adamsprogs.bimba.api

import android.net.ConnectivityManager
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.withContext
import ml.adamsprogs.bimba.R
import java.io.IOException
import java.io.InputStream
import java.net.HttpURLConnection
import java.net.URL
import java.net.URLEncoder

data class Server(val host: String, val token: String, val feeds: String)

data class Result(val stream: InputStream?, val error: Error?)

data class Error(val statusCode: Int, val stringResource: Int)

@Suppress("BlockingMethodInNonBlockingContext")
suspend fun getFeeds(cm: ConnectivityManager, server: Server): Result {
	return rawRequest(URL("${hostWithScheme(server.host)}/api/"), server, cm)  // todo(error-handling) if is not a valid URL
}

suspend fun queryItems(cm: ConnectivityManager, server: Server, query: String, limit: Int? = null): Result {
	val params = mutableMapOf("q" to query)
	if (limit != null) {
		params["limit"] = limit.toString()
	}
	return request(server, "items", params, cm)
}

suspend fun locateItems(cm:ConnectivityManager, server: Server, plusCode: String): Result {
	return request(server, "items", mapOf("near" to plusCode), cm)
}

suspend fun getDepartures(cm: ConnectivityManager, server: Server, stop: String, line: String? = null): Result {
	val params = mutableMapOf("code" to stop)
	if (line != null) {
		params["line"] = line
	}
	return request(server, "departures", params, cm)
}

@Suppress("BlockingMethodInNonBlockingContext")
suspend fun rawRequest(url: URL, server: Server, cm: ConnectivityManager): Result {
	@Suppress("DEPRECATION")  // fix_later(API_29, API_23) https://developer.android.com/reference/android/net/ConnectivityManager#getActiveNetwork()
	if (cm.activeNetworkInfo == null) {
		// todo check false-positives
		return Result(null, Error(0, R.string.error_offline))
	}
	return withContext(Dispatchers.IO) {
		val c = (url.openConnection() as HttpURLConnection).apply {
			setRequestProperty("X-Bimba-Token", server.token)
		}
		try {
			if (c.responseCode == 200) {
				Result(c.inputStream, null)
			} else {
				val string = when (c.responseCode) {
					400 -> R.string.error_400
					401 -> R.string.error_401
					403 -> R.string.error_403
					404 -> R.string.error_404 // todo check if server returns 404
					429 -> R.string.error_429
					500 -> R.string.error_50x
					502 -> R.string.error_50x
					503 -> R.string.error_50x
					504 -> R.string.error_50x
					else -> R.string.error_unknown
				}
				Result(c.errorStream, Error(c.responseCode, string))
			}
		} catch (e: IOException) {
			// todo timeout, no Internet connection
			Result(null, Error(0, R.string.error_connecting))
		}
	}
}

@Suppress("BlockingMethodInNonBlockingContext")
suspend fun request(
	server: Server,
	resource: String,
	params: Map<String, String>,
	cm: ConnectivityManager
): Result {
	return withContext(Dispatchers.IO) {
		val url = URL(
			"${hostWithScheme(server.host)}/api/${server.feeds}/$resource${
				params.map {
					"${it.key}=${
						URLEncoder.encode(
							it.value,
							"utf-8"
						)
					}"
				}.joinToString("&", "?")
			}"
		)
		rawRequest(url, server, cm)
	}
}

fun hostWithScheme(host: String): String =
	if (host.startsWith("http://") or host.startsWith("https://")) {
		host
	} else {
		"https://$host"
	}