-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
initial invite api #217
initial invite api #217
Conversation
src/invite-api.js
Outdated
/** @typedef {import('./rpc/index.js').MapeoRPC} MapeoRPC */ | ||
|
||
export class InviteApi extends TypedEmitter { | ||
// TODO: are invites persisted beyond this api? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
what happens when we've already accepted an invite a few days ago and got a new invite for the same project? there may need to be some persistence and lookup.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good question.
A previously accepted invite is the same issue as "already on this project". This is what the ALREADY
invite response for. This means that the InviteApi needs to look up which projects the device is already a member of - @achou11 can you suggest the best API for this? We'll need to pass something down to the InviteApi constructor.
If a device is already part of a project and they receive an invite, we should automatically respond with ALREADY
- there is no need for an event since I don't think the UX on the invitee side should show anything. On the invitor side the UX should show a "already invited" UX when receiving an invite response of ALREADY
.
We're not going to persist un-responded invites beyond the lifecycle of the app - they require a connection to the invitor peer to respond anyway.
What to do with previously rejected invites - let's not track rejected invites for now, I've created an issue to deal with this question post-MVP
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This means that the InviteApi needs to look up which projects the device is already a member of - @achou11 can you suggest the best API for this? We'll need to pass something down to the InviteApi constructor.
Initial thought is to provide a callback in the constructor to determine membership for a project id e.g.
new InviteApi({
// can update option name and return a string if there are more complex membership states to represent
isMember: (projectId: string) => Promise<boolean>
})
alternatively can pass a sharedDb
option like we do for MapeoProject
and let the InviteApi handle making the query itself. this approach may be useful if there are other project-related queries that may be needed
don't have a strong preference at the moment, but maybe leaning towards the latter since we kind of have a similar pattern for MapeoProject 🤷
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
with some additional pondering, thinking the sharedDb option is preferred for me, as it allows code specifically needed for the InviteApi to live with the implementation, as opposed to it leaking to the creator of the instance.
I'm sure my thoughts will bounce back and forth before I end up in this position again 😄
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
just as I anticipated, seeing #217 (comment) has me thinking about this again.
So to sum it up, the InviteApi needs a way to do these things:
- Check project membership
- Add the project when accepting a received invite
Passing the sharedDb
option would solve 1 well, but wouldn't (directly) solve 2 without adding duplicate code (I think). Duplicate code isn't necessarily bad though (and I don't think it would be copy-paste identical either...)
If we wanted to solve both, passing specific constructor callback options would be the simplest I guess i.e.
new InviteApi({
isMember: ...,
addProject: ...,
})
maybe that's just the way to go, before I start thinking of some (probably) convoluted approach using events 😄
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
currently using the callback-based approach in https://github.com/digidem/mapeo-core-next/pull/232/files#diff-a2f290f8510d711051a63f76d7d7b06721c1dd5b6ac7d86e8e3024795c2d829dR12, which I think i like right now.
Feels sensible that these API classes don't need to dive into the actual db and do their own queries
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
leaning into the callback-based approach, maybe all of these API classes can have a singular constructor opt called queries
, which is responsible for being the interface between the class and dbs created outside the class.
For example, the InviteApi would have something like:
type InviteApiQueries = {
isMember: ...
addProject: ...
}
and instantiation in the MapeoProject
will look like:
new InviteApi({
queries: {
isMember: async () => ...,
addProject: async () => ...,
},
...
})
honestly this is mostly aesthetic as it's not functionally different than specifying an opt for each callback. gut feeling is that it'd be a little cleaner to work with this, especially if an API class requires many queries to be specified. i also like that when i revisit the code, i'll have a more immediate idea of what db-related operations the API class requires
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I also like the aesthetics of the queries
object. Makes it more clear what those methods are for.
src/invite-api.js
Outdated
// TODO: I'm not seeing encryption keys used in the inviteResponse in the rpc api | ||
// what is the purpose of the encryption keys at this stage of the process or afterward? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I didn't track the encryptionKeys property of the invite yet as there wasn't a way to use it yet.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The encryptionKeys and projectKey will be needed in the accept()
method to add the project. @achou11 is working on the API for this, we should pass something down to the constructor here like addProject()
, or an instance of something that has that method.
The keys will need to be stored in the this.#invites
Map so we can look them up when we accept()
. We don't emit()
the keys though (e.g. expose to the API) because we keep these internal at all times.
src/invite-api.js
Outdated
// TODO: should this reply to one peer with `ACCEPT` and the rest with `ALREADY`? | ||
// How is the `ALREADY` decision determined? It looks like that isn't used yet. | ||
// Does anything bad happen if we respond to multiple peers with `ACCEPT`? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the answer is send ACCEPT
to all peers, just wondering what impact that has if any.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah good question. The impact would be that each of the invitors receiving ACCEPT
would write a role record for the invitee. This would be confusing to deal with - we would have duplicate records of a device being added to the project. We should handle this in case it does happen (I've added this to #189).
On reflection I think what would make sense would be to respond ACCEPT
to one peer (the first that we received an invite from?) and send ALREADY
to the others.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In the case of peers disconnecting, we should send ACCEPT
to the first still-connected peer that we find.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good! I've added some responses to your questions. You'll need to coordinate with @achou11 for the API for looking up existing projects and joining a new one.
One issue to handle is disconnecting peers. We should emit an event for this, and return an error from accept
or reject
if trying to respond to a disconnected peer.
I think we should try not to add a project if we are unable to send the ACCEPT
message, since we will not know if the invitor has actually written us to the auth core to actually access the project.
src/invite-api.js
Outdated
/** @typedef {import('./rpc/index.js').MapeoRPC} MapeoRPC */ | ||
|
||
export class InviteApi extends TypedEmitter { | ||
// TODO: are invites persisted beyond this api? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good question.
A previously accepted invite is the same issue as "already on this project". This is what the ALREADY
invite response for. This means that the InviteApi needs to look up which projects the device is already a member of - @achou11 can you suggest the best API for this? We'll need to pass something down to the InviteApi constructor.
If a device is already part of a project and they receive an invite, we should automatically respond with ALREADY
- there is no need for an event since I don't think the UX on the invitee side should show anything. On the invitor side the UX should show a "already invited" UX when receiving an invite response of ALREADY
.
We're not going to persist un-responded invites beyond the lifecycle of the app - they require a connection to the invitor peer to respond anyway.
What to do with previously rejected invites - let's not track rejected invites for now, I've created an issue to deal with this question post-MVP
src/invite-api.js
Outdated
// TODO: I'm not seeing encryption keys used in the inviteResponse in the rpc api | ||
// what is the purpose of the encryption keys at this stage of the process or afterward? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The encryptionKeys and projectKey will be needed in the accept()
method to add the project. @achou11 is working on the API for this, we should pass something down to the constructor here like addProject()
, or an instance of something that has that method.
The keys will need to be stored in the this.#invites
Map so we can look them up when we accept()
. We don't emit()
the keys though (e.g. expose to the API) because we keep these internal at all times.
src/invite-api.js
Outdated
// TODO: should this reply to one peer with `ACCEPT` and the rest with `ALREADY`? | ||
// How is the `ALREADY` decision determined? It looks like that isn't used yet. | ||
// Does anything bad happen if we respond to multiple peers with `ACCEPT`? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah good question. The impact would be that each of the invitors receiving ACCEPT
would write a role record for the invitee. This would be confusing to deal with - we would have duplicate records of a device being added to the project. We should handle this in case it does happen (I've added this to #189).
On reflection I think what would make sense would be to respond ACCEPT
to one peer (the first that we received an invite from?) and send ALREADY
to the others.
src/invite-api.js
Outdated
// TODO: should this reply to one peer with `ACCEPT` and the rest with `ALREADY`? | ||
// How is the `ALREADY` decision determined? It looks like that isn't used yet. | ||
// Does anything bad happen if we respond to multiple peers with `ACCEPT`? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In the case of peers disconnecting, we should send ACCEPT
to the first still-connected peer that we find.
closes #198