Unofficial Flutter plugin for using MongoDB Realm services on Android, iOS and web.
For using it on your app:
import 'package:flutter_mongodb_realm/flutter_mongo_realm.dart';
For API reference check here
The minimum requirements are:
- Android 5.0 (API 21)
- iOS 11.0
Doesn't require any setup besides adding as a dependency Web integration automatically!
Database support (MongoDB Atlas):
- Insert
- Find
- Delete
- Update
- Watch (also to specified IDs or with Filter)
- Aggregate [X]
Auth Providers:
- Email/Password
- Anonymously
- Google [X]
- Facebook [X]
- JWT
- Custom Authentication Function
Authentication:
- Auth Listener
- Reset Password
- Login/Sign in/Logout
Functions
- Calling a Realm function
Note: Other more features will come in the future :)
Note: [X] = Not implemented on Web
Inorder to connect a RealmApp to flutter add your main function:
main() async{
// ADD THESE 2 LINES
WidgetsFlutterBinding.ensureInitialized();
await RealmApp.init(<your_app_id>);
// other configurations..
runApp(MyApp());
}
In order to use the client define:
final client = MongoRealmClient();
In order to use authentication-related operations you have to use RealmApp
object:
final Realm app = RealmApp();
You can retrieve the current logged user by:
final user = app.currentUser;
// for more details (like email,birthday) use his 'profile' attribute
// For example:
final userEmail = user.profile.email;
You can log in using the following providers:
- Anonymous:
CoreRealmUser mongoUser = await app.login(Credentials.anonymous());
- Email\Password:
CoreRealmUser mongoUser = await app.login(
Credentials.emailPassword(username: <email_address>, password: <password>));
- Facebook:
For login with Facebook import the required flutter's package and configure in the native side as their instructions.
usage:
CoreRealmUser mongoUser = await app.login(
Credentials.facebook(<access_token>));
- Google:
1. Remove (if used) the version used from pubspec.yaml (ex. google_sign_in: ^4.5.1)
2. Use git repo version instead (copy from below)
3. In dart code use also serverClientId parmeter
This has to be done in order to get the auth code need by the Mongo Stitch SDK
calling on flutter:
CoreRealmUser mongoUser = await app.login(
Credentials.google(
serverClientId: <Google Server Client Id>, // just the start from "<ID>.apps.googleusercontent.com"
scopes: <list of scopes>,
));
in pubspec.yaml:
.. (other dependencies)
google_sign_in:
git:
url: git://github.com/fbcouch/plugins.git
path: packages/google_sign_in
ref: server-auth-code
CoreRealmUser mongoUser = await app.login(
Credentials.google(
serverClientId: <Google Server Client Id>, // just the start from "<ID>.apps.googleusercontent.com"
scopes: <list of scopes>,
));
- Custom JWT:
CoreRealmUser mongoUser = await app.login(Credentials.jwt(<token>);
- Custom (Auth) Function:
MongoDocument payload = MongoDocument({
"username": "bob"
})
CoreRealmUser mongoUser = await app.login(Credentials.function(payload);
NOTE: In any case , if mongoUser != null the login was successful.
Register a user with Email\Password
CoreRealmUser mongoUser = await app.registerWithEmail(
email: <email_address>, password: <password>);
await app.currentUser.logout(); // Logout the current user
You can send an email to reset user password: (email must be linked to an existing account)
await app.sendResetPasswordEmail(<YOUR_DESIRED_EMAIL>);
You can be notified when the auth state changes, such as login/logout..
/// for using as smart navigation according to user logged in or not
StreamBuilder _authBuilder(BuildContext context) {
return StreamBuilder(
stream: app.authListener(),
builder: (context, AsyncSnapshot snapshot) {
switch (snapshot.connectionState) {
case ConnectionState.none:
case ConnectionState.waiting:
// show loading indicator
return Scaffold(body: Center(child: CircularProgressIndicator()));
case ConnectionState.active:
// log error to console
if (snapshot.error != null) {
return Container(
alignment: Alignment.center,
child: Text(snapshot.error.toString()),
);
}
// redirect to the proper page
return snapshot.hasData ? HomeScreen() : LoginScreen();
default:
return LoginScreen();
}
},
);
}
To get access to a collection:
final collection = client.getDatabase(<database_name>).getCollection(<collection_name>);
collection.insertOne(MongoDocument({
// key-value pairs describing the document
}));
// OR
List<MongoDocument> documents = ...;
collection.insertMany(documents);
filtering can be used with QuerySelector class for more robust code
// fetch all document in the collection
var docs = collection.find();
// fetch all document in the collection applying some filter
var docs = collection.find(
filter: {
"year": QuerySelector.gt(2010)..lte(2014), // == 'year'>2010 && 'year'<=2014
});
// optional: can also add find options (projection/limit/sort)
var docs = await collection.find(
filter: {
"year": QueryOperator.gt(2010)..lte(2014),
},
options: RemoteFindOptions(
projection: {
"title": ProjectionValue.INCLUDE,
"rated": ProjectionValue.INCLUDE,
"year": ProjectionValue.INCLUDE,
},
limit: 70,
sort: {
"year": OrderValue.ASCENDING,
}
),
);
// the same as above, but just get the first matched one
var document = await collection.findOne();
var document = await collection.findOne(
filter: {
"year": QuerySelector.gt(2010)..lte(2014), // == 'year'>2010 && 'year'<=2014
});
// fetch the number of document in the collection
int collectionSize = await collection.count();
// count the number of document that apply to some filter
int size = await collection.count({
"age": 25,
});
filtering can be used with QuerySelector class for more robust code
// fetch all documents in the collection
var deletedDocs = await collection.deleteMany({});
// fetch the first document in the collection applying some filter
var deletedDocs = await collection.deleteOne({"age": 24});
// fetch all document in the collection applying some filter
var deletedDocs =
await collection.deleteMany({"age": QuerySelector.gte(24)});
Updating the only first matched document:
await collection.updateOne(
// adding a filter (optional)
filter:{
"_id": ObjectId('601204a6a80d3fbab2e3a73f'),
},
// adding an update operation (as matched the MongoDB SDK ones)
update: UpdateSelector.set({
"age": 26,
});
);
Updating the only all matched documents:
// adding a filter (optional)
filter:{
"name": "adam",
},
// adding an update operation (as matched the MongoDB SDK ones)
update: UpdateSelector.set({
"quantity": 670,
});
// removing 'apples' from 'favs' array
update: UpdateOperator.pull({
"favs": QueryOperator.in$(['apples'])
});
// adding 'tomatoes' & 'onions into 'favs' array
update: UpdateOperator.push({
"favs": ArrayModifier.each(['tomatoes','onions'])
});
);
First, Get the stream to subscribed to any document change in the collection
final stream = collection.watch();
// (optional) can watch only specified documents by their ids:
// 1. if they defined as ObjectId type
final stream2 = myCollection.watch(ids: ["5eca2d9fff448a4cbf8f6627"]);
// 2. if they defined as String type (`asObjectIds` is true by default)
final stream3 = myCollection.watch(ids: ["22", "8"], asObjectIds: false);
OR get the stream to subscribed to a part of the collection applying filter on the listened documents
final streamFilter = collection.watchWithFilter({
"age": QuerySelector.lte(26)
});
Afterwards, set a listener to a change in the collection
stream.listen((data) {
// data contains JSON string of the document that was changed
var fullDocument = MongoDocument.parse(data);
// Do other stuff...
});
define Pipeline Stages for aggregation, i.e:
List<PipelineStage> pipeline = [
PipelineStage.addFields({
"totalHomework": AggregateOperator.sum("homework"),
"totalQuiz": AggregateOperator.sum("quiz"),
}),
PipelineStage.addFields({
"totalScore": AggregateOperator.add(
["totalHomework", "totalQuiz", "extraCredit"]),
}),
];
And then set the pipelines stages to the aggregate function:
var list = await collection.aggregate(pipeline);
Another usages (not all stages are supported):
List<PipelineStage> pipeline = [
PipelineStage.skip(2),
PipelineStage.match({"status": "A"}),
PipelineStage.group(
"cust_id",
accumulators: {"total": AggregateOperator.sum("amount")},
),
];
List<PipelineStage> pipeline = [
PipelineStage.sample(2),
];
// can also RAW typed one
List<PipelineStage> pipeline = [
PipelineStage.raw({
//... some expression according to the MongoDB API
}),
PipelineStage.raw({
//... some expression according to the MongoDB API
}),
...
];
for calling a defined Realm function "sum" with argument 3&4
var result = await client.callFunction("sum", args: [3, 4])
You can also add a timeout (in ms), i.e 60 seconds:
var result = await client.callFunction("sum", args: [3, 4], requestTimeout: 60000)
If you found this project helpful or you learned something from the source code and want to thank me, consider buying me a cup of ☕