- Make raw tables easier to use:
- Introduce the
RawTableSchemastruct storing the name of a raw table in the database. When set,putanddeletestatements can be inferred automatically. - Add
RawTable//jsonDescription, which can be passed to thepowersync_create_raw_table_crud_triggerSQL function to auto-create triggers forwarding writes tops_crud.
- Introduce the
- Fix
SyncStreamStatusfields not being visible. - Update PowerSync core extension to version 0.4.11.
- Add support for sync streams.
- Fixed MainActor isolation warning for UserAgent function.
- Support tvOS.
- Fix "Linking a static library that was built with
-gmodules, but the module cache was not found.` build warnings. - Update to SQLite 3.51.2.
- Add
initialStatementsparameter toPowerSyncDatabase(). These statements run before anything else when databases are opened, which is useful to e.g. enable encryption.
-
Fixed user agent strings not following expected format.
-
Added Alpha
PowerSyncGRDBproduct which supports sharing GRDBDatabasePools with PowerSync and application logic. -
Add
appMetadataparameter toPowerSyncDatabase.connect()(viaConnectOptions) to include application metadata in sync requests. This metadata is merged into sync requests and displayed in PowerSync service logs.Note: This requires a PowerSync service version
>=1.17.0in order for logs to display metadata.try await database.connect( connector: connector, options: ConnectOptions( appMetadata: ["appVersion": "1.0.0", "deviceId": "device456"] ) )
- Fix
PowerSyncKotlinframeworks not containing aCFBundleVersion.
- Pin dependency on
CSQLiteto avoid "unstable version package" errors.
- Enable the
newClientImplementationby default. This should improve performance and memory usage. - Potential Breaking Change The
newClientImplementationnow uses WebSockets to connect to the PowerSync service. These WebSockets connections do not log events toSyncClientConfiguration->requestLogger. - Add the
softflag todisconnectAndClear()which keeps an internal copy of synced data in the database, allowing faster re-sync if a compatible token is used in the next connect() call - Update PowerSync SQLite core to 0.4.10
- Update Kotlin SDK to 1.9.0.
- Bumped the minimum Swift tools version requirement to 6.1
- Update Kotlin SDK to 1.7.0.
- Add
close(deleteDatabase:)method toPowerSyncDatabaseProtocolfor deleting SQLite database files when closing the database. This includes the main database file and all WAL mode files (.wal, .shm, .journal). Files that don't exist are ignored, but an error is thrown if a file exists but cannot be deleted.
// Close the database and delete all SQLite files
try await database.close(deleteDatabase: true)
// Close the database without deleting files (default behavior)
try await database.close(deleteDatabase: false)
// or simply
try await database.close()- Add
PowerSyncDataTypeConvertibleprotocol for casting query parameters to SQLite supported types. - [Internal] Removed unnecessary
Taskcreation in Attachment helperFileManagerStorageAdapter.
- Update core extension to 0.4.6 (changelog)
- Add
getCrudTransactions(), returning an async sequence of transactions. - Compatibility with Swift 6.2 and XCode 26.
- Update minimum MacOS target to v12
- Update minimum iOS target to v15
- [Attachment Helpers] Added automatic verification or records'
local_urivalues onAttachmentQueueinitialization. initialization can be awaited withAttachmentQueue.waitForInit().AttachmentQueue.startSync()also performs this verification.waitForInit()is only recommended ifstartSyncis not called directly after creating the queue.
- Update core extension to 0.4.5 (changelog)
- Additional Swift 6 Strict Concurrency Checking declarations added for remaining protocols.
- Fix issue in legacy sync client where local writes made offline could have their upload delayed until a keepalive event was received. This could also cause downloaded updates to be delayed even further until all uploads were completed.
-
Fix null values in CRUD entries being reported as strings.
-
Added support for Swift 6 strict concurrency checking.
- Accepted query parameter types have been updated from
[Any]to[Sendable]. This should cover all supported query parameter types. - Query and lock methods' return
Resultgeneric types now should extendSendable. - Deprecated default
open class PowerSyncBackendConnector. Devs should preferably implement thePowerSyncBackendConnectorProtocol
- Accepted query parameter types have been updated from
-
Potential Breaking Change: Attachment helpers have been updated to better support Swift 6 strict concurrency checking.
Actorisolation is improved, but developers who customize or extendAttachmentQueuewill need to update their implementations. The default instantiation ofAttachmentQueueremains unchanged.AttachmentQueueProtocolnow defines the basic requirements for an attachment queue, with most base functionality provided via an extension. Custom implementations should extendAttachmentQueueProtocol. -
Added
PowerSyncDynamicproduct to package. Importing this product should restore XCode preview functionality. -
[Internal] Instantiate Kotlin Kermit logger directly.
-
[Internal] Improved connection context error handling.
- Added the ability to log PowerSync sync network requests.
try await database.connect(
connector: Connector(),
options: ConnectOptions(
clientConfiguration: SyncClientConfiguration(
requestLogger: SyncRequestLoggerConfiguration(
requestLevel: .headers
) { message in
// Handle Network request logs here
print(message)
}
)
)
)- Update core extension to 0.4.4, fixing a bug where
hasSyncedwould turnfalsewhen losing connectivity.
- Update SQLite to 3.50.3.
- Support receiving binary sync lines over HTTP when the Rust client is enabled.
- Remove the experimental websocket transport mode.
- Use version
0.4.2of the PowerSync core extension, which improves the reliability of the new Rust client implementation. - Add support for raw tables, which are custom tables managed by the user instead of JSON-based views managed by the SDK.
- Fix attachments never downloading again when the sandbox path of the app (e.g. on the simulator) changes.
- Use version
0.4.1of the PowerSync core extension, which fixes an issue with the new Rust client implementation. - Fix crud uploads when web sockets are used as a connection method.
- Improved
CrudBatchandCrudTransactioncompletefunction extensions. Developers no longer need to specifynilas an argument forwriteCheckpointwhen callingCrudBatch.complete. The basecompletefunctions still accept an optionalwriteCheckpointargument if developers use custom write checkpoints.
guard let finalBatch = try await powersync.getCrudBatch(limit: 100) else {
return nil
}
- try await batch.complete(writeCheckpoint: nil)
+ try await batch.complete()- Fix reported progress around compactions / defrags on the sync service.
- Use version
0.4.0of the PowerSync core extension, which improves sync performance. - Add a new sync client implementation written in Rust instead of Kotlin. While this client is still
experimental, we intend to make it the default in the future. The main benefit of this client is
faster sync performance, but upcoming features will also require this client. We encourage
interested users to try it out by opting in to experimental APIs and passing options when
connecting:
Switching between the clients can be done at any time without compatibility issues. If you run into issues with the new client, please reach out to us!
@_spi(PowerSyncExperimental) import PowerSync try await db.connect(connector: connector, options: ConnectOptions( newClientImplementation: true, ))
- In addition to HTTP streams, the Swift SDK also supports fetching sync instructions from the
PowerSync service in a binary format. This requires the new sync client, and can then be enabled
on the sync options:
@_spi(PowerSyncExperimental) import PowerSync try await db.connect(connector: connector, options: ConnectOptions( newClientImplementation: true, connectionMethod: .webSocket, ))
- Add sync progress information through
SyncStatusData.downloadProgress. - Add
trackPreviousValuesoption onTablewhich setsCrudEntry.previousValuesto previous values on updates. - Add
trackMetadataoption onTablewhich adds a_metadatacolumn that can be used for updates. The configured metadata is available throughCrudEntry.metadata. - Add
ignoreEmptyUpdatesoption which skips creating CRUD entries for updates that don't change any values.
-
Improved the stability of watched queries. Watched queries were previously susceptible to runtime crashes if an exception was thrown in the update stream. Errors are now gracefully handled.
-
Deprecated
PowerSyncCredentialsuserIdfield. This value is not used by the PowerSync service. -
Added
readLockandwriteLockAPIs. These methods allow obtaining a SQLite connection context without starting a transaction. -
Removed references to the PowerSync Kotlin SDK from all public API protocols. Dedicated Swift protocols are now defined. These protocols align better with Swift primitives. See the
BRAKING CHANGESsection for more details. Updated protocols include:ConnectionContext- The context provided byreadLockandwriteLockTransaction- The context provided byreadTransactionandwriteTransactionCrudBatch- Response fromgetCrudBatchCrudTransactionResponse fromgetNextCrudTransactionCrudEntry- Crud entries forCrudBatchandCrudTransactionUpdateType- Operation type forCrudEntrysSqlCursor- Cursor used to map SQLite results to typed result setsJsonParam- JSON parameters used to declare client parameters in theconnectmethodJsonValue- Individual JSON field types forJsonParam
-
Database and transaction/lock level query
executemethods now have@discardableResultannotation. -
Query methods'
parameterstyping has been updated to[Any?]from[Any]. This makes passingnilor optional values to queries easier. -
AttachmentContext,AttachmentQueue,AttachmentServiceandSyncingServiceare are now explicitly declared asopenclasses, allowing them to be subclassed outside the defining module.
BREAKING CHANGES:
- Completing CRUD transactions or CRUD batches, in the
PowerSyncBackendConnectoruploadDatahandler, now has a simpler invocation.
- _ = try await transaction.complete.invoke(p1: nil)
+ try await transaction.complete()indexbasedSqlCursorgetters now throw if the query result column value isnil. This is now consistent with the behaviour of named column getter operations. NewgetXxxxxOptional(index: index)methods are available if the query result value could benil.
let results = try transaction.getAll(
sql: "SELECT * FROM my_table",
parameters: [id]
) { cursor in
- cursor.getString(index: 0)!
+ cursor.getStringOptional(index: 0)
+ // OR
+ // try cursor.getString(index: 0) // if the value should be required
}SqlCursorgetters now directly return Swift types.getLonghas been replaced withgetInt64.
let results = try transaction.getAll(
sql: "SELECT * FROM my_table",
parameters: [id]
) { cursor in
- cursor.getBoolean(index: 0)?.boolValue,
+ cursor.getBooleanOptional(index: 0),
- cursor.getLong(index: 0)?.int64Value,
+ cursor.getInt64Optional(index: 0)
+ // OR
+ // try cursor.getInt64(index: 0) // if the value should be required
}- Client parameters now need to be specified with strictly typed
JsonValueenums.
try await database.connect(
connector: PowerSyncBackendConnector(),
params: [
- "foo": "bar"
+ "foo": .string("bar")
]
)SyncStatusvalues now use Swift primitives for status attributes.lastSyncedAtnow is ofDatetype.
- let lastTime: Date? = db.currentStatus.lastSyncedAt.map {
- Date(timeIntervalSince1970: TimeInterval($0.epochSeconds))
- }
+ let time: Date? = db.currentStatus.lastSyncedAtcrudThrottleMsandretryDelayMsin theconnectmethod have been updated tocrudThrottleandretryDelaywhich are now of typeTimeInterval. Previously the parameters were specified in milliseconds, theTimeIntervaltyping now requires values to be specified in seconds.
try await database.connect(
connector: PowerSyncBackendConnector(),
- crudThrottleMs: 1000,
- retryDelayMs: 5000,
+ crudThrottle: 1,
+ retryDelay: 5,
params: [
"foo": .string("bar"),
]
)throttleMsin the watched queryWatchOptionshas been updated tothrottlewhich is now of typeTimeInterval. Previously the parameters were specified in milliseconds, theTimeIntervaltyping now requires values to be specified in seconds.
let stream = try database.watch(
options: WatchOptions(
sql: "SELECT name FROM users ORDER BY id",
- throttleMs: 1000,
+ throttle: 1,
mapper: { cursor in
try cursor.getString(index: 0)
}
))- Update
powersync-kotlindependency to version1.0.0-BETA32, which includes:- Removed unnecessary
User-Idheader from internal PowerSync service requests. - Fix
getNextCrudTransaction()only returning a single item.
- Removed unnecessary
- Added attachment sync helpers
- Added support for cancellations in watched queries
- Fix deadlock when
connect()is called immediately after opening a database.
- Added the ability to specify a custom logging implementation
let db = PowerSyncDatabase(
schema: Schema(
tables: [
Table(
name: "users",
columns: [
.text("name"),
.text("email")
]
)
]
),
logger: DefaultLogger(minSeverity: .debug)
)- added
.close()method onPowerSyncDatabaseProtocol - Update
powersync-kotlindependency to version1.0.0-BETA29, which fixes these issues:- Fix potential race condition between jobs in
connect()anddisconnect(). - Fix race condition causing data received during uploads not to be applied.
- Fixed issue where automatic driver migrations would fail with the error:
- Fix potential race condition between jobs in
Sqlite operation failure database is locked attempted to run migration and failed. closing connection
- Update PowerSync SQLite core extension to 0.3.12.
- Added queuing protection and warnings when connecting multiple PowerSync clients to the same database file.
- Improved concurrent SQLite connection support. A single write connection and multiple read connections are used for concurrent read queries.
- Internally improved the linking of SQLite.
- Enabled Full Text Search support.
- Added the ability to update the schema for existing PowerSync clients.
- Fixed bug where local only, insert only and view name overrides were not applied for schema tables.
- Improved watch query internals. Added the ability to throttle watched queries.
- Added support for sync bucket priorities.
- Fixed an issue where throwing exceptions in the query
mappercould cause a runtime crash. - Internally improved type casting.
- BREAKING CHANGE:
watchqueries are now throwable and therefore will need to be accompanied by atrye.g.
try database.watch()- BREAKING CHANGE:
transactionfunctions are now throwable and therefore will need to be accompanied by atrye.g.
try await database.writeTransaction { transaction in
try transaction.execute(...)
}- Allow
executeerrors to be handled userIdis now set tonilby default and therefore it is no longer required to be set tonilwhen instantiatingPowerSyncCredentialsand can therefore be left out.
- Implement improvements to errors originating in Kotlin so that they can be handled in Swift
- Improve
__fetchCredentialsto log the error but not cause an app crash on error
- Allow cursor to use column name to get value by including the following functions that accept a column name parameter:
getBoolean,getBooleanOptional,getString,getStringOptional,getLong,getLongOptional,getDouble,getDoubleOptional - BREAKING CHANGE: This should not affect anyone but made
KotlinPowerSyncCredentials,KotlinPowerSyncDatabaseandKotlinPowerSyncBackendConnectorprivate as these should never have been public.
- BREAKING CHANGE: Update underlying powersync-kotlin package to BETA18.0 which requires transactions to become synchronous as opposed to asynchronous.
to
try await database.writeTransaction { transaction in try await transaction.execute( sql: "INSERT INTO users (id, name, email) VALUES (?, ?, ?)", parameters: ["1", "Test User", "test@example.com"] ) }
try await database.writeTransaction { transaction in transaction.execute( // <- This has become synchronous sql: "INSERT INTO users (id, name, email) VALUES (?, ?, ?)", parameters: ["1", "Test User", "test@example.com"] ) }
- Upgrade PowerSyncSqliteCore to 0.3.8
- Initial Beta release