| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443 |
- 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<Suggestion?, Never> {
- 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)
- let tdd = self.loadFileFromStorage(name: OpenAPS.Monitor.tdd)
- let tdd_averages = self.loadFileFromStorage(name: OpenAPS.Monitor.tdd_averages)
- // 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 preferences = self.loadFileFromStorage(name: Settings.preferences)
- let suggested = self.determineBasal(
- glucose: glucose,
- currentTemp: tempBasal,
- iob: iob,
- profile: profile,
- autosens: autosens.isEmpty ? .null : autosens,
- meal: meal,
- microBolusAllowed: true,
- reservoir: reservoir,
- pumpHistory: pumpHistory,
- preferences: preferences,
- basalProfile: basalProfile,
- tdd: tdd,
- tdd_averages: tdd_averages
- )
- 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<Autosens?, Never> {
- 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<Autotune?, Never> {
- 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<Autotune?, Never> {
- 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,
- pumpHistory: JSON,
- preferences: JSON,
- basalProfile: JSON,
- tdd: JSON,
- tdd_averages: 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,
- false, // clock
- pumpHistory,
- preferences,
- basalProfile,
- tdd,
- tdd_averages
- ]
- )
- }
- }
- 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)) ?? ""
- }
- }
|