Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

[ODM] Add a way to store the document ID in the model #7613

Closed
rrousselGit opened this issue Dec 15, 2021 · 10 comments · Fixed by #9600
Closed

[ODM] Add a way to store the document ID in the model #7613

rrousselGit opened this issue Dec 15, 2021 · 10 comments · Fixed by #9600
Assignees
Labels
plugin: odm type: enhancement New feature or request

Comments

@rrousselGit
Copy link
Contributor

Users want a way to store the ID of a document within the deserialized class

Some things to consider:

  • calling ref.set(...) or ref.update(...) should not allow changing the "id", since it isn't part of the document but instead some metadata
@arnoutvandervorst
Copy link

arnoutvandervorst commented Dec 15, 2021

Same goes for the documentPath, which would make an update self-contained. When deserialising using json_serializable I augment all objects with id, path and parent. Since this is not handled by json_serializable I have an abstract BaseModel with only these properties (which are set to @jsonkey(ignore: true), all other Firestore based classes extend BaseModel.

This means changing the withConverter to:

fromFirestore: (doc, _) => fromFirestore(doc)..augment(doc),
toFirestore: (model, _) => toFirestore(model),

void augment(DocumentSnapshot<Map<String, dynamic>> doc) {
  id = doc.id;
  path = doc.reference.path;
  parent = doc.reference.parent;
}

@darshankawar darshankawar added triage Issue is currently being triaged. plugin: odm type: enhancement New feature or request and removed triage Issue is currently being triaged. labels Dec 16, 2021
@Lyokone
Copy link
Contributor

Lyokone commented Dec 18, 2021

Would something like that be acceptable for a PR : https://github.com/Lyokone/flutterfire/pull/1/files ?

I would add the ability to configure how the mapping is done.
Would love know if this approach is acceptable or if it's not something maintainable enough to make it to master ?

@rrousselGit
Copy link
Contributor Author

@Lyokone no, there's a lot more to it than this

There's some disambiguation to do. For example what if the model contains a property called path? We don't want the document path to override the actual path property.
Meaning there's some error handling to do too: what if the json contains a path/id property but they were supposed to be injected by Firestore?

Mutating the map passed to fromJson is a no-go either, since that could have side-effects

We also don't want to generate update(path: ...)/update(id: ...)

And there's testing needed

@poirierlouis
Copy link

I propose to add an annotation, something like:

@JsonSerializable('galaxies')
class Galaxy {

  Galaxy({this.documentId, required this.name});

  @JsonDocumentId()
  final String? documentId;

  final String name;

}

Type would be nullable:

  • when instantiating id is null by default.
  • when reading from Firestore, id is obtained from document reference using annotation metadata and passed to fromJson map.

If possible, annotation could assert Type is nullable.

I think it would prevent ambiguation as mentionned above.
It would also allow to not generate update(documentId: ...).

Would it also have side-effects to mutate the map passed to fromJson while using this annotation?

Finally I guess this might be done along with json_serializable package.

@AgDude
Copy link

AgDude commented Dec 31, 2021

I would like to propose making the snapshot or reference available rather than only the id. This has the advantage that it can be used to access subcollections directly from the model instance.

I am new to dart, so not sure on the feasibility, but what about something like below, where the new annotation excludes the field from json serializable as well as identifies it for the odm generator.

@JsonSerializable('galaxies')
class Galaxy {

  Galaxy({this.documentId, required this.name});

  @docementSnapshot()
  final FirestoreDocumentSnapshot? snapshot;

  final String name;

}

@rrousselGit
Copy link
Contributor Author

Attaching the snapshot is counter-productive IMO.

If that solution is fine with you, then you could directly pass the Firestore snapshot instead of passing only the document content.

The purpose of this issue I believe is for code that is not firebase aware

@poirierlouis
Copy link

@rrousselGit Could my proposal be acceptable ?

@Jjagg
Copy link
Contributor

Jjagg commented Feb 24, 2022

I implemented this for my project using an annotation:

// user.dart

@JsonSerializable()
class User {
  User({
    required this.name,
  });

  factory User.fromJson(Map<String, dynamic> json) => _$UserFromJson(json);

  @Inject(FirestoreValue.id)
  String? _id;
  String? get id => _id;

  final String name;
}

// user.g.dart

static User fromFirestore(
  DocumentSnapshot<Map<String, Object?>> snapshot,
  SnapshotOptions? options,
) {
  final data = User.fromJson(snapshot.data()!);
  data._id = snapshot.id;
  return data;
}

Injected fields are filtered out of queryable fields, so users can use public mutable fields if they prefer.
I also added support for setters, but that's mostly because I was experimenting with the analyzer API; I don't think it is very useful to support setters for injection.

My branch is here: https://github.com/Jjagg/flutterfire/tree/odm-inject

I added logic to inject:

  • id
  • path
  • parent document id (for subcollections)

With some experimentation this is the way I thought worked best for injecting stuff.

@larssn
Copy link
Contributor

larssn commented Mar 25, 2022

I suggest not deviating the API from the official one. Otherwise we could find ourselves in a future scenario where the official API (iOS/Android) is changed in a way that makes it incompatible with this version.

IMHO this plugin should mainly exist as a wrapper around the official API.

@Jjagg
Copy link
Contributor

Jjagg commented Jun 21, 2022

Any feedback on my comment?
The implementation is fairly straightforward. If my approach is seen as a good solution I could try to set up a PR for it :)

@Ehesp Ehesp added this to the [ODM] September 2022 milestone Aug 3, 2022
@Ehesp Ehesp changed the title Add a way to store the document ID in the model [ODM] Add a way to store the document ID in the model Aug 3, 2022
@firebase firebase locked and limited conversation to collaborators Oct 28, 2022
# for free to subscribe to this conversation on GitHub. Already have an account? #.
Labels
plugin: odm type: enhancement New feature or request
Projects
None yet
Development

Successfully merging a pull request may close this issue.

9 participants