Teams & Organizations

Overview

Text, title, and styling in standard markdown.

Supersaas is built with multi-tenancy in mind, using a "Team" or "Organization" structure as the primary way to group users and resources.

Concept

  • Teams as Containers: Each Team acts as an isolated workspace. Users belong to teams, and data associated with your application (e.g., posts, projects, settings) should typically be scoped to a specific team.
  • Ownership: Every team has a single owner. The owner is usually the user who created the team and has the highest level of permissions within that team.
  • Unique Identification: Teams are identified by a unique id (generated using nanoid) and a user-defined slug. The slug is used in URLs for easy access.

Creating a Team

  • UI: New teams are created using the App/NewTeamForm.vue component.
  • Process:
    1. The user provides a Team Name.
    2. The user provides a Team URL Slug (e.g., my-awesome-company). This slug must be unique and follow the specified format (lowercase letters, numbers, single hyphens). The form provides real-time feedback on the final URL structure (e.g., yourdomain.com/dashboard/my-awesome-company).
    3. The user can optionally upload a Team Logo. If no logo is uploaded, a default avatar is generated based on the team name using DiceBear (https://api.dicebear.com/). Uploaded logos are handled by the /api/upload-image endpoint and stored (by default, the path assumes /images/, but you might adjust this based on your storage solution).
    4. On submission, the useTeam().createTeam function calls the POST /api/teams endpoint.
    5. The backend (server/api/teams/index.post.ts) validates the input using createTeamSchema and calls the createTeam database query (server/database/queries/teams.ts).
    6. The createTeam query inserts the team into the teams table and automatically adds the creator to the teamMembers table with the owner role.
  • State Management: Newly created teams are added to the teams state managed by useState<Team[]>('teams'). The useTeamPreferences composable (not fully shown, but referenced) is used to remember the last used team slug.

Team Context and Navigation

  • Current Team: The useTeam() composable provides a currentTeam computed property. This determines the active team based on the :teamSlug route parameter in the URL (e.g., /dashboard/:teamSlug/...).
  • Team Switching: The App/TeamDropdown.vue component (referenced in App/Sidebar/Team.vue) handles the UI for switching between teams the user belongs to. When a team is selected, the user is navigated to the corresponding team's dashboard (/dashboard/:teamSlug).
  • Sidebar: The App/Sidebar/Team.vue component dynamically updates navigation links based on the currentTeam.slug. It also conditionally shows settings links (Workspace Settings, Workspace Members, Billing) only if the current user isTeamOwner.

Managing Teams

  • Updating:
    • Team owners can update basic team settings (like name and logo). This functionality is within the Workspace Settings page linked from the sidebar.
    • The useTeam().updateTeam function handles updates via the PATCH /api/teams/:id endpoint.
    • The backend uses the updateTeam query (server/database/queries/teams.ts). Access control happens within the API route or middleware to ensure only owners (or potentially admins) can update.
  • Deleting:
    • Team deletion is a critical action restricted to the team owner.
    • The App/TeamDelete.vue component provides the UI for this, requiring the owner to type the team name for confirmation.
    • It uses the useTeam().deleteTeam function, which calls the DELETE /api/teams/:id endpoint.
    • The backend API route for deletion must use a mechanism like the validateTeamOwnership utility (server/utils/teamValidation.ts.ts) to ensure only the owner can perform this action. This utility verifies the user's session and checks their role within the target team using findUserTeams.
    • Deleting a team is irreversible and removes the team record and associated data (due to database cascade rules, e.g., teamMembers, teamInvites).

Relevant Files

  • Composables: app/composables/useTeam.ts
  • Components: App/NewTeamForm.vue, App/TeamDelete.vue, App/Sidebar/Team.vue, App/TeamDropdown.vue (implied)
  • API Routes: server/api/teams/index.post.ts, server/api/teams/[id]/index.patch.ts (implied), server/api/teams/[id]/index.delete.ts (implied)
  • Database: server/database/schema/teams.ts (defines teams, teamMembers), server/database/queries/teams.ts (contains createTeam, updateTeam, deleteTeam, findUserTeams)
  • Validation: shared/validations/team.ts (defines createTeamSchema), server/utils/teamValidation.ts
  • Middleware: app/middleware/team-owner.ts (protects owner-specific routes)