package org.jetbrains.dokka.resolvers

import org.jetbrains.dokka.links.DRI
import java.net.HttpURLConnection
import java.net.URL
import java.net.URLConnection

object ExternalLocationProvider { // TODO: Refactor this!!!
    private const val DOKKA_PARAM_PREFIX = "\$dokka."

    private val cache: MutableMap<URL, LocationInfo> = mutableMapOf()

    fun getLocation(dri: DRI, packageLocations: List<URL>): String {
        val toResolve: MutableList<URL> = mutableListOf()
        for(url in packageLocations){
            val info = cache[url]
            if(info == null) {
                toResolve.add(url)
            } else if(info.packages.contains(dri.packageName)) {
                return getLink(dri, info)
            }
        }
        // Not in cache, resolve packageLists
        while (toResolve.isNotEmpty()){
            val loc = toResolve.first().also { toResolve.remove(it) }
            val locationInfo = loadPackageList(loc)
            if(locationInfo.packages.contains(dri.packageName)) {
                return getLink(dri, locationInfo)
            }
        }
        return ""
    }

    private fun getLink(dri: DRI, locationInfo: LocationInfo): String = when(locationInfo.format) {
        "javadoc" ->  dri.toJavadocLocation(8)
        "kotlin-website-html", "html" -> locationInfo.locations[dri.packageName + "." + dri.classNames] ?: dri.toDokkaLocation(".html")
        "markdown" -> locationInfo.locations[dri.packageName + "." + dri.classNames] ?: dri.toDokkaLocation(".md")
        // TODO: rework this
        else -> throw RuntimeException("Unrecognized format")
    }


    private fun loadPackageList(url: URL): LocationInfo {
        val packageListStream = url.doOpenConnectionToReadContent().getInputStream()
        val (params, packages) =
            packageListStream
                .bufferedReader()
                .useLines { lines -> lines.partition { it.startsWith(DOKKA_PARAM_PREFIX) } }

        val paramsMap = params.asSequence()
            .map { it.removePrefix(DOKKA_PARAM_PREFIX).split(":", limit = 2) }
            .groupBy({ (key, _) -> key }, { (_, value) -> value })

        val format = paramsMap["format"]?.singleOrNull() ?: "javadoc"

        val locations = paramsMap["location"].orEmpty()
            .map { it.split("\u001f", limit = 2) }
            .map { (key, value) -> key to value }
            .toMap()

        val info = LocationInfo(format, packages.toSet(), locations)
        cache[url] = info
        return info
    }

    private fun URL.doOpenConnectionToReadContent(timeout: Int = 10000, redirectsAllowed: Int = 16): URLConnection {
        val connection = this.openConnection().apply {
            connectTimeout = timeout
            readTimeout = timeout
        }

        when (connection) {
            is HttpURLConnection -> {
                return when (connection.responseCode) {
                    in 200..299 -> {
                        connection
                    }
                    HttpURLConnection.HTTP_MOVED_PERM,
                    HttpURLConnection.HTTP_MOVED_TEMP,
                    HttpURLConnection.HTTP_SEE_OTHER -> {
                        if (redirectsAllowed > 0) {
                            val newUrl = connection.getHeaderField("Location")
                            URL(newUrl).doOpenConnectionToReadContent(timeout, redirectsAllowed - 1)
                        } else {
                            throw RuntimeException("Too many redirects")
                        }
                    }
                    else -> {
                        throw RuntimeException("Unhandled http code: ${connection.responseCode}")
                    }
                }
            }
            else -> return connection
        }
    }
    data class LocationInfo(val format: String, val packages: Set<String>, val locations: Map<String, String>)

}