2020-06-22 17:02:36 +00:00
|
|
|
/*
|
|
|
|
Mangadex@Home
|
|
|
|
Copyright (c) 2020, MangaDex Network
|
|
|
|
This file is part of MangaDex@Home.
|
|
|
|
|
|
|
|
MangaDex@Home is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
MangaDex@Home is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this MangaDex@Home. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2020-06-19 20:16:24 +00:00
|
|
|
package mdnet.base
|
|
|
|
|
2020-06-20 16:32:15 +00:00
|
|
|
import ch.qos.logback.classic.LoggerContext
|
2020-06-19 20:16:24 +00:00
|
|
|
import com.fasterxml.jackson.core.JsonProcessingException
|
|
|
|
import com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException
|
|
|
|
import com.fasterxml.jackson.module.kotlin.readValue
|
|
|
|
import java.io.FileReader
|
|
|
|
import java.io.FileWriter
|
|
|
|
import java.io.IOException
|
|
|
|
import java.util.regex.Pattern
|
|
|
|
import kotlin.system.exitProcess
|
2020-07-02 16:06:32 +00:00
|
|
|
import mdnet.base.Constants.JACKSON
|
|
|
|
import mdnet.base.settings.ClientSettings
|
|
|
|
import org.slf4j.LoggerFactory
|
2020-06-19 20:16:24 +00:00
|
|
|
|
|
|
|
object Main {
|
|
|
|
private val LOGGER = LoggerFactory.getLogger(Main::class.java)
|
|
|
|
|
|
|
|
@JvmStatic
|
|
|
|
fun main(args: Array<String>) {
|
|
|
|
println(
|
2020-06-22 17:02:36 +00:00
|
|
|
"Mangadex@Home Client Version ${Constants.CLIENT_VERSION} (Build ${Constants.CLIENT_BUILD}) initializing"
|
2020-06-19 20:16:24 +00:00
|
|
|
)
|
2020-06-22 17:02:36 +00:00
|
|
|
println()
|
2020-06-19 20:16:24 +00:00
|
|
|
println("Copyright (c) 2020, MangaDex Network")
|
2020-06-22 17:02:36 +00:00
|
|
|
println("""
|
|
|
|
Mangadex@Home is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
2020-06-22 17:09:11 +00:00
|
|
|
|
2020-06-22 17:02:36 +00:00
|
|
|
Mangadex@Home is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
2020-06-22 17:09:11 +00:00
|
|
|
|
2020-06-22 17:02:36 +00:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with Mangadex@Home. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
""".trimIndent())
|
2020-06-19 20:16:24 +00:00
|
|
|
|
|
|
|
var file = "settings.json"
|
|
|
|
if (args.size == 1) {
|
|
|
|
file = args[0]
|
|
|
|
} else if (args.isNotEmpty()) {
|
|
|
|
dieWithError("Expected one argument: path to config file, or nothing")
|
|
|
|
}
|
|
|
|
|
2020-06-21 19:49:10 +00:00
|
|
|
val settings = try {
|
2020-06-19 20:16:24 +00:00
|
|
|
JACKSON.readValue<ClientSettings>(FileReader(file))
|
|
|
|
} catch (e: UnrecognizedPropertyException) {
|
|
|
|
dieWithError("'${e.propertyName}' is not a valid setting")
|
|
|
|
} catch (e: JsonProcessingException) {
|
|
|
|
dieWithError(e)
|
|
|
|
} catch (ignored: IOException) {
|
|
|
|
ClientSettings().also {
|
|
|
|
LOGGER.warn("Settings file {} not found, generating file", file)
|
|
|
|
try {
|
|
|
|
FileWriter(file).use { writer -> JACKSON.writeValue(writer, it) }
|
|
|
|
} catch (e: IOException) {
|
|
|
|
dieWithError(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.apply(::validateSettings)
|
|
|
|
|
|
|
|
if (LOGGER.isInfoEnabled) {
|
|
|
|
LOGGER.info("Client settings loaded: {}", settings)
|
|
|
|
}
|
|
|
|
val client = MangaDexClient(settings)
|
|
|
|
Runtime.getRuntime().addShutdownHook(Thread { client.shutdown() })
|
|
|
|
client.runLoop()
|
|
|
|
}
|
|
|
|
|
|
|
|
fun dieWithError(e: Throwable): Nothing {
|
|
|
|
if (LOGGER.isErrorEnabled) {
|
|
|
|
LOGGER.error("Critical Error", e)
|
|
|
|
}
|
2020-06-20 16:32:15 +00:00
|
|
|
(LoggerFactory.getILoggerFactory() as LoggerContext).stop()
|
2020-06-19 20:16:24 +00:00
|
|
|
exitProcess(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
fun dieWithError(error: String): Nothing {
|
|
|
|
if (LOGGER.isErrorEnabled) {
|
|
|
|
LOGGER.error("Critical Error: {}", error)
|
|
|
|
}
|
2020-06-20 16:32:15 +00:00
|
|
|
(LoggerFactory.getILoggerFactory() as LoggerContext).stop()
|
2020-06-19 20:16:24 +00:00
|
|
|
exitProcess(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun validateSettings(settings: ClientSettings) {
|
|
|
|
if (!isSecretValid(settings.clientSecret)) dieWithError("Config Error: API Secret is invalid, must be 52 alphanumeric characters")
|
|
|
|
if (settings.clientPort == 0) {
|
|
|
|
dieWithError("Config Error: Invalid port number")
|
|
|
|
}
|
|
|
|
if (settings.maxCacheSizeInMebibytes < 1024) {
|
|
|
|
dieWithError("Config Error: Invalid max cache size, must be >= 1024 MiB (1GiB)")
|
|
|
|
}
|
|
|
|
if (settings.threads < 4) {
|
|
|
|
dieWithError("Config Error: Invalid number of threads, must be >= 4")
|
|
|
|
}
|
|
|
|
if (settings.maxMebibytesPerHour < 0) {
|
|
|
|
dieWithError("Config Error: Max bandwidth must be >= 0")
|
|
|
|
}
|
|
|
|
if (settings.maxKilobitsPerSecond < 0) {
|
|
|
|
dieWithError("Config Error: Max burst rate must be >= 0")
|
|
|
|
}
|
2020-06-22 16:26:14 +00:00
|
|
|
if (settings.gracefulShutdownWaitSeconds < 15) {
|
|
|
|
dieWithError("Config Error: Graceful shutdown wait be >= 15")
|
|
|
|
}
|
2020-06-19 20:16:24 +00:00
|
|
|
if (settings.webSettings != null) {
|
|
|
|
if (settings.webSettings.uiPort == 0) {
|
|
|
|
dieWithError("Config Error: Invalid UI port number")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private const val CLIENT_KEY_LENGTH = 52
|
|
|
|
private fun isSecretValid(clientSecret: String): Boolean {
|
|
|
|
return Pattern.matches("^[a-zA-Z0-9]{$CLIENT_KEY_LENGTH}$", clientSecret)
|
|
|
|
}
|
|
|
|
}
|