import Combine import Foundation import JavaScriptCore final class OpenAPS { private let jsWorker = JavaScriptWorker() private let processQueue = DispatchQueue(label: "OpenAPS.processQueue", qos: .utility) private let storage: FileStorage init(storage: FileStorage) { self.storage = storage } func determineBasal(currentTemp: TempBasal, clock: Date = Date()) -> Future { Future { promise in self.processQueue.async { debug(.openAPS, "Start determineBasal") // clock self.storage.save(clock, as: Monitor.clock) // temp_basal let tempBasal = currentTemp.rawJSON self.storage.save(tempBasal, as: Monitor.tempBasal) // meal let pumpHistory = self.loadFileFromStorage(name: OpenAPS.Monitor.pumpHistory) let carbs = self.loadFileFromStorage(name: Monitor.carbHistory) let glucose = self.loadFileFromStorage(name: Monitor.glucose) let profile = self.loadFileFromStorage(name: Settings.profile) let basalProfile = self.loadFileFromStorage(name: Settings.basalProfile) let meal = self.meal( pumphistory: pumpHistory, profile: profile, basalProfile: basalProfile, clock: clock, carbs: carbs, glucose: glucose ) self.storage.save(meal, as: Monitor.meal) // iob let autosens = self.loadFileFromStorage(name: Settings.autosense) let iob = self.iob( pumphistory: pumpHistory, profile: profile, clock: clock, autosens: autosens.isEmpty ? .null : autosens ) self.storage.save(iob, as: Monitor.iob) // determine-basal let reservoir = self.loadFileFromStorage(name: Monitor.reservoir) let suggested = self.determineBasal( glucose: glucose, currentTemp: tempBasal, iob: iob, profile: profile, autosens: autosens.isEmpty ? .null : autosens, meal: meal, microBolusAllowed: true, reservoir: reservoir ) debug(.openAPS, "SUGGESTED: \(suggested)") if var suggestion = Suggestion(from: suggested) { suggestion.timestamp = suggestion.deliverAt ?? clock self.storage.save(suggestion, as: Enact.suggested) promise(.success(suggestion)) } else { promise(.success(nil)) } } } } func autosense() -> Future { Future { promise in self.processQueue.async { debug(.openAPS, "Start autosens") let pumpHistory = self.loadFileFromStorage(name: OpenAPS.Monitor.pumpHistory) let carbs = self.loadFileFromStorage(name: Monitor.carbHistory) let glucose = self.loadFileFromStorage(name: Monitor.glucose) let profile = self.loadFileFromStorage(name: Settings.profile) let basalProfile = self.loadFileFromStorage(name: Settings.basalProfile) let tempTargets = self.loadFileFromStorage(name: Settings.tempTargets) let autosensResult = self.autosense( glucose: glucose, pumpHistory: pumpHistory, basalprofile: basalProfile, profile: profile, carbs: carbs, temptargets: tempTargets ) debug(.openAPS, "AUTOSENS: \(autosensResult)") if var autosens = Autosens(from: autosensResult) { autosens.timestamp = Date() self.storage.save(autosens, as: Settings.autosense) promise(.success(autosens)) } else { promise(.success(nil)) } } } } func autotune(categorizeUamAsBasal: Bool = false, tuneInsulinCurve: Bool = false) -> Future { Future { promise in self.processQueue.async { debug(.openAPS, "Start autotune") let pumpHistory = self.loadFileFromStorage(name: OpenAPS.Monitor.pumpHistory) let glucose = self.loadFileFromStorage(name: Monitor.glucose) let profile = self.loadFileFromStorage(name: Settings.profile) let pumpProfile = self.loadFileFromStorage(name: Settings.pumpProfile) let carbs = self.loadFileFromStorage(name: Monitor.carbHistory) let autotunePreppedGlucose = self.autotunePrepare( pumphistory: pumpHistory, profile: profile, glucose: glucose, pumpprofile: pumpProfile, carbs: carbs, categorizeUamAsBasal: categorizeUamAsBasal, tuneInsulinCurve: tuneInsulinCurve ) debug(.openAPS, "AUTOTUNE PREP: \(autotunePreppedGlucose)") let previousAutotune = self.storage.retrieve(Settings.autotune, as: RawJSON.self) let autotuneResult = self.autotuneRun( autotunePreparedData: autotunePreppedGlucose, previousAutotuneResult: previousAutotune ?? profile, pumpProfile: pumpProfile ) debug(.openAPS, "AUTOTUNE RESULT: \(autotuneResult)") if let autotune = Autotune(from: autotuneResult) { self.storage.save(autotuneResult, as: Settings.autotune) promise(.success(autotune)) } else { promise(.success(nil)) } } } } func makeProfiles(useAutotune: Bool) -> Future { Future { promise in debug(.openAPS, "Start makeProfiles") self.processQueue.async { var preferences = self.loadFileFromStorage(name: Settings.preferences) if preferences.isEmpty { preferences = Preferences().rawJSON } let pumpSettings = self.loadFileFromStorage(name: Settings.settings) let bgTargets = self.loadFileFromStorage(name: Settings.bgTargets) let basalProfile = self.loadFileFromStorage(name: Settings.basalProfile) let isf = self.loadFileFromStorage(name: Settings.insulinSensitivities) let cr = self.loadFileFromStorage(name: Settings.carbRatios) let tempTargets = self.loadFileFromStorage(name: Settings.tempTargets) let model = self.loadFileFromStorage(name: Settings.model) let autotune = useAutotune ? self.loadFileFromStorage(name: Settings.autotune) : .empty let pumpProfile = self.makeProfile( preferences: preferences, pumpSettings: pumpSettings, bgTargets: bgTargets, basalProfile: basalProfile, isf: isf, carbRatio: cr, tempTargets: tempTargets, model: model, autotune: RawJSON.null ) let profile = self.makeProfile( preferences: preferences, pumpSettings: pumpSettings, bgTargets: bgTargets, basalProfile: basalProfile, isf: isf, carbRatio: cr, tempTargets: tempTargets, model: model, autotune: autotune.isEmpty ? .null : autotune ) self.storage.save(pumpProfile, as: Settings.pumpProfile) self.storage.save(profile, as: Settings.profile) if let tunedProfile = Autotune(from: profile) { promise(.success(tunedProfile)) return } promise(.success(nil)) } } } // MARK: - Private private func iob(pumphistory: JSON, profile: JSON, clock: JSON, autosens: JSON) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.iob)) worker.evaluate(script: Script(name: Prepare.iob)) return worker.call(function: Function.generate, with: [ pumphistory, profile, clock, autosens ]) } } private func meal(pumphistory: JSON, profile: JSON, basalProfile: JSON, clock: JSON, carbs: JSON, glucose: JSON) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.meal)) worker.evaluate(script: Script(name: Prepare.meal)) return worker.call(function: Function.generate, with: [ pumphistory, profile, clock, glucose, basalProfile, carbs ]) } } private func autotunePrepare( pumphistory: JSON, profile: JSON, glucose: JSON, pumpprofile: JSON, carbs: JSON, categorizeUamAsBasal: Bool, tuneInsulinCurve: Bool ) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.autotunePrep)) worker.evaluate(script: Script(name: Prepare.autotunePrep)) return worker.call(function: Function.generate, with: [ pumphistory, profile, glucose, pumpprofile, carbs, categorizeUamAsBasal, tuneInsulinCurve ]) } } private func autotuneRun( autotunePreparedData: JSON, previousAutotuneResult: JSON, pumpProfile: JSON ) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.autotuneCore)) worker.evaluate(script: Script(name: Prepare.autotuneCore)) return worker.call(function: Function.generate, with: [ autotunePreparedData, previousAutotuneResult, pumpProfile ]) } } private func determineBasal( glucose: JSON, currentTemp: JSON, iob: JSON, profile: JSON, autosens: JSON, meal: JSON, microBolusAllowed: Bool, reservoir: JSON ) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Prepare.determineBasal)) worker.evaluate(script: Script(name: Bundle.basalSetTemp)) worker.evaluate(script: Script(name: Bundle.getLastGlucose)) worker.evaluate(script: Script(name: Bundle.determineBasal)) if let middleware = self.middlewareScript(name: OpenAPS.Middleware.determineBasal) { worker.evaluate(script: middleware) } return worker.call( function: Function.generate, with: [ iob, currentTemp, glucose, profile, autosens, meal, microBolusAllowed, reservoir ] ) } } private func autosense( glucose: JSON, pumpHistory: JSON, basalprofile: JSON, profile: JSON, carbs: JSON, temptargets: JSON ) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.autosens)) worker.evaluate(script: Script(name: Prepare.autosens)) return worker.call( function: Function.generate, with: [ glucose, pumpHistory, basalprofile, profile, carbs, temptargets ] ) } } private func exportDefaultPreferences() -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.profile)) worker.evaluate(script: Script(name: Prepare.profile)) return worker.call(function: Function.exportDefaults, with: []) } } private func makeProfile( preferences: JSON, pumpSettings: JSON, bgTargets: JSON, basalProfile: JSON, isf: JSON, carbRatio: JSON, tempTargets: JSON, model: JSON, autotune: JSON ) -> RawJSON { dispatchPrecondition(condition: .onQueue(processQueue)) return jsWorker.inCommonContext { worker in worker.evaluate(script: Script(name: Prepare.log)) worker.evaluate(script: Script(name: Bundle.profile)) worker.evaluate(script: Script(name: Prepare.profile)) return worker.call( function: Function.generate, with: [ pumpSettings, bgTargets, isf, basalProfile, preferences, carbRatio, tempTargets, model, autotune ] ) } } private func loadJSON(name: String) -> String { try! String(contentsOf: Foundation.Bundle.main.url(forResource: "json/\(name)", withExtension: "json")!) } private func loadFileFromStorage(name: String) -> RawJSON { storage.retrieveRaw(name) ?? OpenAPS.defaults(for: name) } private func middlewareScript(name: String) -> Script? { if let body = storage.retrieveRaw(name) { return Script(name: "Middleware", body: body) } if let url = Foundation.Bundle.main.url(forResource: "javascript/\(name)", withExtension: "") { return Script(name: "Middleware", body: try! String(contentsOf: url)) } return nil } static func defaults(for file: String) -> RawJSON { let prefix = file.hasSuffix(".json") ? "json/defaults" : "javascript" guard let url = Foundation.Bundle.main.url(forResource: "\(prefix)/\(file)", withExtension: "") else { return "" } return (try? String(contentsOf: url)) ?? "" } }