Initial commit
This commit is contained in:
parent
093553eecc
commit
82b3ee21e4
9
.gitignore
vendored
Normal file
9
.gitignore
vendored
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
.gradle
|
||||||
|
.swp
|
||||||
|
server/build
|
||||||
|
server/.gradle
|
||||||
|
server/userDatabase
|
||||||
|
server/messageHistory
|
||||||
|
client-cli/build
|
||||||
|
client-cli/.gradle
|
||||||
|
client-cli/.chookpen.profile
|
3
client-cli/README.md
Normal file
3
client-cli/README.md
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# Chookpen Client (CLI)
|
||||||
|
|
||||||
|
This is the reference client for Chookpen. It is quite a basic client. This is a good starting point if you need an example, or are building a GUI Kotlin client.
|
35
client-cli/build.gradle.kts
Normal file
35
client-cli/build.gradle.kts
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm") version "2.0.0"
|
||||||
|
application
|
||||||
|
distribution
|
||||||
|
}
|
||||||
|
|
||||||
|
application {
|
||||||
|
mainClass.set("xyz.maxwellj.chookpen.client.MainKt")
|
||||||
|
layout.buildDirectory.dir("distributions/")
|
||||||
|
}
|
||||||
|
|
||||||
|
group = "xyz.maxwellj.chookpen.client"
|
||||||
|
version = "0.0.2"
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType<Jar> {
|
||||||
|
manifest {
|
||||||
|
attributes["Main-Class"] = "xyz.maxwellj.chookpen.client.MainKt"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
testImplementation(kotlin("test"))
|
||||||
|
implementation("com.github.kittinunf.fuel:fuel:3.0.0-alpha03")
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.test {
|
||||||
|
useJUnitPlatform()
|
||||||
|
}
|
||||||
|
kotlin {
|
||||||
|
jvmToolchain(17)
|
||||||
|
}
|
5
client-cli/settings.gradle.kts
Normal file
5
client-cli/settings.gradle.kts
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
plugins {
|
||||||
|
id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
|
||||||
|
}
|
||||||
|
rootProject.name = "chookpen.client"
|
||||||
|
|
105
client-cli/src/main/kotlin/Main.kt
Normal file
105
client-cli/src/main/kotlin/Main.kt
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
package xyz.maxwellj.chookpen.client
|
||||||
|
|
||||||
|
import fuel.Fuel
|
||||||
|
import fuel.get
|
||||||
|
import java.io.File
|
||||||
|
|
||||||
|
import kotlin.system.exitProcess
|
||||||
|
|
||||||
|
import java.math.BigInteger
|
||||||
|
import java.security.MessageDigest
|
||||||
|
|
||||||
|
fun md5(input:String): String {
|
||||||
|
val md = MessageDigest.getInstance("MD5")
|
||||||
|
return BigInteger(1, md.digest(input.toByteArray())).toString(16).padStart(32, '0')
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun main(args: Array<String>) {
|
||||||
|
// Variables
|
||||||
|
var name = ""
|
||||||
|
var server = ""
|
||||||
|
var port = ""
|
||||||
|
var hasHTTPS = ""
|
||||||
|
var password = ""
|
||||||
|
var configFile = File(".chookpen.profile")
|
||||||
|
|
||||||
|
if (!configFile.exists()) {
|
||||||
|
println("You don't have a Chookpen profile set up yet. If you've got a Chookpen profile, type 'l' to login and press enter. Otherwise, just press enter.")
|
||||||
|
val userInput = readln()
|
||||||
|
if (userInput == "l") {
|
||||||
|
println("Username:")
|
||||||
|
name = readln()
|
||||||
|
println("Password:")
|
||||||
|
password = readln()
|
||||||
|
val request = Fuel.get("http://localhost:8000/api/createAccount/username:{$name}token:{${md5(password)}}").body.string()
|
||||||
|
if (request == "Invalid token! Please try putting in your password right") {
|
||||||
|
println("Invalid password! Please rerun the program and put in the right password")
|
||||||
|
exitProcess(1)
|
||||||
|
} else {
|
||||||
|
configFile.createNewFile()
|
||||||
|
configFile.writeText("$name:$password:localhost:8000:0")
|
||||||
|
println("Logged in! Run the command again to start talking!")
|
||||||
|
exitProcess(0)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println("Choose a username:")
|
||||||
|
val newName = readln()
|
||||||
|
if (newName == "") {
|
||||||
|
println("Please choose a username! Rerun the program and try again")
|
||||||
|
exitProcess(1)
|
||||||
|
println("Choose a password:")
|
||||||
|
val newPassword = readln()
|
||||||
|
if (newPassword == "") {
|
||||||
|
println("Please choose a password! Rerun the program and try again")
|
||||||
|
exitProcess(1)
|
||||||
|
}
|
||||||
|
val request = Fuel.get("http://localhost:8000/api/createAccount/username:{$newName}token:{${md5(newPassword)}}").body.string()
|
||||||
|
if (request == "That username already exists on the server! Please choose a different username") {
|
||||||
|
println("That username already exists on the server! Rerun the program and choose a different username")
|
||||||
|
exitProcess(1)
|
||||||
|
} else {
|
||||||
|
configFile.createNewFile()
|
||||||
|
configFile.writeText("$newName:$newPassword:localhost:8000:0")
|
||||||
|
println("Account created!")
|
||||||
|
exitProcess(0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var configStage = 0
|
||||||
|
for (char in configFile.readText()) {
|
||||||
|
if (char == ':') {configStage ++}
|
||||||
|
if (configStage == 0) {
|
||||||
|
name += char
|
||||||
|
} else if (configStage == 1) {
|
||||||
|
password += char
|
||||||
|
} else if (configStage == 2) {
|
||||||
|
server += char
|
||||||
|
} else if (configStage == 3) {
|
||||||
|
port += char
|
||||||
|
} else if (configStage == 4) {
|
||||||
|
hasHTTPS += char
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
server = server.replace(":", "")
|
||||||
|
port = port.replace(":", "")
|
||||||
|
hasHTTPS = hasHTTPS.replace(":", "")
|
||||||
|
password = password.replace(":", "")
|
||||||
|
|
||||||
|
// Actual code
|
||||||
|
println("Chookpen Client initialised!")
|
||||||
|
println("Hello $name@$server")
|
||||||
|
val protocol = "http"
|
||||||
|
if (hasHTTPS == "1") {
|
||||||
|
val protocol = "https"
|
||||||
|
}
|
||||||
|
if (args.count() == 0) {
|
||||||
|
println(Fuel.get("$protocol://$server:$port/api/username:{$name}token:{${md5(password)}}").body.string())
|
||||||
|
} else {
|
||||||
|
val message = args[0]
|
||||||
|
println(Fuel.get("$protocol://$server:$port/api/username:{$name}token:{${md5(password)}}message:{$message}").body.string())
|
||||||
|
}
|
||||||
|
exitProcess(0)
|
||||||
|
}
|
5
server/README.md
Normal file
5
server/README.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# Chookpen Server
|
||||||
|
|
||||||
|
This is the official Chookpen server implementation. It provides a backend and exposes an API.
|
||||||
|
|
||||||
|
|
34
server/build.gradle.kts
Normal file
34
server/build.gradle.kts
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm") version "2.0.0"
|
||||||
|
application
|
||||||
|
distribution
|
||||||
|
}
|
||||||
|
|
||||||
|
application {
|
||||||
|
mainClass.set("xyz.maxwellj.chookpen.MainKt")
|
||||||
|
layout.buildDirectory.dir("distributions/")
|
||||||
|
}
|
||||||
|
|
||||||
|
group = "xyz.maxwellj.chookpen"
|
||||||
|
version = "0.0.1"
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType<Jar> {
|
||||||
|
manifest {
|
||||||
|
attributes["Main-Class"] = "xyz.maxwellj.chookpen.MainKt"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
testImplementation(kotlin("test"))
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.test {
|
||||||
|
useJUnitPlatform()
|
||||||
|
}
|
||||||
|
kotlin {
|
||||||
|
jvmToolchain(17)
|
||||||
|
}
|
7
server/chatHistory
Normal file
7
server/chatHistory
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
max: dingus
|
||||||
|
max: dongus
|
||||||
|
henry: fndskjnfjkdsnjfdsfdsnfdsnds
|
||||||
|
henry: fnjdsknfjkdnsjnfjdsnjfndsinifewnfewnofnf
|
||||||
|
max: nfljdsjfdsnfdsnkjds
|
||||||
|
max: dingus
|
||||||
|
max: dongus
|
5
server/settings.gradle.kts
Normal file
5
server/settings.gradle.kts
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
plugins {
|
||||||
|
id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
|
||||||
|
}
|
||||||
|
rootProject.name = "chookpen"
|
||||||
|
|
219
server/src/main/kotlin/Main.kt
Normal file
219
server/src/main/kotlin/Main.kt
Normal file
|
@ -0,0 +1,219 @@
|
||||||
|
package xyz.maxwellj.chookpen
|
||||||
|
|
||||||
|
import com.sun.net.httpserver.HttpExchange
|
||||||
|
import com.sun.net.httpserver.HttpHandler
|
||||||
|
import com.sun.net.httpserver.HttpServer
|
||||||
|
import java.net.InetSocketAddress
|
||||||
|
|
||||||
|
import java.io.File
|
||||||
|
import java.io.BufferedReader
|
||||||
|
|
||||||
|
fun main(args: Array<String>) {
|
||||||
|
// Crete a server and start listening for arguments
|
||||||
|
val server = HttpServer.create(InetSocketAddress(8000), 0)
|
||||||
|
server.createContext("/test", ServerTester())
|
||||||
|
server.createContext("/api", ApiHandler())
|
||||||
|
server.createContext("/api/createAccount", CreateAccount())
|
||||||
|
server.executor = null
|
||||||
|
server.start()
|
||||||
|
println("Server has started on port 8000")
|
||||||
|
}
|
||||||
|
|
||||||
|
class ServerTester : HttpHandler {
|
||||||
|
override fun handle(t: HttpExchange) {
|
||||||
|
val response = "Testing, testing, 1, 2, 3... Somehow this is going to make it to production one day"
|
||||||
|
t.sendResponseHeaders(200, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
println("Test request recieved!")
|
||||||
|
os.close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ApiHandler : HttpHandler {
|
||||||
|
override fun handle(t: HttpExchange) {
|
||||||
|
println("Request recieved...")
|
||||||
|
// Get the data
|
||||||
|
val path = t.requestURI.path
|
||||||
|
val inputData = path.substringAfter("/api/")
|
||||||
|
println("API request recieved: $inputData")
|
||||||
|
|
||||||
|
// Parse data sent to the server by client
|
||||||
|
var username = ""
|
||||||
|
var token = ""
|
||||||
|
var message = ""
|
||||||
|
var dataType = ""
|
||||||
|
var isParsingData = 0
|
||||||
|
for (char in inputData) {
|
||||||
|
val character = char
|
||||||
|
if (character == ':') {
|
||||||
|
isParsingData = 1
|
||||||
|
} else if (isParsingData == 1) {
|
||||||
|
if (character == '}') {
|
||||||
|
isParsingData = 0
|
||||||
|
dataType = ""
|
||||||
|
} else if (character != '{') {
|
||||||
|
if (dataType == "username") {
|
||||||
|
username += character
|
||||||
|
} else if (dataType == "token") {
|
||||||
|
token += character
|
||||||
|
} else if (dataType == "message") {
|
||||||
|
message += character
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
dataType += character
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val userDatabaseParser = BufferedReader(File("userDatabase").reader())
|
||||||
|
var lineNumber = 1
|
||||||
|
var userLine = ""
|
||||||
|
|
||||||
|
// Search the user database to find required information about the user
|
||||||
|
userDatabaseParser.forEachLine { line ->
|
||||||
|
if (line.contains(username)) {
|
||||||
|
userLine = line
|
||||||
|
}
|
||||||
|
lineNumber++
|
||||||
|
}
|
||||||
|
userDatabaseParser.close()
|
||||||
|
|
||||||
|
if (userLine == "") {
|
||||||
|
val response = "That account does not exist on this server."
|
||||||
|
t.sendResponseHeaders(403, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
os.close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var usernameInDatabase = ""
|
||||||
|
var tokenInDatabase = ""
|
||||||
|
var currentStage = 0
|
||||||
|
for (char in userLine) {
|
||||||
|
if (char == ':') {
|
||||||
|
currentStage ++
|
||||||
|
}
|
||||||
|
if (currentStage == 0) {
|
||||||
|
usernameInDatabase += char
|
||||||
|
} else if (currentStage == 1) {
|
||||||
|
tokenInDatabase += char
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tokenInDatabase = tokenInDatabase.replace(":", "")
|
||||||
|
if (token != tokenInDatabase) {
|
||||||
|
val response = "Invalid token! Please try putting in your password right"
|
||||||
|
t.sendResponseHeaders(403, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
os.close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Make the message to respond to the client
|
||||||
|
val chatHistoryView = File("chatHistory")
|
||||||
|
var fullMessage = ""
|
||||||
|
if (message != "") {
|
||||||
|
fullMessage = "${chatHistoryView.readText()}$username: $message"
|
||||||
|
// Add the client's message to the chat history
|
||||||
|
val chatHistory = File("chatHistory")
|
||||||
|
chatHistory.appendText("$username: $message ${System.lineSeparator()}")
|
||||||
|
message = ""
|
||||||
|
} else {
|
||||||
|
fullMessage = "${chatHistoryView.readText()}"
|
||||||
|
}
|
||||||
|
val response = if (inputData.isNotEmpty()) {
|
||||||
|
fullMessage
|
||||||
|
} else {
|
||||||
|
"No data provided"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send the message to the client
|
||||||
|
t.sendResponseHeaders(200, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
println("API request handled: $inputData")
|
||||||
|
os.close()
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
class CreateAccount : HttpHandler {
|
||||||
|
override fun handle(t: HttpExchange) {
|
||||||
|
val path = t.requestURI.path
|
||||||
|
val inputData = path.substringAfter("/api/createAccount/")
|
||||||
|
println("Account creation request recieved: $inputData")
|
||||||
|
|
||||||
|
// Parse data sent to the server by client
|
||||||
|
var username = ""
|
||||||
|
var token = ""
|
||||||
|
var message = ""
|
||||||
|
var dataType = ""
|
||||||
|
var isParsingData = 0
|
||||||
|
for (char in inputData) {
|
||||||
|
val character = char
|
||||||
|
if (character == ':') {
|
||||||
|
isParsingData = 1
|
||||||
|
} else if (isParsingData == 1) {
|
||||||
|
if (character == '}') {
|
||||||
|
isParsingData = 0
|
||||||
|
dataType = ""
|
||||||
|
} else if (character != '{') {
|
||||||
|
if (dataType == "username") {
|
||||||
|
username += character
|
||||||
|
} else if (dataType == "token") {
|
||||||
|
token += character
|
||||||
|
} else if (dataType == "message") {
|
||||||
|
message += character
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
dataType += character
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val userDatabaseParser = BufferedReader(File("userDatabase").reader())
|
||||||
|
var lineNumber = 1
|
||||||
|
var userExists = 0
|
||||||
|
|
||||||
|
// Search the user database to find required information about the user
|
||||||
|
userDatabaseParser.forEachLine { line ->
|
||||||
|
if (line.contains(username)) {
|
||||||
|
val response = "That username already exists on the server! Please choose a different username"
|
||||||
|
t.sendResponseHeaders(422, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
os.close()
|
||||||
|
userExists = 1
|
||||||
|
}
|
||||||
|
lineNumber++
|
||||||
|
}
|
||||||
|
userDatabaseParser.close()
|
||||||
|
if (userExists == 1) {return}
|
||||||
|
println("$username:$token")
|
||||||
|
|
||||||
|
if (username == "") {
|
||||||
|
val response = "Please input a username"
|
||||||
|
t.sendResponseHeaders(422, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
os.close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token == "") {
|
||||||
|
val response = "Please input a password"
|
||||||
|
t.sendResponseHeaders(422, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
os.close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
val userDatabaseFile = File("userDatabase")
|
||||||
|
userDatabaseFile.appendText("${System.lineSeparator()}$username:$token")
|
||||||
|
val response = "Creating account was successful!"
|
||||||
|
t.sendResponseHeaders(200, response.length.toLong())
|
||||||
|
val os = t.responseBody
|
||||||
|
os.write(response.toByteArray())
|
||||||
|
println("")
|
||||||
|
os.close()
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user