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
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|