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