Sharing data through Organizations

Organizations are a way to share a set of data between users. Different apps have different names for this concept, such as "teams" or "workspaces".

We'll use the term Organization.

See the full example here.

Defining the schema for an Organization

Create a CoMap shared by the users of the same organization to act as a root (or "main database") for the shared data within an organization.

For this example, users within an Organization will be sharing Projects.

// schema.ts export class Project extends CoMap { name = co.string; } export class ListOfProjects extends CoList.Of(co.ref(Project)) {} export class Organization extends CoMap { name = co.string; // shared data between users of each organization projects = co.ref(ListOfProjects); } export class ListOfOrganizations extends CoList.Of(co.ref(Organization)) {}

Learn more about defining schemas.

Adding a list of Organizations to the user's Account

Let's add the list of Organizations to the user's Account root so they can access them.

// schema.ts export class JazzAccountRoot extends CoMap { organizations = co.ref(ListOfOrganizations); } export class JazzAccount extends Account { root = co.ref(JazzAccountRoot); async migrate() { if (this.root === undefined) { // Using a Group as an owner allows you to give access to other users const organizationGroup = Group.create(); const organizations = ListOfOrganizations.create( [ // Create the first Organization so users can start right away Organization.create( { name: "My organization", projects: ListOfProjects.create([], organizationGroup), }, organizationGroup, ), ], ); this.root = JazzAccountRoot.create( { organizations }, ); } } }

This schema now allows users to create Organizations and add Projects to them.

See the schema for the example app here.

Adding other users to an Organization

To give users access to an Organization, you can either send them an invite link, or add their Account manually.

Here's how you can generate an invite link.

When the user accepts the invite, add the Organization to the user's organizations list.

const onAccept = async (organizationId: ID<Organization>) => { const me = await MusicaAccount.getMe().ensureLoaded({ root: { organizations: [], }, }); if (!me) throw new Error("Failed to load account data"); const organization = await Organization.load(organizationId, []); if (!organization) throw new Error("Failed to load organization data"); const ids = me.root.organizations.map( (organization) => organization?.id, ); if (ids.includes(organizationId)) return; me.root.organizations.push(organization); navigate("/organizations/" + organizationId); }; useAcceptInvite({ invitedObjectSchema: Organization, onAccept, });

Adding users through their Account ID

...more on this coming soon