+ {{ t('text.connectZoom') }} +
+{{ t('ftue.customVideoMeetingText') }}
+diff --git a/.editorconfig b/.editorconfig index 8e3788746..cb758f52e 100644 --- a/.editorconfig +++ b/.editorconfig @@ -9,7 +9,7 @@ charset = utf-8 indent_style = space indent_size = 4 -[*.{vue,html,css,js,json,yml}] +[*.{vue,html,css,js,json,yml,ts}] charset = utf-8 indent_style = space indent_size = 2 diff --git a/backend/requirements.txt b/backend/requirements.txt index 0d6e417ed..55057f646 100644 --- a/backend/requirements.txt +++ b/backend/requirements.txt @@ -19,7 +19,7 @@ nh3==0.2.17 python-dotenv==1.0.1 python-multipart==0.0.9 PyJWT==2.6.0 -pydantic==2.7.4 +pydantic[email]==2.7.4 sentry-sdk==2.7.1 starlette-context==0.3.6 sqlalchemy-utils==0.41.2 diff --git a/backend/src/appointment/controller/auth.py b/backend/src/appointment/controller/auth.py index 31d6bd6b9..4ce5acb22 100644 --- a/backend/src/appointment/controller/auth.py +++ b/backend/src/appointment/controller/auth.py @@ -7,6 +7,7 @@ import hashlib import hmac import datetime +import urllib.parse from sqlalchemy.orm import Session @@ -47,11 +48,13 @@ def signed_url_by_subscriber(subscriber: schemas.Subscriber): if not short_url: short_url = base_url + url_safe_username = urllib.parse.quote_plus(subscriber.username) + # We sign with a different hash that the end-user doesn't have access to # We also need to use the default url, as short urls are currently setup as a redirect - url = f'{base_url}/{subscriber.username}/{subscriber.short_link_hash}' + url = f'{base_url}/{url_safe_username}/{subscriber.short_link_hash}' signature = sign_url(url) # We return with the signed url signature - return f'{short_url}/{subscriber.username}/{signature}' + return f'{short_url}/{url_safe_username}/{signature}' diff --git a/backend/src/appointment/database/models.py b/backend/src/appointment/database/models.py index 578297d5e..0c437eb65 100644 --- a/backend/src/appointment/database/models.py +++ b/backend/src/appointment/database/models.py @@ -139,6 +139,8 @@ class Subscriber(HasSoftDelete, Base): # Only accept the times greater than the one specified in the `iat` claim of the jwt token minimum_valid_iat_time = Column('minimum_valid_iat_time', encrypted_type(DateTime)) + ftue_level = Column(Integer, nullable=False, default=0, index=True) + calendars = relationship('Calendar', cascade='all,delete', back_populates='owner') slots = relationship('Slot', cascade='all,delete', back_populates='subscriber') external_connections = relationship('ExternalConnections', cascade='all,delete', back_populates='owner') @@ -153,6 +155,11 @@ def preferred_email(self): """Returns the preferred email address.""" return self.secondary_email if self.secondary_email is not None else self.email + @property + def is_setup(self) -> bool: + """Has the user been through the First Time User Experience?""" + return self.ftue_level > 0 + class Calendar(Base): __tablename__ = 'calendars' diff --git a/backend/src/appointment/database/repo/subscriber.py b/backend/src/appointment/database/repo/subscriber.py index cf660f2b4..efd8424b2 100644 --- a/backend/src/appointment/database/repo/subscriber.py +++ b/backend/src/appointment/database/repo/subscriber.py @@ -5,6 +5,7 @@ import re import datetime +import urllib.parse from sqlalchemy.orm import Session from .. import models, schemas @@ -125,6 +126,8 @@ def verify_link(db: Session, url: str): """ username, signature, clean_url = utils.retrieve_user_url_data(url) + username = urllib.parse.unquote_plus(username) + subscriber = get_by_username(db, username) if not subscriber: return False diff --git a/backend/src/appointment/database/schemas.py b/backend/src/appointment/database/schemas.py index 82ab41f7d..f40b752e8 100644 --- a/backend/src/appointment/database/schemas.py +++ b/backend/src/appointment/database/schemas.py @@ -155,7 +155,7 @@ class Config: class ScheduleBase(BaseModel): active: bool | None = True - name: str = Field(min_length=1) + name: str = Field(min_length=1, max_length=128) slug: Optional[str] = None calendar_id: int location_type: LocationType | None = LocationType.inperson @@ -250,15 +250,16 @@ class Invite(BaseModel): class SubscriberIn(BaseModel): timezone: str | None = None - username: str - name: str | None = None + username: str = Field(min_length=1, max_length=128) + name: Optional[str] = Field(min_length=1, max_length=128, default=None) avatar_url: str | None = None secondary_email: str | None = None class SubscriberBase(SubscriberIn): - email: str + email: str = Field(min_length=1, max_length=200) preferred_email: str | None = None + is_setup: bool | None = None level: SubscriberLevel | None = SubscriberLevel.basic @@ -270,6 +271,7 @@ class Subscriber(SubscriberAuth): id: int calendars: list[Calendar] = [] slots: list[Slot] = [] + ftue_level: Optional[int] = Field(gte=0) class Config: from_attributes = True diff --git a/backend/src/appointment/migrations/versions/2024_06_25_2133-156b3b0d77b9_add_ftue_level_to_subscribers.py b/backend/src/appointment/migrations/versions/2024_06_25_2133-156b3b0d77b9_add_ftue_level_to_subscribers.py new file mode 100644 index 000000000..9dd171fa6 --- /dev/null +++ b/backend/src/appointment/migrations/versions/2024_06_25_2133-156b3b0d77b9_add_ftue_level_to_subscribers.py @@ -0,0 +1,24 @@ +"""add ftue_level to subscribers + +Revision ID: 156b3b0d77b9 +Revises: 12c7e1b34dd6 +Create Date: 2024-06-25 21:33:27.094632 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = '156b3b0d77b9' +down_revision = '12c7e1b34dd6' +branch_labels = None +depends_on = None + + +def upgrade() -> None: + op.add_column('subscribers', sa.Column('ftue_level', sa.Integer, default=0, nullable=False, index=True)) + + +def downgrade() -> None: + op.drop_column('subscribers', 'ftue_level') diff --git a/backend/src/appointment/migrations/versions/2024_07_02_1636-0c22678e25db_.py b/backend/src/appointment/migrations/versions/2024_07_02_1636-0c22678e25db_.py new file mode 100644 index 000000000..84fd81e13 --- /dev/null +++ b/backend/src/appointment/migrations/versions/2024_07_02_1636-0c22678e25db_.py @@ -0,0 +1,24 @@ +"""empty message + +Revision ID: 0c22678e25db +Revises: 156b3b0d77b9, a9ca5a4325ec +Create Date: 2024-07-02 16:36:47.372956 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = '0c22678e25db' +down_revision = ('156b3b0d77b9', 'a9ca5a4325ec') +branch_labels = None +depends_on = None + + +def upgrade() -> None: + pass + + +def downgrade() -> None: + pass diff --git a/backend/src/appointment/routes/api.py b/backend/src/appointment/routes/api.py index 3dc4ca560..1fa959721 100644 --- a/backend/src/appointment/routes/api.py +++ b/backend/src/appointment/routes/api.py @@ -66,6 +66,7 @@ def update_me( name=me.name, level=me.level, timezone=me.timezone, + is_setup=me.is_setup, ) diff --git a/backend/src/appointment/routes/auth.py b/backend/src/appointment/routes/auth.py index 1a1084345..c2a975437 100644 --- a/backend/src/appointment/routes/auth.py +++ b/backend/src/appointment/routes/auth.py @@ -292,6 +292,7 @@ def me( level=subscriber.level, timezone=subscriber.timezone, avatar_url=subscriber.avatar_url, + is_setup=subscriber.is_setup, ) diff --git a/backend/src/appointment/routes/schedule.py b/backend/src/appointment/routes/schedule.py index 273664da6..ae551b5e0 100644 --- a/backend/src/appointment/routes/schedule.py +++ b/backend/src/appointment/routes/schedule.py @@ -108,6 +108,14 @@ def update_schedule( and subscriber.get_external_connection(ExternalConnectionType.zoom) is None ): raise validation.ZoomNotConnectedException() + + if schedule.slug is None: + # If slug isn't provided, give them the last 8 characters from a uuid4 + schedule.slug = repo.schedule.generate_slug(db, id) + if not schedule.slug: + # A little extra, but things are a little out of place right now.. + raise validation.ScheduleCreationException() + return repo.schedule.update(db=db, schedule=schedule, schedule_id=id) @@ -435,7 +443,7 @@ def decide_on_schedule_availability_slot( title=title, start=slot.start.replace(tzinfo=timezone.utc), end=slot.start.replace(tzinfo=timezone.utc) + timedelta(minutes=slot.duration), - description=schedule.details, + description=schedule.details or '', location=schemas.EventLocation( type=schedule.location_type, url=location_url, diff --git a/backend/src/appointment/routes/subscriber.py b/backend/src/appointment/routes/subscriber.py index 4825aba04..e16d6fe2f 100644 --- a/backend/src/appointment/routes/subscriber.py +++ b/backend/src/appointment/routes/subscriber.py @@ -4,13 +4,16 @@ from ..database import repo, schemas, models from ..database.models import Subscriber -from ..dependencies.auth import get_admin_subscriber +from ..dependencies.auth import get_admin_subscriber, get_subscriber from ..dependencies.database import get_db from ..exceptions import validation router = APIRouter() +""" ADMIN ROUTES +These require get_admin_subscriber! +""" @router.get('/', response_model=list[schemas.SubscriberAdminOut]) def get_all_subscriber(db: Session = Depends(get_db), _: Subscriber = Depends(get_admin_subscriber)): @@ -47,3 +50,16 @@ def enable_subscriber(email: str, db: Session = Depends(get_db), _: Subscriber = # Set active flag to true on the subscribers model. return repo.subscriber.enable(db, subscriber_to_enable) + + +""" NON-ADMIN ROUTES """ + + +@router.post('/setup') +def subscriber_is_setup(db: Session = Depends(get_db), subscriber: Subscriber = Depends(get_subscriber)): + """Flips ftue_level to 1""" + subscriber.ftue_level = 1 + db.add(subscriber) + db.commit() + + return True diff --git a/docker-compose.yml b/docker-compose.yml index 42c799569..575d886c0 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -19,7 +19,7 @@ services: build: ./frontend volumes: - './frontend:/app' - - '/app/node_modules' + - 'node_modules:/app/node_modules' ports: - 8080:8080 environment: @@ -54,3 +54,4 @@ services: volumes: db: {} cache: {} + node_modules: {} diff --git a/frontend/.eslintrc.cjs b/frontend/.eslintrc.cjs index 95bc8a8b4..8a4d03172 100644 --- a/frontend/.eslintrc.cjs +++ b/frontend/.eslintrc.cjs @@ -6,9 +6,11 @@ module.exports = { browser: true, es2021: true, }, + parser: '@typescript-eslint/parser', extends: [ 'plugin:vue/vue3-essential', 'plugin:tailwindcss/recommended', + 'plugin:@typescript-eslint/recommended', 'airbnb-base', ], overrides: [], @@ -18,6 +20,7 @@ module.exports = { }, plugins: [ 'vue', + '@typescript-eslint', ], rules: { 'import/extensions': ['error', 'ignorePackages', { @@ -31,6 +34,18 @@ module.exports = { 'tailwindcss/no-custom-classname': 'off', 'import/prefer-default-export': 'off', radix: 'off', + '@typescript-eslint/no-explicit-any': 'off', + // Disable full warning, and customize the typescript one + // Warn about unused vars unless they start with an underscore + 'no-unused-vars': 'off', + '@typescript-eslint/no-unused-vars': [ + 'warn', + { + argsIgnorePattern: '^_', + varsIgnorePattern: '^_', + caughtErrorsIgnorePattern: '^_', + }, + ], }, settings: { 'import/resolver': { diff --git a/frontend/README.md b/frontend/README.md index 32ac9d0ae..18d155625 100644 --- a/frontend/README.md +++ b/frontend/README.md @@ -24,6 +24,19 @@ yarn run serve yarn run build ``` +### Post-CSS + +We use post-css to enhance our css. Any post-css that isn't in a SFC must be in a `.pcss` file and imported into the scoped style like so: +```css +@import '@/assets/styles/custom-media.pcss'; + +@media (--md) { + .container { + ... + } +} +``` + ### Lints and fixes files Frontend is formatted using ESlint with airbnb rules. diff --git a/frontend/index.html b/frontend/index.html index 3e4d6e951..b2de56838 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -6,10 +6,12 @@
+ {{ t('text.connectZoom') }} +
+{{ t('ftue.customVideoMeetingText') }}
+{{ t('ftue.finishScreenText') }}
+{{ t('text.googlePermissionDisclaimer') }}
+