Saltar al contingut principal

Ús del SDK iOS

Inicialització

El SDK es consumeix com un framework KMP a iOS. Utilitzeu el patró builder per crear una instància. La funció build() és async i realitza la generació del parell de claus DPoP (Secure Enclave), l'activació del token i la verificació de drets.

import VitaleraSdkCore

let sdk = try await VitaleraSdk.companion.builder()
.clientId(clientId: "your-org-license-key")
.addProvider(providerName: "omron")
.addProvider(providerName: "polar")
.addProvider(providerName: "standard-ble")
.connectedMode(enabled: true)
.build()

Gestioneu errors d'autenticació:

do {
let sdk = try await VitaleraSdk.companion.builder()
.clientId(clientId: "your-org-license-key")
.build()
} catch let error as SdkAuthException {
switch error.errorCode {
case "activation_failed":
showInvalidLicenseError()
case "entitlements_expired":
showConnectToInternetPrompt()
case "no_cache":
showFirstLaunchOfflineError()
default:
showGenericError(error.message)
}
}

Permisos

Afegiu els permisos necessaris al vostre Info.plist:

<key>NSBluetoothAlwaysUsageDescription</key>
<string>This app uses Bluetooth to connect with medical devices.</string>
<key>NSBluetoothPeripheralUsageDescription</key>
<string>This app uses Bluetooth to communicate with devices.</string>

Per a HealthKit:

<key>NSHealthShareUsageDescription</key>
<string>This app reads health data for monitoring.</string>

Descobriment de dispositius

El descobriment utilitza Swift AsyncSequence. Cada dispositiu descobert es retorna com un DeviceDescriptor.

let filter = DiscoveryFilter(
providers: nil,
capabilities: [DeviceCapability.bloodPressure],
namePrefix: nil
)

for try await descriptor in sdk.devices.discover(filter: filter, timeoutMs: 15_000) {
print("Found: \(descriptor.name) (\(descriptor.provider))")
}

Opcions de filtre

// Only Polar devices
let polarFilter = DiscoveryFilter(
providers: ["polar"],
capabilities: nil,
namePrefix: nil
)

// Only heart rate monitors
let hrFilter = DiscoveryFilter(
providers: nil,
capabilities: [DeviceCapability.heartRate],
namePrefix: nil
)

Connexió de dispositius

Creeu un dispositiu a partir d'un descriptor i connecteu-vos-hi. La funció connect() és async amb un temps d'espera configurable.

let device = sdk.devices.createDevice(descriptor: descriptor)

// Connect (async, with timeout)
try await device.connect(timeoutMs: 10_000)

Recollida de dades

El mètode collect() retorna un AsyncSequence de classes d'observació tipades.

for try await observation in device.collect() {
if let bp = observation as? BloodPressureObservation {
print("BP: \(bp.systolic)/\(bp.diastolic) mmHg")
if let pulse = bp.pulseRate {
print("Pulse: \(pulse) bpm")
}
} else if let hr = observation as? HeartRateObservation {
print("HR: \(hr.heartRate) bpm")
} else if let temp = observation as? TemperatureObservation {
print("Temp: \(temp.temperature) C")
} else if let weight = observation as? WeightObservation {
print("Weight: \(weight.weight) kg")
} else if let spo2 = observation as? OxygenSaturationObservation {
print("SpO2: \(spo2.spo2)%")
}
}

// Disconnect when done
try await device.disconnect()

Mode connectat

Quan el mode connectat està habilitat, resolveu pacients i envieu observacions:

let patient = try await sdk.connected().resolveMonitored(externalId: "patient-ext-id")

try await sdk.connected().postObservation(
monitoredId: patient.id,
observation: BaseObservation(
observationType: "blood-pressure",
issued: ISO8601DateFormatter().string(from: Date()),
components: [
ObservationComponent(name: "systolic", value: "120", valueUnit: "mmHg"),
ObservationComponent(name: "diastolic", value: "80", valueUnit: "mmHg"),
]
)
)

Gestió d'errors

do {
try await device.connect(timeoutMs: 10_000)
for try await obs in device.collect() {
// process observation
}
} catch let error as SdkError {
switch error {
case is SdkError.ConnectionTimeout:
showError("Connection timed out. Move closer to the device.")
case is SdkError.BluetoothDisabled:
showError("Please enable Bluetooth.")
case is SdkError.BluetoothPermissionDenied:
showError("Bluetooth permissions required.")
default:
showError("Device error: \(error.localizedDescription)")
}
}

HealthKit

Llegiu d'Apple HealthKit (no es requereix escaneig BLE):

let descriptor = try await sdk.devices.discover(
filter: DiscoveryFilter(providers: ["apple-healthkit"], capabilities: nil, namePrefix: nil),
timeoutMs: 5_000
).first { _ in true }!

let store = sdk.devices.createDevice(descriptor: descriptor)
try await store.connect()
for try await observation in store.collect() {
// Process HealthKit observations
}
try await store.disconnect()

Neteja de recursos

Crideu sempre close() quan el SDK ja no sigui necessari:

sdk.close()