atrade/src/main/kotlin/Main.kt

152 lines
6.3 KiB
Kotlin
Raw Normal View History

2026-02-19 15:47:31 +09:00
import ConfigTable.grade_1_buy
import ConfigTable.grade_1_profit
import ConfigTable.grade_2_buy
import ConfigTable.grade_2_profit
import ConfigTable.grade_3_buy
import ConfigTable.grade_3_profit
import ConfigTable.grade_4_buy
import ConfigTable.grade_4_profit
import ConfigTable.grade_5_buy
import ConfigTable.grade_5_profit
import ConfigTable.max_count
2026-01-10 18:16:50 +09:00
import androidx.compose.foundation.layout.*
import androidx.compose.material.*
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.window.Window
2026-01-22 16:21:18 +09:00
import androidx.compose.ui.window.WindowPlacement
2026-01-10 18:16:50 +09:00
import androidx.compose.ui.window.application
2026-01-22 16:21:18 +09:00
import androidx.compose.ui.window.rememberWindowState
import io.ktor.client.HttpClient
import io.ktor.client.engine.cio.CIO
import io.ktor.client.plugins.contentnegotiation.ContentNegotiation
import io.ktor.client.plugins.logging.DEFAULT
import io.ktor.client.plugins.logging.LogLevel
import io.ktor.client.plugins.logging.Logger
import io.ktor.client.plugins.logging.Logging
import io.ktor.serialization.kotlinx.json.json
2026-02-06 17:53:17 +09:00
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch
2026-01-22 16:21:18 +09:00
import kotlinx.serialization.json.Json
2026-01-10 18:16:50 +09:00
import org.jetbrains.exposed.sql.*
import org.jetbrains.exposed.sql.transactions.transaction
import model.AppConfig
2026-01-13 16:04:25 +09:00
import model.KisSession
2026-01-22 16:21:18 +09:00
import network.DartCodeManager
2026-02-06 17:53:17 +09:00
import network.KisTradeService
2026-01-23 17:05:09 +09:00
import service.LlamaServerManager
2026-01-21 18:59:55 +09:00
import network.NewsService
2026-01-10 18:16:50 +09:00
import org.jetbrains.exposed.sql.selectAll
2026-02-06 17:53:17 +09:00
import service.AutoTradingManager
2026-02-04 14:52:09 +09:00
import service.SystemSleepPreventer
2026-02-06 17:53:17 +09:00
import service.TradingDecisionCallback
2026-01-10 18:16:50 +09:00
import ui.DashboardScreen
import ui.SettingsScreen
// 화면 상태 정의
enum class AppScreen { Settings, Dashboard }
fun main() = application {
2026-02-04 14:52:09 +09:00
SystemSleepPreventer.start()
2026-02-06 17:53:17 +09:00
2026-01-22 16:21:18 +09:00
LaunchedEffect(Unit) {
// NewsService나 KisTradeService에서 사용하는 client를 전달
DartCodeManager.updateCorpCodes(HttpClient(CIO) {
install(ContentNegotiation) {
json(Json {
ignoreUnknownKeys = true
encodeDefaults = true // 기본값이 포함된 요청 바디를 정확히 전송하기 위해 필요
})
}
// [수정] 모든 로그(Headers + Body)를 찍도록 설정
install(Logging) {
logger = Logger.DEFAULT
level = LogLevel.BODY
}
})
}
2026-01-13 16:04:25 +09:00
// 앱 실행 시 필요한 바이너리 경로 (실행 파일 위치)
2026-01-10 18:16:50 +09:00
val binPath = "./src/main/resources/bin/llama-server"
2026-01-22 16:21:18 +09:00
val windowState = rememberWindowState(
placement = WindowPlacement.Maximized
)
Window(onCloseRequest = ::exitApplication, title = "KIS AI 자동매매", state = windowState) {
2026-01-10 18:16:50 +09:00
var currentScreen by remember { mutableStateOf(AppScreen.Settings) }
2026-01-13 16:04:25 +09:00
var isLoaded by remember { mutableStateOf(false) }
val scope = rememberCoroutineScope()
2026-01-10 18:16:50 +09:00
2026-01-13 16:04:25 +09:00
// 1. 앱 시작 시 DB에서 마지막 설정 로드 (KisSession에 주입)
2026-01-10 18:16:50 +09:00
LaunchedEffect(Unit) {
DatabaseFactory.init()
2026-01-13 16:04:25 +09:00
transaction {
2026-01-10 18:16:50 +09:00
ConfigTable.selectAll().lastOrNull()?.let {
2026-01-13 16:04:25 +09:00
KisSession.config = AppConfig(
realAppKey = it[ConfigTable.realAppKey],
realSecretKey = it[ConfigTable.realSecretKey],
realAccountNo = it[ConfigTable.realAccountNo],
vtsAppKey = it[ConfigTable.vtsAppKey],
vtsSecretKey = it[ConfigTable.vtsSecretKey],
vtsAccountNo = it[ConfigTable.vtsAccountNo],
2026-01-10 18:16:50 +09:00
isSimulation = it[ConfigTable.isSimulation],
2026-01-14 15:42:26 +09:00
htsId = it[ConfigTable.htsId],
2026-01-21 18:30:03 +09:00
modelPath = it[ConfigTable.modelPath],
2026-02-19 15:47:31 +09:00
embedModelPath = it[ConfigTable.embedModelPath],
FEES_AND_TAXRATE = it[ConfigTable.fees_and_taxrate],
MINIMUM_NET_PROFIT = it[ConfigTable.minimum_net_profit],
BUY_WEIGHT = it[ConfigTable.buy_weight],
MAX_BUDGET = it[ConfigTable.max_budget],
MAX_PRICE = it[ConfigTable.max_price],
MIN_PRICE = it[ConfigTable.min_price],
MIN_PURCHASE_SCORE = it[ConfigTable.min_purchase_score],
GRADE_5_BUY = it[grade_5_buy],
GRADE_5_PROFIT = it[grade_5_profit],
GRADE_4_BUY = it[grade_4_buy],
GRADE_4_PROFIT = it[grade_4_profit],
GRADE_3_BUY = it[grade_3_buy],
GRADE_3_PROFIT = it[grade_3_profit],
GRADE_2_BUY = it[grade_2_buy],
GRADE_2_PROFIT = it[grade_2_profit],
GRADE_1_BUY = it[grade_1_buy],
GRADE_1_PROFIT = it[grade_1_profit],
MAX_COUNT = it[max_count],
2026-01-10 18:16:50 +09:00
)
}
}
2026-01-19 17:09:37 +09:00
2026-01-13 16:04:25 +09:00
isLoaded = true
2026-01-10 18:16:50 +09:00
}
2026-01-13 16:04:25 +09:00
if (!isLoaded) {
// 로딩 중 표시
CircularProgressIndicator()
2026-01-10 18:16:50 +09:00
} else {
when (currentScreen) {
AppScreen.Settings -> {
SettingsScreen(
2026-01-13 16:04:25 +09:00
onAuthSuccess = {
// 2. 설정 및 인증 완료 시점의 처리
val config = KisSession.config
// LLM 서버 시작 (설정된 모델 경로 사용)
if (config.modelPath.isNotEmpty()) {
2026-01-21 18:30:03 +09:00
LlamaServerManager.startServer(binPath, config.modelPath,port = 8080)
}
if (config.embedModelPath.isNotEmpty()) {
LlamaServerManager.startServer(binPath, config.embedModelPath, port = 8081)
2026-01-13 16:04:25 +09:00
}
// 대시보드로 화면 전환
2026-01-10 18:16:50 +09:00
currentScreen = AppScreen.Dashboard
}
)
}
AppScreen.Dashboard -> {
2026-01-13 16:04:25 +09:00
DashboardScreen()
2026-01-10 18:16:50 +09:00
}
}
}
}
}