Skip to content

Commit

Permalink
Opening a Realm as read-only (realm#260)
Browse files Browse the repository at this point in the history
* Configuration isReadOnly implemented. Fixes realm#92.
* Configuration read-only tests added + documentation
* CHANGELOG updated
  • Loading branch information
desistefanova authored Feb 18, 2022
1 parent 26e88d3 commit 2f77fdf
Show file tree
Hide file tree
Showing 4 changed files with 68 additions and 4 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ x.x.x Release notes (yyyy-MM-dd)
* Support change notifications on list collections. ([261](https://github.com/realm/realm-dart/pull/261))
* Added support checking if Realm lists and Realm objects are valid. ([#183](https://github.com/realm/realm-dart/pull/183))
* Support query on lists of realm objects. ([239](https://github.com/realm/realm-dart/pull/239))
* Added support for opening Realm in read-only mode. ([#260](https://github.com/realm/realm-dart/pull/260))

### Compatibility
* Dart ^2.15 on Windows, MacOS and Linux
Expand Down
22 changes: 18 additions & 4 deletions lib/src/configuration.dart
Original file line number Diff line number Diff line change
Expand Up @@ -36,11 +36,17 @@ class Configuration {
RealmSchema get schema => _schema;

/// Creates a [Configuration] with schema objects for opening a [Realm].
Configuration(List<SchemaObject> schemaObjects)
/// [readOnly] controls whether a [Realm] is opened as readonly.
/// This allows opening it from locked locations such as resources,
/// bundled with an application. The realm file must already exists.
Configuration(List<SchemaObject> schemaObjects, {bool readOnly = false})
: _schema = RealmSchema(schemaObjects),
_handle = realmCore.createConfig() {
schemaVersion = 0;
path = defaultPath;
if (readOnly) {
isReadOnly = true;
}
realmCore.setSchema(this);
}

Expand All @@ -53,7 +59,7 @@ class Configuration {
}

/// The platform dependent path to the default realm file - `default.realm`.
///
///
/// If set it should contain the name of the realm file. Ex. /mypath/myrealm.realm
static String get defaultPath => _defaultPath ??= _initDefaultPath();
static set defaultPath(String value) => _defaultPath = value;
Expand Down Expand Up @@ -83,10 +89,18 @@ class Configuration {
/// If omitted the [defaultPath] for the platform will be used.
String get path => realmCore.getConfigPath(this);
set path(String value) => realmCore.setConfigPath(this, value);

/// Gets or sets a value indicating whether a [Realm] is opened as readonly.
/// This allows opening it from locked locations such as resources,
/// bundled with an application.
///
/// The realm file must already exists at [path]
bool get isReadOnly => realmCore.getConfigReadOnly(this);
set isReadOnly(bool value) => realmCore.setConfigReadOnly(this, value);
}

/// A collection of properties describing the underlying schema of a [RealmObject].
///
///
/// {@category Configuration}
class SchemaObject {
/// Schema object type.
Expand All @@ -103,7 +117,7 @@ class SchemaObject {
}

/// Describes the complete set of classes which may be stored in a `Realm`
///
///
/// {@category Configuration}
class RealmSchema extends Iterable<SchemaObject> {
///@nodoc
Expand Down
10 changes: 10 additions & 0 deletions lib/src/native/realm_core.dart
Original file line number Diff line number Diff line change
Expand Up @@ -155,6 +155,16 @@ class _RealmCore {
_realmLib.realm_config_set_schema_version(config.handle._pointer, version);
}

bool getConfigReadOnly(Configuration config) {
int mode = _realmLib.realm_config_get_schema_mode(config.handle._pointer);
return mode == realm_schema_mode.RLM_SCHEMA_MODE_READ_ONLY;
}

void setConfigReadOnly(Configuration config, bool value) {
int mode = value ? realm_schema_mode.RLM_SCHEMA_MODE_READ_ONLY : realm_schema_mode.RLM_SCHEMA_MODE_AUTOMATIC;
_realmLib.realm_config_set_schema_mode(config.handle._pointer, mode);
}

ConfigHandle createConfig() {
final configPtr = _realmLib.realm_config_new();
return ConfigHandle._(configPtr);
Expand Down
39 changes: 39 additions & 0 deletions test/realm_test.dart
Original file line number Diff line number Diff line change
Expand Up @@ -202,6 +202,45 @@ Future<void> main([List<String>? args]) async {
config.schemaVersion = 3;
expect(config.schemaVersion, equals(3));
});

test('Configuration readOnly - opening non existing realm throws', () {
Configuration config = Configuration([Car.schema], readOnly: true);
expect(() => Realm(config), throws<RealmException>("Message: No such table exists"));
});

test('Configuration readOnly - open existing realm with read-only config', () {
Configuration config = Configuration([Car.schema]);
var realm = Realm(config);
realm.close();

// Open an existing realm as readonly.
config = Configuration([Car.schema], readOnly: true);
realm = Realm(config);
realm.close();
});

test('Configuration readOnly - reading is possible', () {
Configuration config = Configuration([Car.schema]);
var realm = Realm(config);
realm.write(() => realm.add(Car("Mustang")));
realm.close();

config.isReadOnly = true;
realm = Realm(config);
var cars = realm.all<Car>();
realm.close();
});

test('Configuration readOnly - writing on read-only Realms throws', () {
Configuration config = Configuration([Car.schema]);
var realm = Realm(config);
realm.close();

config = Configuration([Car.schema], readOnly: true);
realm = Realm(config);
expect(() => realm.write(() {}), throws<RealmException>("Can't perform transactions on read-only Realms."));
realm.close();
});
});

group('RealmClass tests:', () {
Expand Down

0 comments on commit 2f77fdf

Please sign in to comment.