mirror of
https://github.com/Monadical-SAS/reflector.git
synced 2025-12-21 12:49:06 +00:00
Compare commits
11 Commits
v0.3.0
...
codingagen
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
47cb75ee39 | ||
| db3beae5cd | |||
|
|
03b9a18c1b | ||
|
|
7e3027adb6 | ||
|
|
27b43d85ab | ||
| 2289a1a231 | |||
| d0e130eb13 | |||
| 24fabe3e86 | |||
| 6fedbbe63f | |||
| b39175cdc9 | |||
| 2a2af5fff2 |
29
CHANGELOG.md
29
CHANGELOG.md
@@ -1,5 +1,34 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## [0.4.0](https://github.com/Monadical-SAS/reflector/compare/v0.3.2...v0.4.0) (2025-07-25)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* Diarization cli ([#509](https://github.com/Monadical-SAS/reflector/issues/509)) ([ffc8003](https://github.com/Monadical-SAS/reflector/commit/ffc8003e6dad236930a27d0fe3e2f2adfb793890))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* remove faulty import Meeting ([#512](https://github.com/Monadical-SAS/reflector/issues/512)) ([0e68c79](https://github.com/Monadical-SAS/reflector/commit/0e68c798434e1b481f9482cc3a4702ea00365df4))
|
||||||
|
* room concurrency (theoretically) ([#511](https://github.com/Monadical-SAS/reflector/issues/511)) ([7bb3676](https://github.com/Monadical-SAS/reflector/commit/7bb367653afeb2778cff697a0eb217abf0b81b84))
|
||||||
|
|
||||||
|
## [0.3.2](https://github.com/Monadical-SAS/reflector/compare/v0.3.1...v0.3.2) (2025-07-22)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* match font size for the filter sidebar ([#507](https://github.com/Monadical-SAS/reflector/issues/507)) ([4b8ba5d](https://github.com/Monadical-SAS/reflector/commit/4b8ba5db1733557e27b098ad3d1cdecadf97ae52))
|
||||||
|
* whereby consent not displaying ([#505](https://github.com/Monadical-SAS/reflector/issues/505)) ([1120552](https://github.com/Monadical-SAS/reflector/commit/1120552c2c83d084d3a39272ad49b6aeda1af98f))
|
||||||
|
|
||||||
|
## [0.3.1](https://github.com/Monadical-SAS/reflector/compare/v0.3.0...v0.3.1) (2025-07-22)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* remove fief out of the source code ([#502](https://github.com/Monadical-SAS/reflector/issues/502)) ([890dd15](https://github.com/Monadical-SAS/reflector/commit/890dd15ba5a2be10dbb841e9aeb75d377885f4af))
|
||||||
|
* remove primary color for room action menu ([#504](https://github.com/Monadical-SAS/reflector/issues/504)) ([2e33f89](https://github.com/Monadical-SAS/reflector/commit/2e33f89c0f9e5fbaafa80e8d2ae9788450ea2f31))
|
||||||
|
|
||||||
## [0.3.0](https://github.com/Monadical-SAS/reflector/compare/v0.2.1...v0.3.0) (2025-07-21)
|
## [0.3.0](https://github.com/Monadical-SAS/reflector/compare/v0.2.1...v0.3.0) (2025-07-21)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -146,7 +146,7 @@ All endpoints prefixed `/v1/`:
|
|||||||
- `REDIS_URL` - Redis broker for Celery
|
- `REDIS_URL` - Redis broker for Celery
|
||||||
- `MODAL_TOKEN_ID`, `MODAL_TOKEN_SECRET` - Modal.com GPU processing
|
- `MODAL_TOKEN_ID`, `MODAL_TOKEN_SECRET` - Modal.com GPU processing
|
||||||
- `WHEREBY_API_KEY` - Video platform integration
|
- `WHEREBY_API_KEY` - Video platform integration
|
||||||
- `REFLECTOR_AUTH_BACKEND` - Authentication method (none, fief, jwt)
|
- `REFLECTOR_AUTH_BACKEND` - Authentication method (none, jwt)
|
||||||
|
|
||||||
**Frontend** (`www/.env`):
|
**Frontend** (`www/.env`):
|
||||||
- `NEXTAUTH_URL`, `NEXTAUTH_SECRET` - Authentication configuration
|
- `NEXTAUTH_URL`, `NEXTAUTH_SECRET` - Authentication configuration
|
||||||
|
|||||||
@@ -6,11 +6,6 @@ LLM_BACKEND=modal
|
|||||||
LLM_URL=https://monadical-sas--reflector-llm-web.modal.run
|
LLM_URL=https://monadical-sas--reflector-llm-web.modal.run
|
||||||
LLM_MODAL_API_KEY=***REMOVED***
|
LLM_MODAL_API_KEY=***REMOVED***
|
||||||
|
|
||||||
AUTH_BACKEND=fief
|
|
||||||
AUTH_FIEF_URL=https://auth.reflector.media/reflector-local
|
|
||||||
AUTH_FIEF_CLIENT_ID=***REMOVED***
|
|
||||||
AUTH_FIEF_CLIENT_SECRET=<ask in zulip> <-----------------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
TRANSLATE_URL=https://monadical-sas--reflector-translator-web.modal.run
|
TRANSLATE_URL=https://monadical-sas--reflector-translator-web.modal.run
|
||||||
ZEPHYR_LLM_URL=https://monadical-sas--reflector-llm-zephyr-web.modal.run
|
ZEPHYR_LLM_URL=https://monadical-sas--reflector-llm-zephyr-web.modal.run
|
||||||
DIARIZATION_URL=https://monadical-sas--reflector-diarizer-web.modal.run
|
DIARIZATION_URL=https://monadical-sas--reflector-diarizer-web.modal.run
|
||||||
|
|||||||
1
server/.gitignore
vendored
1
server/.gitignore
vendored
@@ -180,3 +180,4 @@ reflector.sqlite3
|
|||||||
data/
|
data/
|
||||||
|
|
||||||
dump.rdb
|
dump.rdb
|
||||||
|
|
||||||
|
|||||||
@@ -7,11 +7,9 @@
|
|||||||
## User authentication
|
## User authentication
|
||||||
## =======================================================
|
## =======================================================
|
||||||
|
|
||||||
## Using fief (fief.dev)
|
## Using jwt/authentik
|
||||||
AUTH_BACKEND=fief
|
AUTH_BACKEND=jwt
|
||||||
AUTH_FIEF_URL=https://auth.reflector.media/reflector-local
|
AUTH_JWT_AUDIENCE=
|
||||||
AUTH_FIEF_CLIENT_ID=***REMOVED***
|
|
||||||
AUTH_FIEF_CLIENT_SECRET=<ask in zulip>
|
|
||||||
|
|
||||||
## =======================================================
|
## =======================================================
|
||||||
## Transcription backend
|
## Transcription backend
|
||||||
@@ -88,4 +86,3 @@ DIARIZATION_URL=https://monadical-sas--reflector-diarizer-web.modal.run
|
|||||||
|
|
||||||
## Sentry DSN configuration
|
## Sentry DSN configuration
|
||||||
#SENTRY_DSN=
|
#SENTRY_DSN=
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,26 @@
|
|||||||
|
"""add_room_background_information
|
||||||
|
|
||||||
|
Revision ID: 082fa608201c
|
||||||
|
Revises: b7df9609542c
|
||||||
|
Create Date: 2025-07-29 01:41:37.912195
|
||||||
|
|
||||||
|
"""
|
||||||
|
from typing import Sequence, Union
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision: str = '082fa608201c'
|
||||||
|
down_revision: Union[str, None] = 'b7df9609542c'
|
||||||
|
branch_labels: Union[str, Sequence[str], None] = None
|
||||||
|
depends_on: Union[str, Sequence[str], None] = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade() -> None:
|
||||||
|
op.add_column('room', sa.Column('background_information', sa.Text(), nullable=True))
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade() -> None:
|
||||||
|
op.drop_column('room', 'background_information')
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
"""add_unique_constraint_one_active_meeting_per_room
|
||||||
|
|
||||||
|
Revision ID: b7df9609542c
|
||||||
|
Revises: d7fbb74b673b
|
||||||
|
Create Date: 2025-07-25 16:27:06.959868
|
||||||
|
|
||||||
|
"""
|
||||||
|
from typing import Sequence, Union
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision: str = 'b7df9609542c'
|
||||||
|
down_revision: Union[str, None] = 'd7fbb74b673b'
|
||||||
|
branch_labels: Union[str, Sequence[str], None] = None
|
||||||
|
depends_on: Union[str, Sequence[str], None] = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade() -> None:
|
||||||
|
# Create a partial unique index that ensures only one active meeting per room
|
||||||
|
# This works for both PostgreSQL and SQLite
|
||||||
|
op.create_index(
|
||||||
|
'idx_one_active_meeting_per_room',
|
||||||
|
'meeting',
|
||||||
|
['room_id'],
|
||||||
|
unique=True,
|
||||||
|
postgresql_where=sa.text('is_active = true'),
|
||||||
|
sqlite_where=sa.text('is_active = 1')
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade() -> None:
|
||||||
|
op.drop_index('idx_one_active_meeting_per_room', table_name='meeting')
|
||||||
@@ -22,7 +22,6 @@ dependencies = [
|
|||||||
"fastapi-pagination>=0.12.6",
|
"fastapi-pagination>=0.12.6",
|
||||||
"databases[aiosqlite, asyncpg]>=0.7.0",
|
"databases[aiosqlite, asyncpg]>=0.7.0",
|
||||||
"sqlalchemy<1.5",
|
"sqlalchemy<1.5",
|
||||||
"fief-client[fastapi]>=0.17.0",
|
|
||||||
"alembic>=1.11.3",
|
"alembic>=1.11.3",
|
||||||
"nltk>=3.8.1",
|
"nltk>=3.8.1",
|
||||||
"prometheus-fastapi-instrumentator>=6.1.0",
|
"prometheus-fastapi-instrumentator>=6.1.0",
|
||||||
|
|||||||
@@ -1,25 +0,0 @@
|
|||||||
from fastapi.security import OAuth2AuthorizationCodeBearer
|
|
||||||
from fief_client import FiefAccessTokenInfo, FiefAsync, FiefUserInfo
|
|
||||||
from fief_client.integrations.fastapi import FiefAuth
|
|
||||||
from reflector.settings import settings
|
|
||||||
|
|
||||||
fief = FiefAsync(
|
|
||||||
settings.AUTH_FIEF_URL,
|
|
||||||
settings.AUTH_FIEF_CLIENT_ID,
|
|
||||||
settings.AUTH_FIEF_CLIENT_SECRET,
|
|
||||||
)
|
|
||||||
|
|
||||||
scheme = OAuth2AuthorizationCodeBearer(
|
|
||||||
f"{settings.AUTH_FIEF_URL}/authorize",
|
|
||||||
f"{settings.AUTH_FIEF_URL}/api/token",
|
|
||||||
scopes={"openid": "openid", "offline_access": "offline_access"},
|
|
||||||
auto_error=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
auth = FiefAuth(fief, scheme)
|
|
||||||
|
|
||||||
UserInfo = FiefUserInfo
|
|
||||||
AccessTokenInfo = FiefAccessTokenInfo
|
|
||||||
authenticated = auth.authenticated()
|
|
||||||
current_user = auth.current_user()
|
|
||||||
current_user_optional = auth.current_user(optional=True)
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
from reflector.db import database
|
|
||||||
from reflector.db.meetings import meetings
|
|
||||||
from reflector.db.rooms import rooms
|
|
||||||
from reflector.db.transcripts import transcripts
|
|
||||||
|
|
||||||
users_to_migrate = [
|
|
||||||
["123@lifex.pink", "63b727f5-485d-449f-b528-563d779b11ef", None],
|
|
||||||
["ana@monadical.com", "1bae2e4d-5c04-49c2-932f-a86266a6ca13", None],
|
|
||||||
["cspencer@sprocket.org", "614ed0be-392e-488c-bd19-6a9730fd0e9e", None],
|
|
||||||
["daniel.f.lopez.j@gmail.com", "ca9561bd-c989-4a1e-8877-7081cf62ae7f", None],
|
|
||||||
["jenalee@monadical.com", "c7c1e79e-b068-4b28-a9f4-29d98b1697ed", None],
|
|
||||||
["jennifer@rootandseed.com", "f5321727-7546-4b2b-b69d-095a931ef0c4", None],
|
|
||||||
["jose@monadical.com", "221f079c-7ce0-4677-90b7-0359b6315e27", None],
|
|
||||||
["labenclayton@gmail.com", "40078cd0-543c-40e4-9c2e-5ce57a686428", None],
|
|
||||||
["mathieu@monadical.com", "c7a36151-851e-4afa-9fab-aaca834bfd30", None],
|
|
||||||
["michal.flak.96@gmail.com", "3096eb5e-b590-41fc-a0d1-d152c1895402", None],
|
|
||||||
["sara@monadical.com", "31ab0cfe-5d2c-4c7a-84de-a29494714c99", None],
|
|
||||||
["sara@monadical.com", "b871e5f0-754e-447f-9c3d-19f629f0082b", None],
|
|
||||||
["sebastian@monadical.com", "f024f9d0-15d0-480f-8529-43959fc8b639", None],
|
|
||||||
["sergey@monadical.com", "5c4798eb-b9ab-4721-a540-bd96fc434156", None],
|
|
||||||
["sergey@monadical.com", "9dd8a6b4-247e-48fe-b1fb-4c84dd3c01bc", None],
|
|
||||||
["transient.tran@gmail.com", "617ba2d3-09b6-4b1f-a435-a7f41c3ce060", None],
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
async def migrate_user(email, user_id):
|
|
||||||
# if the email match the email in the users_to_migrate list
|
|
||||||
# reassign all transcripts/rooms/meetings to the new user_id
|
|
||||||
|
|
||||||
user_ids = [user[1] for user in users_to_migrate if user[0] == email]
|
|
||||||
if not user_ids:
|
|
||||||
return
|
|
||||||
|
|
||||||
# do not migrate back
|
|
||||||
if user_id in user_ids:
|
|
||||||
return
|
|
||||||
|
|
||||||
for old_user_id in user_ids:
|
|
||||||
query = (
|
|
||||||
transcripts.update()
|
|
||||||
.where(transcripts.c.user_id == old_user_id)
|
|
||||||
.values(user_id=user_id)
|
|
||||||
)
|
|
||||||
await database.execute(query)
|
|
||||||
|
|
||||||
query = (
|
|
||||||
rooms.update().where(rooms.c.user_id == old_user_id).values(user_id=user_id)
|
|
||||||
)
|
|
||||||
await database.execute(query)
|
|
||||||
|
|
||||||
query = (
|
|
||||||
meetings.update()
|
|
||||||
.where(meetings.c.user_id == old_user_id)
|
|
||||||
.values(user_id=user_id)
|
|
||||||
)
|
|
||||||
await database.execute(query)
|
|
||||||
@@ -39,6 +39,7 @@ rooms = sqlalchemy.Table(
|
|||||||
sqlalchemy.Column(
|
sqlalchemy.Column(
|
||||||
"is_shared", sqlalchemy.Boolean, nullable=False, server_default=false()
|
"is_shared", sqlalchemy.Boolean, nullable=False, server_default=false()
|
||||||
),
|
),
|
||||||
|
sqlalchemy.Column("background_information", sqlalchemy.Text),
|
||||||
sqlalchemy.Index("idx_room_is_shared", "is_shared"),
|
sqlalchemy.Index("idx_room_is_shared", "is_shared"),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -58,6 +59,7 @@ class Room(BaseModel):
|
|||||||
"none", "prompt", "automatic", "automatic-2nd-participant"
|
"none", "prompt", "automatic", "automatic-2nd-participant"
|
||||||
] = "automatic-2nd-participant"
|
] = "automatic-2nd-participant"
|
||||||
is_shared: bool = False
|
is_shared: bool = False
|
||||||
|
background_information: str = ""
|
||||||
|
|
||||||
|
|
||||||
class RoomController:
|
class RoomController:
|
||||||
@@ -106,6 +108,7 @@ class RoomController:
|
|||||||
recording_type: str,
|
recording_type: str,
|
||||||
recording_trigger: str,
|
recording_trigger: str,
|
||||||
is_shared: bool,
|
is_shared: bool,
|
||||||
|
background_information: str = "",
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
Add a new room
|
Add a new room
|
||||||
@@ -121,6 +124,7 @@ class RoomController:
|
|||||||
recording_type=recording_type,
|
recording_type=recording_type,
|
||||||
recording_trigger=recording_trigger,
|
recording_trigger=recording_trigger,
|
||||||
is_shared=is_shared,
|
is_shared=is_shared,
|
||||||
|
background_information=background_information,
|
||||||
)
|
)
|
||||||
query = rooms.insert().values(**room.model_dump())
|
query = rooms.insert().values(**room.model_dump())
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -454,15 +454,47 @@ class PipelineMainFinalSummaries(PipelineMainFromTopics):
|
|||||||
Generate summaries from the topics
|
Generate summaries from the topics
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
async def get_room(self):
|
||||||
|
"""Get room information for the transcript"""
|
||||||
|
if not self._transcript.room_id:
|
||||||
|
return None
|
||||||
|
return await rooms_controller.get_by_id(self._transcript.room_id)
|
||||||
|
|
||||||
def get_processors(self) -> list:
|
def get_processors(self) -> list:
|
||||||
return [
|
return [
|
||||||
TranscriptFinalSummaryProcessor.as_threaded(
|
TranscriptFinalSummaryProcessor.as_threaded(
|
||||||
transcript=self._transcript,
|
transcript=self._transcript,
|
||||||
|
room=getattr(self, '_room', None),
|
||||||
callback=self.on_long_summary,
|
callback=self.on_long_summary,
|
||||||
on_short_summary=self.on_short_summary,
|
on_short_summary=self.on_short_summary,
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
async def create(self) -> Pipeline:
|
||||||
|
self.prepare()
|
||||||
|
|
||||||
|
# get transcript
|
||||||
|
self._transcript = transcript = await self.get_transcript()
|
||||||
|
|
||||||
|
# get room information
|
||||||
|
self._room = await self.get_room()
|
||||||
|
|
||||||
|
# create pipeline
|
||||||
|
processors = self.get_processors()
|
||||||
|
pipeline = Pipeline(*processors)
|
||||||
|
pipeline.options = self
|
||||||
|
pipeline.logger.bind(transcript_id=transcript.id)
|
||||||
|
pipeline.logger.info(f"{self.__class__.__name__} pipeline created")
|
||||||
|
|
||||||
|
# push topics
|
||||||
|
topics = self.get_transcript_topics(transcript)
|
||||||
|
for topic in topics:
|
||||||
|
await self.push(topic)
|
||||||
|
|
||||||
|
await self.flush()
|
||||||
|
|
||||||
|
return pipeline
|
||||||
|
|
||||||
|
|
||||||
class PipelineMainWaveform(PipelineMainFromTopics):
|
class PipelineMainWaveform(PipelineMainFromTopics):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -135,7 +135,7 @@ class Messages:
|
|||||||
|
|
||||||
|
|
||||||
class SummaryBuilder:
|
class SummaryBuilder:
|
||||||
def __init__(self, llm, filename: str | None = None, logger=None):
|
def __init__(self, llm, filename: str | None = None, logger=None, room=None):
|
||||||
self.transcript: str | None = None
|
self.transcript: str | None = None
|
||||||
self.recap: str | None = None
|
self.recap: str | None = None
|
||||||
self.summaries: list[dict] = []
|
self.summaries: list[dict] = []
|
||||||
@@ -147,6 +147,7 @@ class SummaryBuilder:
|
|||||||
self.llm_instance: LLM = llm
|
self.llm_instance: LLM = llm
|
||||||
self.model_name: str = llm.model_name
|
self.model_name: str = llm.model_name
|
||||||
self.logger = logger or structlog.get_logger()
|
self.logger = logger or structlog.get_logger()
|
||||||
|
self.room = room
|
||||||
self.m = Messages(model_name=self.model_name, logger=self.logger)
|
self.m = Messages(model_name=self.model_name, logger=self.logger)
|
||||||
if filename:
|
if filename:
|
||||||
self.read_transcript_from_file(filename)
|
self.read_transcript_from_file(filename)
|
||||||
@@ -465,8 +466,8 @@ class SummaryBuilder:
|
|||||||
self.logger.debug("--- extract main subjects")
|
self.logger.debug("--- extract main subjects")
|
||||||
|
|
||||||
m = Messages(model_name=self.model_name, logger=self.logger)
|
m = Messages(model_name=self.model_name, logger=self.logger)
|
||||||
m.add_system(
|
|
||||||
(
|
system_prompt = (
|
||||||
"You are an advanced transcription summarization assistant."
|
"You are an advanced transcription summarization assistant."
|
||||||
"Your task is to summarize discussions by focusing only on the main ideas contributed by participants."
|
"Your task is to summarize discussions by focusing only on the main ideas contributed by participants."
|
||||||
# Prevent generating another transcription
|
# Prevent generating another transcription
|
||||||
@@ -484,7 +485,12 @@ class SummaryBuilder:
|
|||||||
# Avoid finishing the summary with "No conclusions were added by the summarizer"
|
# Avoid finishing the summary with "No conclusions were added by the summarizer"
|
||||||
"Do not mention conclusion if there is no conclusion"
|
"Do not mention conclusion if there is no conclusion"
|
||||||
)
|
)
|
||||||
)
|
|
||||||
|
# Add room context if available
|
||||||
|
if self.room and self.room.background_information:
|
||||||
|
system_prompt += f"\n\nContext about this meeting room: {self.room.background_information}"
|
||||||
|
|
||||||
|
m.add_system(system_prompt)
|
||||||
m.add_user(
|
m.add_user(
|
||||||
f"# Transcript\n\n{self.transcript}\n\n"
|
f"# Transcript\n\n{self.transcript}\n\n"
|
||||||
+ (
|
+ (
|
||||||
|
|||||||
@@ -12,9 +12,10 @@ class TranscriptFinalSummaryProcessor(Processor):
|
|||||||
INPUT_TYPE = TitleSummary
|
INPUT_TYPE = TitleSummary
|
||||||
OUTPUT_TYPE = FinalLongSummary
|
OUTPUT_TYPE = FinalLongSummary
|
||||||
|
|
||||||
def __init__(self, transcript=None, **kwargs):
|
def __init__(self, transcript=None, room=None, **kwargs):
|
||||||
super().__init__(**kwargs)
|
super().__init__(**kwargs)
|
||||||
self.transcript = transcript
|
self.transcript = transcript
|
||||||
|
self.room = room
|
||||||
self.chunks: list[TitleSummary] = []
|
self.chunks: list[TitleSummary] = []
|
||||||
self.llm = LLM.get_instance(model_name="NousResearch/Hermes-3-Llama-3.1-8B")
|
self.llm = LLM.get_instance(model_name="NousResearch/Hermes-3-Llama-3.1-8B")
|
||||||
self.builder = None
|
self.builder = None
|
||||||
@@ -23,7 +24,7 @@ class TranscriptFinalSummaryProcessor(Processor):
|
|||||||
self.chunks.append(data)
|
self.chunks.append(data)
|
||||||
|
|
||||||
async def get_summary_builder(self, text) -> SummaryBuilder:
|
async def get_summary_builder(self, text) -> SummaryBuilder:
|
||||||
builder = SummaryBuilder(self.llm)
|
builder = SummaryBuilder(self.llm, room=self.room)
|
||||||
builder.set_transcript(text)
|
builder.set_transcript(text)
|
||||||
await builder.identify_participants()
|
await builder.identify_participants()
|
||||||
await builder.generate_summary()
|
await builder.generate_summary()
|
||||||
|
|||||||
@@ -90,14 +90,9 @@ class Settings(BaseSettings):
|
|||||||
# Sentry
|
# Sentry
|
||||||
SENTRY_DSN: str | None = None
|
SENTRY_DSN: str | None = None
|
||||||
|
|
||||||
# User authentication (none, fief)
|
# User authentication (none, jwt)
|
||||||
AUTH_BACKEND: str = "none"
|
AUTH_BACKEND: str = "none"
|
||||||
|
|
||||||
# User authentication using fief
|
|
||||||
AUTH_FIEF_URL: str | None = None
|
|
||||||
AUTH_FIEF_CLIENT_ID: str | None = None
|
|
||||||
AUTH_FIEF_CLIENT_SECRET: str | None = None
|
|
||||||
|
|
||||||
# User authentication using JWT
|
# User authentication using JWT
|
||||||
AUTH_JWT_ALGORITHM: str = "RS256"
|
AUTH_JWT_ALGORITHM: str = "RS256"
|
||||||
AUTH_JWT_PUBLIC_KEY: str | None = "authentik.monadical.com_public.pem"
|
AUTH_JWT_PUBLIC_KEY: str | None = "authentik.monadical.com_public.pem"
|
||||||
|
|||||||
314
server/reflector/tools/process_with_diarization.py
Normal file
314
server/reflector/tools/process_with_diarization.py
Normal file
@@ -0,0 +1,314 @@
|
|||||||
|
"""
|
||||||
|
@vibe-generated
|
||||||
|
Process audio file with diarization support
|
||||||
|
===========================================
|
||||||
|
|
||||||
|
Extended version of process.py that includes speaker diarization.
|
||||||
|
This tool processes audio files locally without requiring the full server infrastructure.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import tempfile
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import List
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
import av
|
||||||
|
from reflector.logger import logger
|
||||||
|
from reflector.processors import (
|
||||||
|
AudioChunkerProcessor,
|
||||||
|
AudioMergeProcessor,
|
||||||
|
AudioTranscriptAutoProcessor,
|
||||||
|
AudioFileWriterProcessor,
|
||||||
|
Pipeline,
|
||||||
|
PipelineEvent,
|
||||||
|
TranscriptFinalSummaryProcessor,
|
||||||
|
TranscriptFinalTitleProcessor,
|
||||||
|
TranscriptLinerProcessor,
|
||||||
|
TranscriptTopicDetectorProcessor,
|
||||||
|
TranscriptTranslatorProcessor,
|
||||||
|
)
|
||||||
|
from reflector.processors.base import BroadcastProcessor, Processor
|
||||||
|
from reflector.processors.types import (
|
||||||
|
AudioDiarizationInput,
|
||||||
|
TitleSummary,
|
||||||
|
TitleSummaryWithId,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TopicCollectorProcessor(Processor):
|
||||||
|
"""Collect topics for diarization"""
|
||||||
|
|
||||||
|
INPUT_TYPE = TitleSummary
|
||||||
|
OUTPUT_TYPE = TitleSummary
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
super().__init__(**kwargs)
|
||||||
|
self.topics: List[TitleSummaryWithId] = []
|
||||||
|
self._topic_id = 0
|
||||||
|
|
||||||
|
async def _push(self, data: TitleSummary):
|
||||||
|
# Convert to TitleSummaryWithId and collect
|
||||||
|
self._topic_id += 1
|
||||||
|
topic_with_id = TitleSummaryWithId(
|
||||||
|
id=str(self._topic_id),
|
||||||
|
title=data.title,
|
||||||
|
summary=data.summary,
|
||||||
|
timestamp=data.timestamp,
|
||||||
|
duration=data.duration,
|
||||||
|
transcript=data.transcript,
|
||||||
|
)
|
||||||
|
self.topics.append(topic_with_id)
|
||||||
|
|
||||||
|
# Pass through the original topic
|
||||||
|
await self.emit(data)
|
||||||
|
|
||||||
|
def get_topics(self) -> List[TitleSummaryWithId]:
|
||||||
|
return self.topics
|
||||||
|
|
||||||
|
|
||||||
|
async def process_audio_file_with_diarization(
|
||||||
|
filename,
|
||||||
|
event_callback,
|
||||||
|
only_transcript=False,
|
||||||
|
source_language="en",
|
||||||
|
target_language="en",
|
||||||
|
enable_diarization=True,
|
||||||
|
diarization_backend="modal",
|
||||||
|
):
|
||||||
|
# Create temp file for audio if diarization is enabled
|
||||||
|
audio_temp_path = None
|
||||||
|
if enable_diarization:
|
||||||
|
audio_temp_file = tempfile.NamedTemporaryFile(suffix=".wav", delete=False)
|
||||||
|
audio_temp_path = audio_temp_file.name
|
||||||
|
audio_temp_file.close()
|
||||||
|
|
||||||
|
# Create processor for collecting topics
|
||||||
|
topic_collector = TopicCollectorProcessor()
|
||||||
|
|
||||||
|
# Build pipeline for audio processing
|
||||||
|
processors = []
|
||||||
|
|
||||||
|
# Add audio file writer at the beginning if diarization is enabled
|
||||||
|
if enable_diarization:
|
||||||
|
processors.append(AudioFileWriterProcessor(audio_temp_path))
|
||||||
|
|
||||||
|
# Add the rest of the processors
|
||||||
|
processors += [
|
||||||
|
AudioChunkerProcessor(),
|
||||||
|
AudioMergeProcessor(),
|
||||||
|
AudioTranscriptAutoProcessor.as_threaded(),
|
||||||
|
]
|
||||||
|
|
||||||
|
processors += [
|
||||||
|
TranscriptLinerProcessor(),
|
||||||
|
TranscriptTranslatorProcessor.as_threaded(),
|
||||||
|
]
|
||||||
|
|
||||||
|
if not only_transcript:
|
||||||
|
processors += [
|
||||||
|
TranscriptTopicDetectorProcessor.as_threaded(),
|
||||||
|
# Collect topics for diarization
|
||||||
|
topic_collector,
|
||||||
|
BroadcastProcessor(
|
||||||
|
processors=[
|
||||||
|
TranscriptFinalTitleProcessor.as_threaded(),
|
||||||
|
TranscriptFinalSummaryProcessor.as_threaded(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
]
|
||||||
|
|
||||||
|
# Create main pipeline
|
||||||
|
pipeline = Pipeline(*processors)
|
||||||
|
pipeline.set_pref("audio:source_language", source_language)
|
||||||
|
pipeline.set_pref("audio:target_language", target_language)
|
||||||
|
pipeline.describe()
|
||||||
|
pipeline.on(event_callback)
|
||||||
|
|
||||||
|
# Start processing audio
|
||||||
|
logger.info(f"Opening {filename}")
|
||||||
|
container = av.open(filename)
|
||||||
|
try:
|
||||||
|
logger.info("Start pushing audio into the pipeline")
|
||||||
|
for frame in container.decode(audio=0):
|
||||||
|
await pipeline.push(frame)
|
||||||
|
finally:
|
||||||
|
logger.info("Flushing the pipeline")
|
||||||
|
await pipeline.flush()
|
||||||
|
|
||||||
|
# Run diarization if enabled and we have topics
|
||||||
|
if enable_diarization and not only_transcript and audio_temp_path:
|
||||||
|
topics = topic_collector.get_topics()
|
||||||
|
|
||||||
|
if topics:
|
||||||
|
logger.info(f"Starting diarization with {len(topics)} topics")
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Import diarization processor
|
||||||
|
from reflector.processors import AudioDiarizationAutoProcessor
|
||||||
|
|
||||||
|
# Create diarization processor
|
||||||
|
diarization_processor = AudioDiarizationAutoProcessor(
|
||||||
|
name=diarization_backend
|
||||||
|
)
|
||||||
|
diarization_processor.on(event_callback)
|
||||||
|
|
||||||
|
# For Modal backend, we need to upload the file to S3 first
|
||||||
|
if diarization_backend == "modal":
|
||||||
|
from reflector.storage import get_transcripts_storage
|
||||||
|
from reflector.utils.s3_temp_file import S3TemporaryFile
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
storage = get_transcripts_storage()
|
||||||
|
|
||||||
|
# Generate a unique filename in evaluation folder
|
||||||
|
timestamp = datetime.utcnow().strftime("%Y%m%d_%H%M%S")
|
||||||
|
audio_filename = f"evaluation/diarization_temp/{timestamp}_{uuid.uuid4().hex}.wav"
|
||||||
|
|
||||||
|
# Use context manager for automatic cleanup
|
||||||
|
async with S3TemporaryFile(storage, audio_filename) as s3_file:
|
||||||
|
# Read and upload the audio file
|
||||||
|
with open(audio_temp_path, "rb") as f:
|
||||||
|
audio_data = f.read()
|
||||||
|
|
||||||
|
audio_url = await s3_file.upload(audio_data)
|
||||||
|
logger.info(f"Uploaded audio to S3: {audio_filename}")
|
||||||
|
|
||||||
|
# Create diarization input with S3 URL
|
||||||
|
diarization_input = AudioDiarizationInput(
|
||||||
|
audio_url=audio_url, topics=topics
|
||||||
|
)
|
||||||
|
|
||||||
|
# Run diarization
|
||||||
|
await diarization_processor.push(diarization_input)
|
||||||
|
await diarization_processor.flush()
|
||||||
|
|
||||||
|
logger.info("Diarization complete")
|
||||||
|
# File will be automatically cleaned up when exiting the context
|
||||||
|
else:
|
||||||
|
# For local backend, use local file path
|
||||||
|
audio_url = audio_temp_path
|
||||||
|
|
||||||
|
# Create diarization input
|
||||||
|
diarization_input = AudioDiarizationInput(
|
||||||
|
audio_url=audio_url, topics=topics
|
||||||
|
)
|
||||||
|
|
||||||
|
# Run diarization
|
||||||
|
await diarization_processor.push(diarization_input)
|
||||||
|
await diarization_processor.flush()
|
||||||
|
|
||||||
|
logger.info("Diarization complete")
|
||||||
|
|
||||||
|
except ImportError as e:
|
||||||
|
logger.error(f"Failed to import diarization dependencies: {e}")
|
||||||
|
logger.error(
|
||||||
|
"Install with: uv pip install pyannote.audio torch torchaudio"
|
||||||
|
)
|
||||||
|
logger.error(
|
||||||
|
"And set HF_TOKEN environment variable for pyannote models"
|
||||||
|
)
|
||||||
|
raise SystemExit(1)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"Diarization failed: {e}")
|
||||||
|
raise SystemExit(1)
|
||||||
|
else:
|
||||||
|
logger.warning("Skipping diarization: no topics available")
|
||||||
|
|
||||||
|
# Clean up temp file
|
||||||
|
if audio_temp_path:
|
||||||
|
try:
|
||||||
|
Path(audio_temp_path).unlink()
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning(f"Failed to clean up temp file {audio_temp_path}: {e}")
|
||||||
|
|
||||||
|
logger.info("All done!")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
import argparse
|
||||||
|
import os
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
description="Process audio files with optional speaker diarization"
|
||||||
|
)
|
||||||
|
parser.add_argument("source", help="Source file (mp3, wav, mp4...)")
|
||||||
|
parser.add_argument(
|
||||||
|
"--only-transcript",
|
||||||
|
"-t",
|
||||||
|
action="store_true",
|
||||||
|
help="Only generate transcript without topics/summaries",
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--source-language", default="en", help="Source language code (default: en)"
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--target-language", default="en", help="Target language code (default: en)"
|
||||||
|
)
|
||||||
|
parser.add_argument("--output", "-o", help="Output file (output.jsonl)")
|
||||||
|
parser.add_argument(
|
||||||
|
"--enable-diarization",
|
||||||
|
"-d",
|
||||||
|
action="store_true",
|
||||||
|
help="Enable speaker diarization",
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--diarization-backend",
|
||||||
|
default="modal",
|
||||||
|
choices=["modal"],
|
||||||
|
help="Diarization backend to use (default: modal)",
|
||||||
|
)
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
# Set REDIS_HOST to localhost if not provided
|
||||||
|
if "REDIS_HOST" not in os.environ:
|
||||||
|
os.environ["REDIS_HOST"] = "localhost"
|
||||||
|
logger.info("REDIS_HOST not set, defaulting to localhost")
|
||||||
|
|
||||||
|
output_fd = None
|
||||||
|
if args.output:
|
||||||
|
output_fd = open(args.output, "w")
|
||||||
|
|
||||||
|
async def event_callback(event: PipelineEvent):
|
||||||
|
processor = event.processor
|
||||||
|
data = event.data
|
||||||
|
|
||||||
|
# Ignore internal processors
|
||||||
|
if processor in (
|
||||||
|
"AudioChunkerProcessor",
|
||||||
|
"AudioMergeProcessor",
|
||||||
|
"AudioFileWriterProcessor",
|
||||||
|
"TopicCollectorProcessor",
|
||||||
|
"BroadcastProcessor",
|
||||||
|
):
|
||||||
|
return
|
||||||
|
|
||||||
|
# If diarization is enabled, skip the original topic events from the pipeline
|
||||||
|
# The diarization processor will emit the same topics but with speaker info
|
||||||
|
if processor == "TranscriptTopicDetectorProcessor" and args.enable_diarization:
|
||||||
|
return
|
||||||
|
|
||||||
|
# Log all events
|
||||||
|
logger.info(f"Event: {processor} - {type(data).__name__}")
|
||||||
|
|
||||||
|
# Write to output
|
||||||
|
if output_fd:
|
||||||
|
output_fd.write(event.model_dump_json())
|
||||||
|
output_fd.write("\n")
|
||||||
|
output_fd.flush()
|
||||||
|
|
||||||
|
asyncio.run(
|
||||||
|
process_audio_file_with_diarization(
|
||||||
|
args.source,
|
||||||
|
event_callback,
|
||||||
|
only_transcript=args.only_transcript,
|
||||||
|
source_language=args.source_language,
|
||||||
|
target_language=args.target_language,
|
||||||
|
enable_diarization=args.enable_diarization,
|
||||||
|
diarization_backend=args.diarization_backend,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
if output_fd:
|
||||||
|
output_fd.close()
|
||||||
|
logger.info(f"Output written to {args.output}")
|
||||||
96
server/reflector/tools/test_diarization.py
Normal file
96
server/reflector/tools/test_diarization.py
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
@vibe-generated
|
||||||
|
Test script for the diarization CLI tool
|
||||||
|
=========================================
|
||||||
|
|
||||||
|
This script helps test the diarization functionality with sample audio files.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from reflector.logger import logger
|
||||||
|
|
||||||
|
|
||||||
|
async def test_diarization(audio_file: str):
|
||||||
|
"""Test the diarization functionality"""
|
||||||
|
|
||||||
|
# Import the processing function
|
||||||
|
from process_with_diarization import process_audio_file_with_diarization
|
||||||
|
|
||||||
|
# Collect events
|
||||||
|
events = []
|
||||||
|
|
||||||
|
async def event_callback(event):
|
||||||
|
events.append({
|
||||||
|
"processor": event.processor,
|
||||||
|
"data": event.data
|
||||||
|
})
|
||||||
|
logger.info(f"Event from {event.processor}")
|
||||||
|
|
||||||
|
# Process the audio file
|
||||||
|
logger.info(f"Processing audio file: {audio_file}")
|
||||||
|
|
||||||
|
try:
|
||||||
|
await process_audio_file_with_diarization(
|
||||||
|
audio_file,
|
||||||
|
event_callback,
|
||||||
|
only_transcript=False,
|
||||||
|
source_language="en",
|
||||||
|
target_language="en",
|
||||||
|
enable_diarization=True,
|
||||||
|
diarization_backend="modal",
|
||||||
|
)
|
||||||
|
|
||||||
|
# Analyze results
|
||||||
|
logger.info(f"Processing complete. Received {len(events)} events")
|
||||||
|
|
||||||
|
# Look for diarization results
|
||||||
|
diarized_topics = []
|
||||||
|
for event in events:
|
||||||
|
if "TitleSummary" in event["processor"]:
|
||||||
|
# Check if words have speaker information
|
||||||
|
if hasattr(event["data"], "transcript") and event["data"].transcript:
|
||||||
|
words = event["data"].transcript.words
|
||||||
|
if words and hasattr(words[0], "speaker"):
|
||||||
|
speakers = set(w.speaker for w in words if hasattr(w, "speaker"))
|
||||||
|
logger.info(f"Found {len(speakers)} speakers in topic: {event['data'].title}")
|
||||||
|
diarized_topics.append(event["data"])
|
||||||
|
|
||||||
|
if diarized_topics:
|
||||||
|
logger.info(f"Successfully diarized {len(diarized_topics)} topics")
|
||||||
|
|
||||||
|
# Print sample output
|
||||||
|
sample_topic = diarized_topics[0]
|
||||||
|
logger.info("Sample diarized output:")
|
||||||
|
for i, word in enumerate(sample_topic.transcript.words[:10]):
|
||||||
|
logger.info(f" Word {i}: '{word.text}' - Speaker {word.speaker}")
|
||||||
|
else:
|
||||||
|
logger.warning("No diarization results found in output")
|
||||||
|
|
||||||
|
return events
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"Error during processing: {e}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if len(sys.argv) < 2:
|
||||||
|
print("Usage: python test_diarization.py <audio_file>")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
audio_file = sys.argv[1]
|
||||||
|
if not Path(audio_file).exists():
|
||||||
|
print(f"Error: Audio file '{audio_file}' not found")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# Run the test
|
||||||
|
asyncio.run(test_diarization(audio_file))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
149
server/reflector/utils/s3_temp_file.py
Normal file
149
server/reflector/utils/s3_temp_file.py
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
"""
|
||||||
|
@vibe-generated
|
||||||
|
S3 Temporary File Context Manager
|
||||||
|
|
||||||
|
Provides automatic cleanup of S3 files with retry logic and proper error handling.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from typing import Optional
|
||||||
|
from reflector.storage.base import Storage
|
||||||
|
from reflector.logger import logger
|
||||||
|
from reflector.utils.retry import retry
|
||||||
|
|
||||||
|
|
||||||
|
class S3TemporaryFile:
|
||||||
|
"""
|
||||||
|
Async context manager for temporary S3 files with automatic cleanup.
|
||||||
|
|
||||||
|
Ensures that uploaded files are deleted even if exceptions occur during processing.
|
||||||
|
Uses retry logic for all S3 operations to handle transient failures.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
async with S3TemporaryFile(storage, "temp/audio.wav") as s3_file:
|
||||||
|
url = await s3_file.upload(audio_data)
|
||||||
|
# Use url for processing
|
||||||
|
# File is automatically cleaned up here
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, storage: Storage, filepath: str):
|
||||||
|
"""
|
||||||
|
Initialize the temporary file context.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
storage: Storage instance for S3 operations
|
||||||
|
filepath: S3 key/path for the temporary file
|
||||||
|
"""
|
||||||
|
self.storage = storage
|
||||||
|
self.filepath = filepath
|
||||||
|
self.uploaded = False
|
||||||
|
self._url: Optional[str] = None
|
||||||
|
|
||||||
|
async def __aenter__(self):
|
||||||
|
"""Enter the context manager."""
|
||||||
|
return self
|
||||||
|
|
||||||
|
async def __aexit__(self, exc_type, exc_val, exc_tb):
|
||||||
|
"""
|
||||||
|
Exit the context manager and clean up the file.
|
||||||
|
|
||||||
|
Cleanup is attempted even if an exception occurred during processing.
|
||||||
|
Cleanup failures are logged but don't raise exceptions.
|
||||||
|
"""
|
||||||
|
if self.uploaded:
|
||||||
|
try:
|
||||||
|
await self._delete_with_retry()
|
||||||
|
logger.info(f"Successfully cleaned up S3 file: {self.filepath}")
|
||||||
|
except Exception as e:
|
||||||
|
# Log the error but don't raise - we don't want cleanup failures
|
||||||
|
# to mask the original exception
|
||||||
|
logger.warning(
|
||||||
|
f"Failed to cleanup S3 file {self.filepath} after retries: {e}"
|
||||||
|
)
|
||||||
|
return False # Don't suppress exceptions
|
||||||
|
|
||||||
|
async def upload(self, data: bytes) -> str:
|
||||||
|
"""
|
||||||
|
Upload data to S3 and return the public URL.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
data: File data to upload
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Public URL for the uploaded file
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Exception: If upload or URL generation fails after retries
|
||||||
|
"""
|
||||||
|
await self._upload_with_retry(data)
|
||||||
|
self.uploaded = True
|
||||||
|
self._url = await self._get_url_with_retry()
|
||||||
|
return self._url
|
||||||
|
|
||||||
|
@property
|
||||||
|
def url(self) -> Optional[str]:
|
||||||
|
"""Get the URL of the uploaded file, if available."""
|
||||||
|
return self._url
|
||||||
|
|
||||||
|
async def _upload_with_retry(self, data: bytes):
|
||||||
|
"""Upload file to S3 with retry logic."""
|
||||||
|
|
||||||
|
async def upload():
|
||||||
|
await self.storage.put_file(self.filepath, data)
|
||||||
|
logger.debug(f"Successfully uploaded file to S3: {self.filepath}")
|
||||||
|
return True # Return something to indicate success
|
||||||
|
|
||||||
|
await retry(upload)(
|
||||||
|
retry_attempts=3,
|
||||||
|
retry_timeout=30.0,
|
||||||
|
retry_backoff_interval=0.5,
|
||||||
|
retry_backoff_max=5.0,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def _get_url_with_retry(self) -> str:
|
||||||
|
"""Get public URL for the file with retry logic."""
|
||||||
|
|
||||||
|
async def get_url():
|
||||||
|
url = await self.storage.get_file_url(self.filepath)
|
||||||
|
logger.debug(f"Generated public URL for S3 file: {self.filepath}")
|
||||||
|
return url
|
||||||
|
|
||||||
|
return await retry(get_url)(
|
||||||
|
retry_attempts=3,
|
||||||
|
retry_timeout=30.0,
|
||||||
|
retry_backoff_interval=0.5,
|
||||||
|
retry_backoff_max=5.0,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def _delete_with_retry(self):
|
||||||
|
"""Delete file from S3 with retry logic."""
|
||||||
|
|
||||||
|
async def delete():
|
||||||
|
await self.storage.delete_file(self.filepath)
|
||||||
|
logger.debug(f"Successfully deleted S3 file: {self.filepath}")
|
||||||
|
return True # Return something to indicate success
|
||||||
|
|
||||||
|
await retry(delete)(
|
||||||
|
retry_attempts=3,
|
||||||
|
retry_timeout=30.0,
|
||||||
|
retry_backoff_interval=0.5,
|
||||||
|
retry_backoff_max=5.0,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
# Convenience function for simpler usage
|
||||||
|
async def temporary_s3_file(storage: Storage, filepath: str):
|
||||||
|
"""
|
||||||
|
Create a temporary S3 file context manager.
|
||||||
|
|
||||||
|
This is a convenience wrapper around S3TemporaryFile for simpler usage.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
storage: Storage instance for S3 operations
|
||||||
|
filepath: S3 key/path for the temporary file
|
||||||
|
|
||||||
|
Example:
|
||||||
|
async with temporary_s3_file(storage, "temp/audio.wav") as s3_file:
|
||||||
|
url = await s3_file.upload(audio_data)
|
||||||
|
# Use url for processing
|
||||||
|
"""
|
||||||
|
return S3TemporaryFile(storage, filepath)
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from typing import Annotated, Optional, Literal
|
from typing import Annotated, Optional, Literal
|
||||||
|
import logging
|
||||||
|
|
||||||
import reflector.auth as auth
|
import reflector.auth as auth
|
||||||
from fastapi import APIRouter, Depends, HTTPException
|
from fastapi import APIRouter, Depends, HTTPException
|
||||||
@@ -11,6 +12,10 @@ from reflector.db.meetings import meetings_controller
|
|||||||
from reflector.db.rooms import rooms_controller
|
from reflector.db.rooms import rooms_controller
|
||||||
from reflector.settings import settings
|
from reflector.settings import settings
|
||||||
from reflector.whereby import create_meeting, upload_logo
|
from reflector.whereby import create_meeting, upload_logo
|
||||||
|
import asyncpg.exceptions
|
||||||
|
import sqlite3
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
router = APIRouter()
|
router = APIRouter()
|
||||||
|
|
||||||
@@ -28,6 +33,7 @@ class Room(BaseModel):
|
|||||||
recording_type: str
|
recording_type: str
|
||||||
recording_trigger: str
|
recording_trigger: str
|
||||||
is_shared: bool
|
is_shared: bool
|
||||||
|
background_information: str
|
||||||
|
|
||||||
|
|
||||||
class Meeting(BaseModel):
|
class Meeting(BaseModel):
|
||||||
@@ -50,6 +56,7 @@ class CreateRoom(BaseModel):
|
|||||||
recording_type: str
|
recording_type: str
|
||||||
recording_trigger: str
|
recording_trigger: str
|
||||||
is_shared: bool
|
is_shared: bool
|
||||||
|
background_information: str = ""
|
||||||
|
|
||||||
|
|
||||||
class UpdateRoom(BaseModel):
|
class UpdateRoom(BaseModel):
|
||||||
@@ -62,6 +69,7 @@ class UpdateRoom(BaseModel):
|
|||||||
recording_type: str
|
recording_type: str
|
||||||
recording_trigger: str
|
recording_trigger: str
|
||||||
is_shared: bool
|
is_shared: bool
|
||||||
|
background_information: str = ""
|
||||||
|
|
||||||
|
|
||||||
class DeletionStatus(BaseModel):
|
class DeletionStatus(BaseModel):
|
||||||
@@ -103,6 +111,7 @@ async def rooms_create(
|
|||||||
recording_type=room.recording_type,
|
recording_type=room.recording_type,
|
||||||
recording_trigger=room.recording_trigger,
|
recording_trigger=room.recording_trigger,
|
||||||
is_shared=room.is_shared,
|
is_shared=room.is_shared,
|
||||||
|
background_information=room.background_information,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -149,19 +158,47 @@ async def rooms_create_meeting(
|
|||||||
|
|
||||||
if meeting is None:
|
if meeting is None:
|
||||||
end_date = current_time + timedelta(hours=8)
|
end_date = current_time + timedelta(hours=8)
|
||||||
meeting = await create_meeting("", end_date=end_date, room=room)
|
|
||||||
await upload_logo(meeting["roomName"], "./images/logo.png")
|
|
||||||
|
|
||||||
|
whereby_meeting = await create_meeting("", end_date=end_date, room=room)
|
||||||
|
await upload_logo(whereby_meeting["roomName"], "./images/logo.png")
|
||||||
|
|
||||||
|
# Now try to save to database
|
||||||
|
try:
|
||||||
meeting = await meetings_controller.create(
|
meeting = await meetings_controller.create(
|
||||||
id=meeting["meetingId"],
|
id=whereby_meeting["meetingId"],
|
||||||
room_name=meeting["roomName"],
|
room_name=whereby_meeting["roomName"],
|
||||||
room_url=meeting["roomUrl"],
|
room_url=whereby_meeting["roomUrl"],
|
||||||
host_room_url=meeting["hostRoomUrl"],
|
host_room_url=whereby_meeting["hostRoomUrl"],
|
||||||
start_date=datetime.fromisoformat(meeting["startDate"]),
|
start_date=datetime.fromisoformat(whereby_meeting["startDate"]),
|
||||||
end_date=datetime.fromisoformat(meeting["endDate"]),
|
end_date=datetime.fromisoformat(whereby_meeting["endDate"]),
|
||||||
user_id=user_id,
|
user_id=user_id,
|
||||||
room=room,
|
room=room,
|
||||||
)
|
)
|
||||||
|
except (asyncpg.exceptions.UniqueViolationError, sqlite3.IntegrityError):
|
||||||
|
# Another request already created a meeting for this room
|
||||||
|
# Log this race condition occurrence
|
||||||
|
logger.info(
|
||||||
|
"Race condition detected for room %s - fetching existing meeting",
|
||||||
|
room.name,
|
||||||
|
)
|
||||||
|
logger.warning(
|
||||||
|
"Whereby meeting %s was created but not used (resource leak) for room %s",
|
||||||
|
whereby_meeting["meetingId"],
|
||||||
|
room.name,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Fetch the meeting that was created by the other request
|
||||||
|
meeting = await meetings_controller.get_active(
|
||||||
|
room=room, current_time=current_time
|
||||||
|
)
|
||||||
|
if meeting is None:
|
||||||
|
# Edge case: meeting was created but expired/deleted between checks
|
||||||
|
logger.error(
|
||||||
|
"Meeting disappeared after race condition for room %s", room.name
|
||||||
|
)
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=503, detail="Unable to join meeting - please try again"
|
||||||
|
)
|
||||||
|
|
||||||
if user_id != room.user_id:
|
if user_id != room.user_id:
|
||||||
meeting.host_room_url = ""
|
meeting.host_room_url = ""
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ from fastapi_pagination.ext.databases import paginate
|
|||||||
from jose import jwt
|
from jose import jwt
|
||||||
from pydantic import BaseModel, Field, field_serializer
|
from pydantic import BaseModel, Field, field_serializer
|
||||||
from reflector.db.meetings import meetings_controller
|
from reflector.db.meetings import meetings_controller
|
||||||
from reflector.db.migrate_user import migrate_user
|
|
||||||
from reflector.db.rooms import rooms_controller
|
from reflector.db.rooms import rooms_controller
|
||||||
from reflector.db.transcripts import (
|
from reflector.db.transcripts import (
|
||||||
SourceKind,
|
SourceKind,
|
||||||
@@ -114,10 +113,6 @@ async def transcripts_list(
|
|||||||
|
|
||||||
user_id = user["sub"] if user else None
|
user_id = user["sub"] if user else None
|
||||||
|
|
||||||
# for fief to jwt migration, migrate user if needed
|
|
||||||
if user:
|
|
||||||
await migrate_user(email=user["email"], user_id=user["sub"])
|
|
||||||
|
|
||||||
return await paginate(
|
return await paginate(
|
||||||
database,
|
database,
|
||||||
await transcripts_controller.get_all(
|
await transcripts_controller.get_all(
|
||||||
|
|||||||
129
server/tests/test_s3_temp_file.py
Normal file
129
server/tests/test_s3_temp_file.py
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
"""
|
||||||
|
@vibe-generated
|
||||||
|
Tests for S3 temporary file context manager.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from unittest.mock import Mock, AsyncMock
|
||||||
|
from reflector.utils.s3_temp_file import S3TemporaryFile
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_successful_upload_and_cleanup():
|
||||||
|
"""Test that file is uploaded and cleaned up on success."""
|
||||||
|
# Mock storage
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.put_file = AsyncMock()
|
||||||
|
mock_storage.get_file_url = AsyncMock(return_value="https://example.com/file.wav")
|
||||||
|
mock_storage.delete_file = AsyncMock()
|
||||||
|
|
||||||
|
# Use context manager
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
url = await s3_file.upload(b"test data")
|
||||||
|
assert url == "https://example.com/file.wav"
|
||||||
|
assert s3_file.url == "https://example.com/file.wav"
|
||||||
|
|
||||||
|
# Verify operations
|
||||||
|
mock_storage.put_file.assert_called_once_with("test/file.wav", b"test data")
|
||||||
|
mock_storage.get_file_url.assert_called_once_with("test/file.wav")
|
||||||
|
mock_storage.delete_file.assert_called_once_with("test/file.wav")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_cleanup_on_exception():
|
||||||
|
"""Test that cleanup happens even when an exception occurs."""
|
||||||
|
# Mock storage
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.put_file = AsyncMock()
|
||||||
|
mock_storage.get_file_url = AsyncMock(return_value="https://example.com/file.wav")
|
||||||
|
mock_storage.delete_file = AsyncMock()
|
||||||
|
|
||||||
|
# Use context manager with exception
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
await s3_file.upload(b"test data")
|
||||||
|
raise ValueError("Simulated error during processing")
|
||||||
|
|
||||||
|
# Verify cleanup still happened
|
||||||
|
mock_storage.delete_file.assert_called_once_with("test/file.wav")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_no_cleanup_if_not_uploaded():
|
||||||
|
"""Test that cleanup is skipped if file was never uploaded."""
|
||||||
|
# Mock storage
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.delete_file = AsyncMock()
|
||||||
|
|
||||||
|
# Use context manager without uploading
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav"):
|
||||||
|
pass # Don't upload anything
|
||||||
|
|
||||||
|
# Verify no cleanup attempted
|
||||||
|
mock_storage.delete_file.assert_not_called()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_cleanup_failure_is_logged_not_raised():
|
||||||
|
"""Test that cleanup failures are logged but don't raise exceptions."""
|
||||||
|
# Mock storage
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.put_file = AsyncMock()
|
||||||
|
mock_storage.get_file_url = AsyncMock(return_value="https://example.com/file.wav")
|
||||||
|
mock_storage.delete_file = AsyncMock(side_effect=Exception("Delete failed"))
|
||||||
|
|
||||||
|
# Use context manager - should not raise
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
await s3_file.upload(b"test data")
|
||||||
|
|
||||||
|
# Verify delete was attempted (3 times due to retry)
|
||||||
|
assert mock_storage.delete_file.call_count == 3
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_upload_retry_on_failure():
|
||||||
|
"""Test that upload is retried on failure."""
|
||||||
|
# Mock storage with failures then success
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.put_file = AsyncMock(
|
||||||
|
side_effect=[Exception("Network error"), None] # Fail once, then succeed
|
||||||
|
)
|
||||||
|
mock_storage.get_file_url = AsyncMock(return_value="https://example.com/file.wav")
|
||||||
|
mock_storage.delete_file = AsyncMock()
|
||||||
|
|
||||||
|
# Use context manager
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
url = await s3_file.upload(b"test data")
|
||||||
|
assert url == "https://example.com/file.wav"
|
||||||
|
|
||||||
|
# Verify upload was retried
|
||||||
|
assert mock_storage.put_file.call_count == 2
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_delete_retry_on_failure():
|
||||||
|
"""Test that delete is retried on failure."""
|
||||||
|
# Mock storage
|
||||||
|
mock_storage = Mock()
|
||||||
|
mock_storage.put_file = AsyncMock()
|
||||||
|
mock_storage.get_file_url = AsyncMock(return_value="https://example.com/file.wav")
|
||||||
|
mock_storage.delete_file = AsyncMock(
|
||||||
|
side_effect=[Exception("Network error"), None] # Fail once, then succeed
|
||||||
|
)
|
||||||
|
|
||||||
|
# Use context manager
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
await s3_file.upload(b"test data")
|
||||||
|
|
||||||
|
# Verify delete was retried
|
||||||
|
assert mock_storage.delete_file.call_count == 2
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_properties_before_upload():
|
||||||
|
"""Test that properties work correctly before upload."""
|
||||||
|
mock_storage = Mock()
|
||||||
|
|
||||||
|
async with S3TemporaryFile(mock_storage, "test/file.wav") as s3_file:
|
||||||
|
assert s3_file.url is None
|
||||||
|
assert s3_file.uploaded is False
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import pytest
|
|
||||||
from unittest.mock import patch
|
|
||||||
from contextlib import asynccontextmanager
|
from contextlib import asynccontextmanager
|
||||||
|
|
||||||
|
import pytest
|
||||||
from httpx import AsyncClient
|
from httpx import AsyncClient
|
||||||
|
|
||||||
|
|
||||||
@@ -261,67 +261,3 @@ async def test_transcript_mark_reviewed():
|
|||||||
response = await ac.get(f"/transcripts/{tid}")
|
response = await ac.get(f"/transcripts/{tid}")
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response.json()["reviewed"] is True
|
assert response.json()["reviewed"] is True
|
||||||
|
|
||||||
|
|
||||||
@asynccontextmanager
|
|
||||||
async def patch_migrate_user():
|
|
||||||
with patch(
|
|
||||||
"reflector.db.migrate_user.users_to_migrate",
|
|
||||||
[["test@mail.com", "randomuserid", None]],
|
|
||||||
):
|
|
||||||
yield
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
|
||||||
async def test_transcripts_list_authenticated_migration():
|
|
||||||
# XXX this test is a bit fragile, as it depends on the storage which
|
|
||||||
# is shared between tests
|
|
||||||
from reflector.app import app
|
|
||||||
|
|
||||||
testx1 = "testmigration1"
|
|
||||||
testx2 = "testmigration2"
|
|
||||||
|
|
||||||
async with patch_migrate_user(), AsyncClient(
|
|
||||||
app=app, base_url="http://test/v1"
|
|
||||||
) as ac:
|
|
||||||
# first ensure client 2 does not have any transcripts related to this test
|
|
||||||
async with authenticated_client2_ctx():
|
|
||||||
response = await ac.get("/transcripts")
|
|
||||||
assert response.status_code == 200
|
|
||||||
# assert len(response.json()["items"]) == 0
|
|
||||||
names = [t["name"] for t in response.json()["items"]]
|
|
||||||
assert testx1 not in names
|
|
||||||
assert testx2 not in names
|
|
||||||
|
|
||||||
# create 2 transcripts with client 1
|
|
||||||
async with authenticated_client_ctx():
|
|
||||||
response = await ac.post("/transcripts", json={"name": testx1})
|
|
||||||
assert response.status_code == 200
|
|
||||||
assert response.json()["name"] == testx1
|
|
||||||
|
|
||||||
response = await ac.post("/transcripts", json={"name": testx2})
|
|
||||||
assert response.status_code == 200
|
|
||||||
assert response.json()["name"] == testx2
|
|
||||||
|
|
||||||
response = await ac.get("/transcripts")
|
|
||||||
assert response.status_code == 200
|
|
||||||
assert len(response.json()["items"]) >= 2
|
|
||||||
names = [t["name"] for t in response.json()["items"]]
|
|
||||||
assert testx1 in names
|
|
||||||
assert testx2 in names
|
|
||||||
|
|
||||||
# now going back to client 2, migration should happen
|
|
||||||
async with authenticated_client2_ctx():
|
|
||||||
response = await ac.get("/transcripts")
|
|
||||||
assert response.status_code == 200
|
|
||||||
names = [t["name"] for t in response.json()["items"]]
|
|
||||||
assert testx1 in names
|
|
||||||
assert testx2 in names
|
|
||||||
|
|
||||||
# and client 1 should have nothing now
|
|
||||||
async with authenticated_client_ctx():
|
|
||||||
response = await ac.get("/transcripts")
|
|
||||||
assert response.status_code == 200
|
|
||||||
names = [t["name"] for t in response.json()["items"]]
|
|
||||||
assert testx1 not in names
|
|
||||||
assert testx2 not in names
|
|
||||||
|
|||||||
43
server/uv.lock
generated
43
server/uv.lock
generated
@@ -817,25 +817,6 @@ wheels = [
|
|||||||
{ url = "https://files.pythonhosted.org/packages/ad/69/28359d152f9e2ec1ff4dff3da47011b6346e9a472f89b409bb13017a7d1f/faster_whisper-1.1.1-py3-none-any.whl", hash = "sha256:5808dc334fb64fb4336921450abccfe5e313a859b31ba61def0ac7f639383d90", size = 1118368, upload-time = "2025-01-01T14:47:16.131Z" },
|
{ url = "https://files.pythonhosted.org/packages/ad/69/28359d152f9e2ec1ff4dff3da47011b6346e9a472f89b409bb13017a7d1f/faster_whisper-1.1.1-py3-none-any.whl", hash = "sha256:5808dc334fb64fb4336921450abccfe5e313a859b31ba61def0ac7f639383d90", size = 1118368, upload-time = "2025-01-01T14:47:16.131Z" },
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "fief-client"
|
|
||||||
version = "0.20.0"
|
|
||||||
source = { registry = "https://pypi.org/simple" }
|
|
||||||
dependencies = [
|
|
||||||
{ name = "httpx" },
|
|
||||||
{ name = "jwcrypto" },
|
|
||||||
]
|
|
||||||
sdist = { url = "https://files.pythonhosted.org/packages/75/af/f6cc3ded8bdb901097b92a3ed444c48576a1b62f01352cb2fa069b0dd166/fief_client-0.20.0.tar.gz", hash = "sha256:dbfb906d03c4a5402ceac5c843aa4708535fb6f5d5c1c4e263ec06fbbbc434d7", size = 32465, upload-time = "2024-10-13T11:54:08.793Z" }
|
|
||||||
wheels = [
|
|
||||||
{ url = "https://files.pythonhosted.org/packages/1c/06/d33506317b4c9b71025eb010d96c4f7a8f89fa620ca30532c2e8e4390593/fief_client-0.20.0-py3-none-any.whl", hash = "sha256:425f40cc7c45c651daec63da402e033c53d91dcaa3f9bf208873fd8692fc16dc", size = 20219, upload-time = "2024-10-13T11:54:07.342Z" },
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.optional-dependencies]
|
|
||||||
fastapi = [
|
|
||||||
{ name = "fastapi" },
|
|
||||||
{ name = "makefun" },
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "filelock"
|
name = "filelock"
|
||||||
version = "3.18.0"
|
version = "3.18.0"
|
||||||
@@ -1211,19 +1192,6 @@ wheels = [
|
|||||||
{ url = "https://files.pythonhosted.org/packages/01/0e/b27cdbaccf30b890c40ed1da9fd4a3593a5cf94dae54fb34f8a4b74fcd3f/jsonschema_specifications-2025.4.1-py3-none-any.whl", hash = "sha256:4653bffbd6584f7de83a67e0d620ef16900b390ddc7939d56684d6c81e33f1af", size = 18437, upload-time = "2025-04-23T12:34:05.422Z" },
|
{ url = "https://files.pythonhosted.org/packages/01/0e/b27cdbaccf30b890c40ed1da9fd4a3593a5cf94dae54fb34f8a4b74fcd3f/jsonschema_specifications-2025.4.1-py3-none-any.whl", hash = "sha256:4653bffbd6584f7de83a67e0d620ef16900b390ddc7939d56684d6c81e33f1af", size = 18437, upload-time = "2025-04-23T12:34:05.422Z" },
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "jwcrypto"
|
|
||||||
version = "1.5.6"
|
|
||||||
source = { registry = "https://pypi.org/simple" }
|
|
||||||
dependencies = [
|
|
||||||
{ name = "cryptography" },
|
|
||||||
{ name = "typing-extensions" },
|
|
||||||
]
|
|
||||||
sdist = { url = "https://files.pythonhosted.org/packages/e1/db/870e5d5fb311b0bcf049630b5ba3abca2d339fd5e13ba175b4c13b456d08/jwcrypto-1.5.6.tar.gz", hash = "sha256:771a87762a0c081ae6166958a954f80848820b2ab066937dc8b8379d65b1b039", size = 87168, upload-time = "2024-03-06T19:58:31.831Z" }
|
|
||||||
wheels = [
|
|
||||||
{ url = "https://files.pythonhosted.org/packages/cd/58/4a1880ea64032185e9ae9f63940c9327c6952d5584ea544a8f66972f2fda/jwcrypto-1.5.6-py3-none-any.whl", hash = "sha256:150d2b0ebbdb8f40b77f543fb44ffd2baeff48788be71f67f03566692fd55789", size = 92520, upload-time = "2024-03-06T19:58:29.765Z" },
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kombu"
|
name = "kombu"
|
||||||
version = "5.5.4"
|
version = "5.5.4"
|
||||||
@@ -1299,15 +1267,6 @@ wheels = [
|
|||||||
{ url = "https://files.pythonhosted.org/packages/0c/29/0348de65b8cc732daa3e33e67806420b2ae89bdce2b04af740289c5c6c8c/loguru-0.7.3-py3-none-any.whl", hash = "sha256:31a33c10c8e1e10422bfd431aeb5d351c7cf7fa671e3c4df004162264b28220c", size = 61595, upload-time = "2024-12-06T11:20:54.538Z" },
|
{ url = "https://files.pythonhosted.org/packages/0c/29/0348de65b8cc732daa3e33e67806420b2ae89bdce2b04af740289c5c6c8c/loguru-0.7.3-py3-none-any.whl", hash = "sha256:31a33c10c8e1e10422bfd431aeb5d351c7cf7fa671e3c4df004162264b28220c", size = 61595, upload-time = "2024-12-06T11:20:54.538Z" },
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "makefun"
|
|
||||||
version = "1.16.0"
|
|
||||||
source = { registry = "https://pypi.org/simple" }
|
|
||||||
sdist = { url = "https://files.pythonhosted.org/packages/7b/cf/6780ab8bc3b84a1cce3e4400aed3d64b6db7d5e227a2f75b6ded5674701a/makefun-1.16.0.tar.gz", hash = "sha256:e14601831570bff1f6d7e68828bcd30d2f5856f24bad5de0ccb22921ceebc947", size = 73565, upload-time = "2025-05-09T15:00:42.313Z" }
|
|
||||||
wheels = [
|
|
||||||
{ url = "https://files.pythonhosted.org/packages/b7/c0/4bc973defd1270b89ccaae04cef0d5fa3ea85b59b108ad2c08aeea9afb76/makefun-1.16.0-py2.py3-none-any.whl", hash = "sha256:43baa4c3e7ae2b17de9ceac20b669e9a67ceeadff31581007cca20a07bbe42c4", size = 22923, upload-time = "2025-05-09T15:00:41.042Z" },
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "mako"
|
name = "mako"
|
||||||
version = "1.3.10"
|
version = "1.3.10"
|
||||||
@@ -2172,7 +2131,6 @@ dependencies = [
|
|||||||
{ name = "fastapi", extra = ["standard"] },
|
{ name = "fastapi", extra = ["standard"] },
|
||||||
{ name = "fastapi-pagination" },
|
{ name = "fastapi-pagination" },
|
||||||
{ name = "faster-whisper" },
|
{ name = "faster-whisper" },
|
||||||
{ name = "fief-client", extra = ["fastapi"] },
|
|
||||||
{ name = "httpx" },
|
{ name = "httpx" },
|
||||||
{ name = "jsonschema" },
|
{ name = "jsonschema" },
|
||||||
{ name = "loguru" },
|
{ name = "loguru" },
|
||||||
@@ -2234,7 +2192,6 @@ requires-dist = [
|
|||||||
{ name = "fastapi", extras = ["standard"], specifier = ">=0.100.1" },
|
{ name = "fastapi", extras = ["standard"], specifier = ">=0.100.1" },
|
||||||
{ name = "fastapi-pagination", specifier = ">=0.12.6" },
|
{ name = "fastapi-pagination", specifier = ">=0.12.6" },
|
||||||
{ name = "faster-whisper", specifier = ">=0.10.0" },
|
{ name = "faster-whisper", specifier = ">=0.10.0" },
|
||||||
{ name = "fief-client", extras = ["fastapi"], specifier = ">=0.17.0" },
|
|
||||||
{ name = "httpx", specifier = ">=0.24.1" },
|
{ name = "httpx", specifier = ">=0.24.1" },
|
||||||
{ name = "jsonschema", specifier = ">=4.23.0" },
|
{ name = "jsonschema", specifier = ">=4.23.0" },
|
||||||
{ name = "loguru", specifier = ">=0.7.0" },
|
{ name = "loguru", specifier = ">=0.7.0" },
|
||||||
|
|||||||
@@ -20,10 +20,11 @@ export default function FilterSidebar({
|
|||||||
const sharedRooms = rooms.filter((room) => room.is_shared);
|
const sharedRooms = rooms.filter((room) => room.is_shared);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box w={{ base: "full", md: "300px" }} p={4} bg="gray.100" rounded="md">
|
<Box w={{ base: "full", md: "200px" }} p={4} bg="gray.100" rounded="md">
|
||||||
<Stack gap={3}>
|
<Stack gap={2}>
|
||||||
<Link
|
<Link
|
||||||
as={NextLink}
|
as={NextLink}
|
||||||
|
fontSize="sm"
|
||||||
href="#"
|
href="#"
|
||||||
onClick={() => onFilterChange(null, "")}
|
onClick={() => onFilterChange(null, "")}
|
||||||
color={selectedSourceKind === null ? "blue.500" : "gray.600"}
|
color={selectedSourceKind === null ? "blue.500" : "gray.600"}
|
||||||
@@ -36,7 +37,7 @@ export default function FilterSidebar({
|
|||||||
|
|
||||||
{myRooms.length > 0 && (
|
{myRooms.length > 0 && (
|
||||||
<>
|
<>
|
||||||
<Heading size="md">My Rooms</Heading>
|
<Heading size="sm">My Rooms</Heading>
|
||||||
|
|
||||||
{myRooms.map((room) => (
|
{myRooms.map((room) => (
|
||||||
<Link
|
<Link
|
||||||
@@ -54,7 +55,7 @@ export default function FilterSidebar({
|
|||||||
? "bold"
|
? "bold"
|
||||||
: "normal"
|
: "normal"
|
||||||
}
|
}
|
||||||
ml={4}
|
fontSize="sm"
|
||||||
>
|
>
|
||||||
{room.name}
|
{room.name}
|
||||||
</Link>
|
</Link>
|
||||||
@@ -64,7 +65,7 @@ export default function FilterSidebar({
|
|||||||
|
|
||||||
{sharedRooms.length > 0 && (
|
{sharedRooms.length > 0 && (
|
||||||
<>
|
<>
|
||||||
<Heading size="md">Shared Rooms</Heading>
|
<Heading size="sm">Shared Rooms</Heading>
|
||||||
|
|
||||||
{sharedRooms.map((room) => (
|
{sharedRooms.map((room) => (
|
||||||
<Link
|
<Link
|
||||||
@@ -82,7 +83,7 @@ export default function FilterSidebar({
|
|||||||
? "bold"
|
? "bold"
|
||||||
: "normal"
|
: "normal"
|
||||||
}
|
}
|
||||||
ml={4}
|
fontSize="sm"
|
||||||
>
|
>
|
||||||
{room.name}
|
{room.name}
|
||||||
</Link>
|
</Link>
|
||||||
@@ -98,6 +99,7 @@ export default function FilterSidebar({
|
|||||||
color={selectedSourceKind === "live" ? "blue.500" : "gray.600"}
|
color={selectedSourceKind === "live" ? "blue.500" : "gray.600"}
|
||||||
_hover={{ color: "blue.300" }}
|
_hover={{ color: "blue.300" }}
|
||||||
fontWeight={selectedSourceKind === "live" ? "bold" : "normal"}
|
fontWeight={selectedSourceKind === "live" ? "bold" : "normal"}
|
||||||
|
fontSize="sm"
|
||||||
>
|
>
|
||||||
Live Transcripts
|
Live Transcripts
|
||||||
</Link>
|
</Link>
|
||||||
@@ -108,6 +110,7 @@ export default function FilterSidebar({
|
|||||||
color={selectedSourceKind === "file" ? "blue.500" : "gray.600"}
|
color={selectedSourceKind === "file" ? "blue.500" : "gray.600"}
|
||||||
_hover={{ color: "blue.300" }}
|
_hover={{ color: "blue.300" }}
|
||||||
fontWeight={selectedSourceKind === "file" ? "bold" : "normal"}
|
fontWeight={selectedSourceKind === "file" ? "bold" : "normal"}
|
||||||
|
fontSize="sm"
|
||||||
>
|
>
|
||||||
Uploaded Files
|
Uploaded Files
|
||||||
</Link>
|
</Link>
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ export function RoomActionsMenu({
|
|||||||
return (
|
return (
|
||||||
<Menu.Root closeOnSelect={true} lazyMount={true}>
|
<Menu.Root closeOnSelect={true} lazyMount={true}>
|
||||||
<Menu.Trigger asChild>
|
<Menu.Trigger asChild>
|
||||||
<IconButton aria-label="actions">
|
<IconButton aria-label="actions" variant="ghost">
|
||||||
<LuMenu />
|
<LuMenu />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Menu.Trigger>
|
</Menu.Trigger>
|
||||||
|
|||||||
@@ -11,13 +11,14 @@ import {
|
|||||||
Input,
|
Input,
|
||||||
Select,
|
Select,
|
||||||
Spinner,
|
Spinner,
|
||||||
|
Textarea,
|
||||||
createListCollection,
|
createListCollection,
|
||||||
useDisclosure,
|
useDisclosure,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import useApi from "../../lib/useApi";
|
import useApi from "../../lib/useApi";
|
||||||
import useRoomList from "./useRoomList";
|
import useRoomList from "./useRoomList";
|
||||||
import { ApiError, Room } from "../../api";
|
import { Room } from "../../api";
|
||||||
import { RoomList } from "./_components/RoomList";
|
import { RoomList } from "./_components/RoomList";
|
||||||
|
|
||||||
interface SelectOption {
|
interface SelectOption {
|
||||||
@@ -54,6 +55,7 @@ const roomInitialState = {
|
|||||||
recordingType: "cloud",
|
recordingType: "cloud",
|
||||||
recordingTrigger: "automatic-2nd-participant",
|
recordingTrigger: "automatic-2nd-participant",
|
||||||
isShared: false,
|
isShared: false,
|
||||||
|
backgroundInformation: "",
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function RoomsList() {
|
export default function RoomsList() {
|
||||||
@@ -170,6 +172,7 @@ export default function RoomsList() {
|
|||||||
recording_type: room.recordingType,
|
recording_type: room.recordingType,
|
||||||
recording_trigger: room.recordingTrigger,
|
recording_trigger: room.recordingTrigger,
|
||||||
is_shared: room.isShared,
|
is_shared: room.isShared,
|
||||||
|
background_information: room.backgroundInformation,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (isEditing) {
|
if (isEditing) {
|
||||||
@@ -189,11 +192,10 @@ export default function RoomsList() {
|
|||||||
setNameError("");
|
setNameError("");
|
||||||
refetch();
|
refetch();
|
||||||
onClose();
|
onClose();
|
||||||
} catch (err) {
|
} catch (err: any) {
|
||||||
if (
|
if (
|
||||||
err instanceof ApiError &&
|
|
||||||
err.status === 400 &&
|
err.status === 400 &&
|
||||||
(err.body as any).detail == "Room name is not unique"
|
err.body?.detail === "Room name is not unique"
|
||||||
) {
|
) {
|
||||||
setNameError(
|
setNameError(
|
||||||
"This room name is already taken. Please choose a different name.",
|
"This room name is already taken. Please choose a different name.",
|
||||||
@@ -215,6 +217,7 @@ export default function RoomsList() {
|
|||||||
recordingType: roomData.recording_type,
|
recordingType: roomData.recording_type,
|
||||||
recordingTrigger: roomData.recording_trigger,
|
recordingTrigger: roomData.recording_trigger,
|
||||||
isShared: roomData.is_shared,
|
isShared: roomData.is_shared,
|
||||||
|
backgroundInformation: roomData.background_information || "",
|
||||||
});
|
});
|
||||||
setEditRoomId(roomId);
|
setEditRoomId(roomId);
|
||||||
setIsEditing(true);
|
setIsEditing(true);
|
||||||
@@ -323,6 +326,20 @@ export default function RoomsList() {
|
|||||||
{nameError && <Field.ErrorText>{nameError}</Field.ErrorText>}
|
{nameError && <Field.ErrorText>{nameError}</Field.ErrorText>}
|
||||||
</Field.Root>
|
</Field.Root>
|
||||||
|
|
||||||
|
<Field.Root mt={4}>
|
||||||
|
<Field.Label>Background Information</Field.Label>
|
||||||
|
<Textarea
|
||||||
|
name="backgroundInformation"
|
||||||
|
placeholder="Provide context about this room's purpose, meeting type, or any relevant background information that will help generate better summaries..."
|
||||||
|
value={room.backgroundInformation}
|
||||||
|
onChange={handleRoomChange}
|
||||||
|
rows={3}
|
||||||
|
/>
|
||||||
|
<Field.HelperText>
|
||||||
|
This information will be used to provide context for AI-generated summaries
|
||||||
|
</Field.HelperText>
|
||||||
|
</Field.Root>
|
||||||
|
|
||||||
<Field.Root mt={4}>
|
<Field.Root mt={4}>
|
||||||
<Checkbox.Root
|
<Checkbox.Root
|
||||||
name="isLocked"
|
name="isLocked"
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useError } from "../../(errors)/errorContext";
|
import { useError } from "../../(errors)/errorContext";
|
||||||
import useApi from "../../lib/useApi";
|
import useApi from "../../lib/useApi";
|
||||||
import { Page_Room_ } from "../../api";
|
import { PageRoom } from "../../api";
|
||||||
|
|
||||||
type RoomList = {
|
type RoomList = {
|
||||||
response: Page_Room_ | null;
|
response: PageRoom | null;
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
error: Error | null;
|
error: Error | null;
|
||||||
refetch: () => void;
|
refetch: () => void;
|
||||||
@@ -12,7 +12,7 @@ type RoomList = {
|
|||||||
|
|
||||||
//always protected
|
//always protected
|
||||||
const useRoomList = (page: number): RoomList => {
|
const useRoomList = (page: number): RoomList => {
|
||||||
const [response, setResponse] = useState<Page_Room_ | null>(null);
|
const [response, setResponse] = useState<PageRoom | null>(null);
|
||||||
const [loading, setLoading] = useState<boolean>(true);
|
const [loading, setLoading] = useState<boolean>(true);
|
||||||
const [error, setErrorState] = useState<Error | null>(null);
|
const [error, setErrorState] = useState<Error | null>(null);
|
||||||
const { setError } = useError();
|
const { setError } = useError();
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useError } from "../../(errors)/errorContext";
|
import { useError } from "../../(errors)/errorContext";
|
||||||
import useApi from "../../lib/useApi";
|
import useApi from "../../lib/useApi";
|
||||||
import { Page_GetTranscriptMinimal_, SourceKind } from "../../api";
|
import { PageGetTranscriptMinimal, SourceKind } from "../../api";
|
||||||
|
|
||||||
type TranscriptList = {
|
type TranscriptList = {
|
||||||
response: Page_GetTranscriptMinimal_ | null;
|
response: PageGetTranscriptMinimal | null;
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
error: Error | null;
|
error: Error | null;
|
||||||
refetch: () => void;
|
refetch: () => void;
|
||||||
@@ -16,7 +16,7 @@ const useTranscriptList = (
|
|||||||
roomId: string | null,
|
roomId: string | null,
|
||||||
searchTerm: string | null,
|
searchTerm: string | null,
|
||||||
): TranscriptList => {
|
): TranscriptList => {
|
||||||
const [response, setResponse] = useState<Page_GetTranscriptMinimal_ | null>(
|
const [response, setResponse] = useState<PageGetTranscriptMinimal | null>(
|
||||||
null,
|
null,
|
||||||
);
|
);
|
||||||
const [loading, setLoading] = useState<boolean>(true);
|
const [loading, setLoading] = useState<boolean>(true);
|
||||||
|
|||||||
@@ -118,7 +118,7 @@ const useConsentDialog = (
|
|||||||
return (
|
return (
|
||||||
<Button
|
<Button
|
||||||
ref={buttonRef}
|
ref={buttonRef}
|
||||||
colorPalette="blue"
|
colorPalette="primary"
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
handleConsent(meetingId, true).then(() => {
|
handleConsent(meetingId, true).then(() => {
|
||||||
@@ -147,9 +147,8 @@ const useConsentDialog = (
|
|||||||
recording on our servers?
|
recording on our servers?
|
||||||
</Text>
|
</Text>
|
||||||
<HStack gap={4} justifyContent="center">
|
<HStack gap={4} justifyContent="center">
|
||||||
<AcceptButton />
|
|
||||||
<Button
|
<Button
|
||||||
colorPalette="gray"
|
variant="ghost"
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
handleConsent(meetingId, false).then(() => {
|
handleConsent(meetingId, false).then(() => {
|
||||||
@@ -160,6 +159,7 @@ const useConsentDialog = (
|
|||||||
>
|
>
|
||||||
No, delete after transcription
|
No, delete after transcription
|
||||||
</Button>
|
</Button>
|
||||||
|
<AcceptButton />
|
||||||
</HStack>
|
</HStack>
|
||||||
</VStack>
|
</VStack>
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -1,37 +0,0 @@
|
|||||||
import type { BaseHttpRequest } from "./core/BaseHttpRequest";
|
|
||||||
import type { OpenAPIConfig } from "./core/OpenAPI";
|
|
||||||
import { Interceptors } from "./core/OpenAPI";
|
|
||||||
import { AxiosHttpRequest } from "./core/AxiosHttpRequest";
|
|
||||||
|
|
||||||
import { DefaultService } from "./services.gen";
|
|
||||||
|
|
||||||
type HttpRequestConstructor = new (config: OpenAPIConfig) => BaseHttpRequest;
|
|
||||||
|
|
||||||
export class OpenApi {
|
|
||||||
public readonly default: DefaultService;
|
|
||||||
|
|
||||||
public readonly request: BaseHttpRequest;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
config?: Partial<OpenAPIConfig>,
|
|
||||||
HttpRequest: HttpRequestConstructor = AxiosHttpRequest,
|
|
||||||
) {
|
|
||||||
this.request = new HttpRequest({
|
|
||||||
BASE: config?.BASE ?? "",
|
|
||||||
VERSION: config?.VERSION ?? "0.1.0",
|
|
||||||
WITH_CREDENTIALS: config?.WITH_CREDENTIALS ?? false,
|
|
||||||
CREDENTIALS: config?.CREDENTIALS ?? "include",
|
|
||||||
TOKEN: config?.TOKEN,
|
|
||||||
USERNAME: config?.USERNAME,
|
|
||||||
PASSWORD: config?.PASSWORD,
|
|
||||||
HEADERS: config?.HEADERS,
|
|
||||||
ENCODE_PATH: config?.ENCODE_PATH,
|
|
||||||
interceptors: {
|
|
||||||
request: config?.interceptors?.request ?? new Interceptors(),
|
|
||||||
response: config?.interceptors?.response ?? new Interceptors(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
this.default = new DefaultService(this.request);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
// NextAuth route handler for Authentik
|
|
||||||
// Refresh rotation has been taken from https://next-auth.js.org/v3/tutorials/refresh-token-rotation even if we are using 4.x
|
|
||||||
|
|
||||||
import NextAuth from "next-auth";
|
|
||||||
import { authOptions } from "../../../lib/auth";
|
|
||||||
|
|
||||||
const handler = NextAuth(authOptions);
|
|
||||||
|
|
||||||
export { handler as GET, handler as POST };
|
|
||||||
28
www/app/api/client.gen.ts
Normal file
28
www/app/api/client.gen.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type { ClientOptions } from "./types.gen";
|
||||||
|
import {
|
||||||
|
type Config,
|
||||||
|
type ClientOptions as DefaultClientOptions,
|
||||||
|
createClient,
|
||||||
|
createConfig,
|
||||||
|
} from "./client";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The `createClientConfig()` function will be called on client initialization
|
||||||
|
* and the returned object will become the client's initial configuration.
|
||||||
|
*
|
||||||
|
* You may want to initialize your client this way instead of calling
|
||||||
|
* `setConfig()`. This is useful for example if you're using Next.js
|
||||||
|
* to ensure your client always has the correct values.
|
||||||
|
*/
|
||||||
|
export type CreateClientConfig<T extends DefaultClientOptions = ClientOptions> =
|
||||||
|
(
|
||||||
|
override?: Config<DefaultClientOptions & T>,
|
||||||
|
) => Config<Required<DefaultClientOptions> & T>;
|
||||||
|
|
||||||
|
export const client = createClient(
|
||||||
|
createConfig<ClientOptions>({
|
||||||
|
baseUrl: "http://127.0.0.1:1250",
|
||||||
|
}),
|
||||||
|
);
|
||||||
195
www/app/api/client/client.ts
Normal file
195
www/app/api/client/client.ts
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
import type { Client, Config, RequestOptions } from "./types";
|
||||||
|
import {
|
||||||
|
buildUrl,
|
||||||
|
createConfig,
|
||||||
|
createInterceptors,
|
||||||
|
getParseAs,
|
||||||
|
mergeConfigs,
|
||||||
|
mergeHeaders,
|
||||||
|
setAuthParams,
|
||||||
|
} from "./utils";
|
||||||
|
|
||||||
|
type ReqInit = Omit<RequestInit, "body" | "headers"> & {
|
||||||
|
body?: any;
|
||||||
|
headers: ReturnType<typeof mergeHeaders>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createClient = (config: Config = {}): Client => {
|
||||||
|
let _config = mergeConfigs(createConfig(), config);
|
||||||
|
|
||||||
|
const getConfig = (): Config => ({ ..._config });
|
||||||
|
|
||||||
|
const setConfig = (config: Config): Config => {
|
||||||
|
_config = mergeConfigs(_config, config);
|
||||||
|
return getConfig();
|
||||||
|
};
|
||||||
|
|
||||||
|
const interceptors = createInterceptors<
|
||||||
|
Request,
|
||||||
|
Response,
|
||||||
|
unknown,
|
||||||
|
RequestOptions
|
||||||
|
>();
|
||||||
|
|
||||||
|
const request: Client["request"] = async (options) => {
|
||||||
|
const opts = {
|
||||||
|
..._config,
|
||||||
|
...options,
|
||||||
|
fetch: options.fetch ?? _config.fetch ?? globalThis.fetch,
|
||||||
|
headers: mergeHeaders(_config.headers, options.headers),
|
||||||
|
};
|
||||||
|
|
||||||
|
if (opts.security) {
|
||||||
|
await setAuthParams({
|
||||||
|
...opts,
|
||||||
|
security: opts.security,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.requestValidator) {
|
||||||
|
await opts.requestValidator(opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.body && opts.bodySerializer) {
|
||||||
|
opts.body = opts.bodySerializer(opts.body);
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove Content-Type header if body is empty to avoid sending invalid requests
|
||||||
|
if (opts.body === undefined || opts.body === "") {
|
||||||
|
opts.headers.delete("Content-Type");
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = buildUrl(opts);
|
||||||
|
const requestInit: ReqInit = {
|
||||||
|
redirect: "follow",
|
||||||
|
...opts,
|
||||||
|
};
|
||||||
|
|
||||||
|
let request = new Request(url, requestInit);
|
||||||
|
|
||||||
|
for (const fn of interceptors.request._fns) {
|
||||||
|
if (fn) {
|
||||||
|
request = await fn(request, opts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fetch must be assigned here, otherwise it would throw the error:
|
||||||
|
// TypeError: Failed to execute 'fetch' on 'Window': Illegal invocation
|
||||||
|
const _fetch = opts.fetch!;
|
||||||
|
let response = await _fetch(request);
|
||||||
|
|
||||||
|
for (const fn of interceptors.response._fns) {
|
||||||
|
if (fn) {
|
||||||
|
response = await fn(response, request, opts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = {
|
||||||
|
request,
|
||||||
|
response,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
if (
|
||||||
|
response.status === 204 ||
|
||||||
|
response.headers.get("Content-Length") === "0"
|
||||||
|
) {
|
||||||
|
return opts.responseStyle === "data"
|
||||||
|
? {}
|
||||||
|
: {
|
||||||
|
data: {},
|
||||||
|
...result,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const parseAs =
|
||||||
|
(opts.parseAs === "auto"
|
||||||
|
? getParseAs(response.headers.get("Content-Type"))
|
||||||
|
: opts.parseAs) ?? "json";
|
||||||
|
|
||||||
|
let data: any;
|
||||||
|
switch (parseAs) {
|
||||||
|
case "arrayBuffer":
|
||||||
|
case "blob":
|
||||||
|
case "formData":
|
||||||
|
case "json":
|
||||||
|
case "text":
|
||||||
|
data = await response[parseAs]();
|
||||||
|
break;
|
||||||
|
case "stream":
|
||||||
|
return opts.responseStyle === "data"
|
||||||
|
? response.body
|
||||||
|
: {
|
||||||
|
data: response.body,
|
||||||
|
...result,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parseAs === "json") {
|
||||||
|
if (opts.responseValidator) {
|
||||||
|
await opts.responseValidator(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.responseTransformer) {
|
||||||
|
data = await opts.responseTransformer(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return opts.responseStyle === "data"
|
||||||
|
? data
|
||||||
|
: {
|
||||||
|
data,
|
||||||
|
...result,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const textError = await response.text();
|
||||||
|
let jsonError: unknown;
|
||||||
|
|
||||||
|
try {
|
||||||
|
jsonError = JSON.parse(textError);
|
||||||
|
} catch {
|
||||||
|
// noop
|
||||||
|
}
|
||||||
|
|
||||||
|
const error = jsonError ?? textError;
|
||||||
|
let finalError = error;
|
||||||
|
|
||||||
|
for (const fn of interceptors.error._fns) {
|
||||||
|
if (fn) {
|
||||||
|
finalError = (await fn(error, response, request, opts)) as string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
finalError = finalError || ({} as string);
|
||||||
|
|
||||||
|
if (opts.throwOnError) {
|
||||||
|
throw finalError;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: we probably want to return error and improve types
|
||||||
|
return opts.responseStyle === "data"
|
||||||
|
? undefined
|
||||||
|
: {
|
||||||
|
error: finalError,
|
||||||
|
...result,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
buildUrl,
|
||||||
|
connect: (options) => request({ ...options, method: "CONNECT" }),
|
||||||
|
delete: (options) => request({ ...options, method: "DELETE" }),
|
||||||
|
get: (options) => request({ ...options, method: "GET" }),
|
||||||
|
getConfig,
|
||||||
|
head: (options) => request({ ...options, method: "HEAD" }),
|
||||||
|
interceptors,
|
||||||
|
options: (options) => request({ ...options, method: "OPTIONS" }),
|
||||||
|
patch: (options) => request({ ...options, method: "PATCH" }),
|
||||||
|
post: (options) => request({ ...options, method: "POST" }),
|
||||||
|
put: (options) => request({ ...options, method: "PUT" }),
|
||||||
|
request,
|
||||||
|
setConfig,
|
||||||
|
trace: (options) => request({ ...options, method: "TRACE" }),
|
||||||
|
};
|
||||||
|
};
|
||||||
22
www/app/api/client/index.ts
Normal file
22
www/app/api/client/index.ts
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
export type { Auth } from "../core/auth";
|
||||||
|
export type { QuerySerializerOptions } from "../core/bodySerializer";
|
||||||
|
export {
|
||||||
|
formDataBodySerializer,
|
||||||
|
jsonBodySerializer,
|
||||||
|
urlSearchParamsBodySerializer,
|
||||||
|
} from "../core/bodySerializer";
|
||||||
|
export { buildClientParams } from "../core/params";
|
||||||
|
export { createClient } from "./client";
|
||||||
|
export type {
|
||||||
|
Client,
|
||||||
|
ClientOptions,
|
||||||
|
Config,
|
||||||
|
CreateClientConfig,
|
||||||
|
Options,
|
||||||
|
OptionsLegacyParser,
|
||||||
|
RequestOptions,
|
||||||
|
RequestResult,
|
||||||
|
ResponseStyle,
|
||||||
|
TDataShape,
|
||||||
|
} from "./types";
|
||||||
|
export { createConfig, mergeHeaders } from "./utils";
|
||||||
216
www/app/api/client/types.ts
Normal file
216
www/app/api/client/types.ts
Normal file
@@ -0,0 +1,216 @@
|
|||||||
|
import type { Auth } from "../core/auth";
|
||||||
|
import type { Client as CoreClient, Config as CoreConfig } from "../core/types";
|
||||||
|
import type { Middleware } from "./utils";
|
||||||
|
|
||||||
|
export type ResponseStyle = "data" | "fields";
|
||||||
|
|
||||||
|
export interface Config<T extends ClientOptions = ClientOptions>
|
||||||
|
extends Omit<RequestInit, "body" | "headers" | "method">,
|
||||||
|
CoreConfig {
|
||||||
|
/**
|
||||||
|
* Base URL for all requests made by this client.
|
||||||
|
*/
|
||||||
|
baseUrl?: T["baseUrl"];
|
||||||
|
/**
|
||||||
|
* Fetch API implementation. You can use this option to provide a custom
|
||||||
|
* fetch instance.
|
||||||
|
*
|
||||||
|
* @default globalThis.fetch
|
||||||
|
*/
|
||||||
|
fetch?: (request: Request) => ReturnType<typeof fetch>;
|
||||||
|
/**
|
||||||
|
* Please don't use the Fetch client for Next.js applications. The `next`
|
||||||
|
* options won't have any effect.
|
||||||
|
*
|
||||||
|
* Install {@link https://www.npmjs.com/package/@hey-api/client-next `@hey-api/client-next`} instead.
|
||||||
|
*/
|
||||||
|
next?: never;
|
||||||
|
/**
|
||||||
|
* Return the response data parsed in a specified format. By default, `auto`
|
||||||
|
* will infer the appropriate method from the `Content-Type` response header.
|
||||||
|
* You can override this behavior with any of the {@link Body} methods.
|
||||||
|
* Select `stream` if you don't want to parse response data at all.
|
||||||
|
*
|
||||||
|
* @default 'auto'
|
||||||
|
*/
|
||||||
|
parseAs?:
|
||||||
|
| "arrayBuffer"
|
||||||
|
| "auto"
|
||||||
|
| "blob"
|
||||||
|
| "formData"
|
||||||
|
| "json"
|
||||||
|
| "stream"
|
||||||
|
| "text";
|
||||||
|
/**
|
||||||
|
* Should we return only data or multiple fields (data, error, response, etc.)?
|
||||||
|
*
|
||||||
|
* @default 'fields'
|
||||||
|
*/
|
||||||
|
responseStyle?: ResponseStyle;
|
||||||
|
/**
|
||||||
|
* Throw an error instead of returning it in the response?
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
throwOnError?: T["throwOnError"];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RequestOptions<
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
Url extends string = string,
|
||||||
|
> extends Config<{
|
||||||
|
responseStyle: TResponseStyle;
|
||||||
|
throwOnError: ThrowOnError;
|
||||||
|
}> {
|
||||||
|
/**
|
||||||
|
* Any body that you want to add to your request.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/fetch#body}
|
||||||
|
*/
|
||||||
|
body?: unknown;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
/**
|
||||||
|
* Security mechanism(s) to use for the request.
|
||||||
|
*/
|
||||||
|
security?: ReadonlyArray<Auth>;
|
||||||
|
url: Url;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type RequestResult<
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
> = ThrowOnError extends true
|
||||||
|
? Promise<
|
||||||
|
TResponseStyle extends "data"
|
||||||
|
? TData extends Record<string, unknown>
|
||||||
|
? TData[keyof TData]
|
||||||
|
: TData
|
||||||
|
: {
|
||||||
|
data: TData extends Record<string, unknown>
|
||||||
|
? TData[keyof TData]
|
||||||
|
: TData;
|
||||||
|
request: Request;
|
||||||
|
response: Response;
|
||||||
|
}
|
||||||
|
>
|
||||||
|
: Promise<
|
||||||
|
TResponseStyle extends "data"
|
||||||
|
?
|
||||||
|
| (TData extends Record<string, unknown>
|
||||||
|
? TData[keyof TData]
|
||||||
|
: TData)
|
||||||
|
| undefined
|
||||||
|
: (
|
||||||
|
| {
|
||||||
|
data: TData extends Record<string, unknown>
|
||||||
|
? TData[keyof TData]
|
||||||
|
: TData;
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
data: undefined;
|
||||||
|
error: TError extends Record<string, unknown>
|
||||||
|
? TError[keyof TError]
|
||||||
|
: TError;
|
||||||
|
}
|
||||||
|
) & {
|
||||||
|
request: Request;
|
||||||
|
response: Response;
|
||||||
|
}
|
||||||
|
>;
|
||||||
|
|
||||||
|
export interface ClientOptions {
|
||||||
|
baseUrl?: string;
|
||||||
|
responseStyle?: ResponseStyle;
|
||||||
|
throwOnError?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
type MethodFn = <
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
>(
|
||||||
|
options: Omit<RequestOptions<TResponseStyle, ThrowOnError>, "method">,
|
||||||
|
) => RequestResult<TData, TError, ThrowOnError, TResponseStyle>;
|
||||||
|
|
||||||
|
type RequestFn = <
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
>(
|
||||||
|
options: Omit<RequestOptions<TResponseStyle, ThrowOnError>, "method"> &
|
||||||
|
Pick<Required<RequestOptions<TResponseStyle, ThrowOnError>>, "method">,
|
||||||
|
) => RequestResult<TData, TError, ThrowOnError, TResponseStyle>;
|
||||||
|
|
||||||
|
type BuildUrlFn = <
|
||||||
|
TData extends {
|
||||||
|
body?: unknown;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
url: string;
|
||||||
|
},
|
||||||
|
>(
|
||||||
|
options: Pick<TData, "url"> & Options<TData>,
|
||||||
|
) => string;
|
||||||
|
|
||||||
|
export type Client = CoreClient<RequestFn, Config, MethodFn, BuildUrlFn> & {
|
||||||
|
interceptors: Middleware<Request, Response, unknown, RequestOptions>;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The `createClientConfig()` function will be called on client initialization
|
||||||
|
* and the returned object will become the client's initial configuration.
|
||||||
|
*
|
||||||
|
* You may want to initialize your client this way instead of calling
|
||||||
|
* `setConfig()`. This is useful for example if you're using Next.js
|
||||||
|
* to ensure your client always has the correct values.
|
||||||
|
*/
|
||||||
|
export type CreateClientConfig<T extends ClientOptions = ClientOptions> = (
|
||||||
|
override?: Config<ClientOptions & T>,
|
||||||
|
) => Config<Required<ClientOptions> & T>;
|
||||||
|
|
||||||
|
export interface TDataShape {
|
||||||
|
body?: unknown;
|
||||||
|
headers?: unknown;
|
||||||
|
path?: unknown;
|
||||||
|
query?: unknown;
|
||||||
|
url: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
type OmitKeys<T, K> = Pick<T, Exclude<keyof T, K>>;
|
||||||
|
|
||||||
|
export type Options<
|
||||||
|
TData extends TDataShape = TDataShape,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
> = OmitKeys<
|
||||||
|
RequestOptions<TResponseStyle, ThrowOnError>,
|
||||||
|
"body" | "path" | "query" | "url"
|
||||||
|
> &
|
||||||
|
Omit<TData, "url">;
|
||||||
|
|
||||||
|
export type OptionsLegacyParser<
|
||||||
|
TData = unknown,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
TResponseStyle extends ResponseStyle = "fields",
|
||||||
|
> = TData extends { body?: any }
|
||||||
|
? TData extends { headers?: any }
|
||||||
|
? OmitKeys<
|
||||||
|
RequestOptions<TResponseStyle, ThrowOnError>,
|
||||||
|
"body" | "headers" | "url"
|
||||||
|
> &
|
||||||
|
TData
|
||||||
|
: OmitKeys<RequestOptions<TResponseStyle, ThrowOnError>, "body" | "url"> &
|
||||||
|
TData &
|
||||||
|
Pick<RequestOptions<TResponseStyle, ThrowOnError>, "headers">
|
||||||
|
: TData extends { headers?: any }
|
||||||
|
? OmitKeys<RequestOptions<TResponseStyle, ThrowOnError>, "headers" | "url"> &
|
||||||
|
TData &
|
||||||
|
Pick<RequestOptions<TResponseStyle, ThrowOnError>, "body">
|
||||||
|
: OmitKeys<RequestOptions<TResponseStyle, ThrowOnError>, "url"> & TData;
|
||||||
417
www/app/api/client/utils.ts
Normal file
417
www/app/api/client/utils.ts
Normal file
@@ -0,0 +1,417 @@
|
|||||||
|
import { getAuthToken } from "../core/auth";
|
||||||
|
import type {
|
||||||
|
QuerySerializer,
|
||||||
|
QuerySerializerOptions,
|
||||||
|
} from "../core/bodySerializer";
|
||||||
|
import { jsonBodySerializer } from "../core/bodySerializer";
|
||||||
|
import {
|
||||||
|
serializeArrayParam,
|
||||||
|
serializeObjectParam,
|
||||||
|
serializePrimitiveParam,
|
||||||
|
} from "../core/pathSerializer";
|
||||||
|
import type { Client, ClientOptions, Config, RequestOptions } from "./types";
|
||||||
|
|
||||||
|
interface PathSerializer {
|
||||||
|
path: Record<string, unknown>;
|
||||||
|
url: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PATH_PARAM_RE = /\{[^{}]+\}/g;
|
||||||
|
|
||||||
|
type ArrayStyle = "form" | "spaceDelimited" | "pipeDelimited";
|
||||||
|
type MatrixStyle = "label" | "matrix" | "simple";
|
||||||
|
type ArraySeparatorStyle = ArrayStyle | MatrixStyle;
|
||||||
|
|
||||||
|
const defaultPathSerializer = ({ path, url: _url }: PathSerializer) => {
|
||||||
|
let url = _url;
|
||||||
|
const matches = _url.match(PATH_PARAM_RE);
|
||||||
|
if (matches) {
|
||||||
|
for (const match of matches) {
|
||||||
|
let explode = false;
|
||||||
|
let name = match.substring(1, match.length - 1);
|
||||||
|
let style: ArraySeparatorStyle = "simple";
|
||||||
|
|
||||||
|
if (name.endsWith("*")) {
|
||||||
|
explode = true;
|
||||||
|
name = name.substring(0, name.length - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (name.startsWith(".")) {
|
||||||
|
name = name.substring(1);
|
||||||
|
style = "label";
|
||||||
|
} else if (name.startsWith(";")) {
|
||||||
|
name = name.substring(1);
|
||||||
|
style = "matrix";
|
||||||
|
}
|
||||||
|
|
||||||
|
const value = path[name];
|
||||||
|
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
serializeArrayParam({ explode, name, style, value }),
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === "object") {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
serializeObjectParam({
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value: value as Record<string, unknown>,
|
||||||
|
valueOnly: true,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style === "matrix") {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
`;${serializePrimitiveParam({
|
||||||
|
name,
|
||||||
|
value: value as string,
|
||||||
|
})}`,
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const replaceValue = encodeURIComponent(
|
||||||
|
style === "label" ? `.${value as string}` : (value as string),
|
||||||
|
);
|
||||||
|
url = url.replace(match, replaceValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createQuerySerializer = <T = unknown>({
|
||||||
|
allowReserved,
|
||||||
|
array,
|
||||||
|
object,
|
||||||
|
}: QuerySerializerOptions = {}) => {
|
||||||
|
const querySerializer = (queryParams: T) => {
|
||||||
|
const search: string[] = [];
|
||||||
|
if (queryParams && typeof queryParams === "object") {
|
||||||
|
for (const name in queryParams) {
|
||||||
|
const value = queryParams[name];
|
||||||
|
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
const serializedArray = serializeArrayParam({
|
||||||
|
allowReserved,
|
||||||
|
explode: true,
|
||||||
|
name,
|
||||||
|
style: "form",
|
||||||
|
value,
|
||||||
|
...array,
|
||||||
|
});
|
||||||
|
if (serializedArray) search.push(serializedArray);
|
||||||
|
} else if (typeof value === "object") {
|
||||||
|
const serializedObject = serializeObjectParam({
|
||||||
|
allowReserved,
|
||||||
|
explode: true,
|
||||||
|
name,
|
||||||
|
style: "deepObject",
|
||||||
|
value: value as Record<string, unknown>,
|
||||||
|
...object,
|
||||||
|
});
|
||||||
|
if (serializedObject) search.push(serializedObject);
|
||||||
|
} else {
|
||||||
|
const serializedPrimitive = serializePrimitiveParam({
|
||||||
|
allowReserved,
|
||||||
|
name,
|
||||||
|
value: value as string,
|
||||||
|
});
|
||||||
|
if (serializedPrimitive) search.push(serializedPrimitive);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return search.join("&");
|
||||||
|
};
|
||||||
|
return querySerializer;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Infers parseAs value from provided Content-Type header.
|
||||||
|
*/
|
||||||
|
export const getParseAs = (
|
||||||
|
contentType: string | null,
|
||||||
|
): Exclude<Config["parseAs"], "auto"> => {
|
||||||
|
if (!contentType) {
|
||||||
|
// If no Content-Type header is provided, the best we can do is return the raw response body,
|
||||||
|
// which is effectively the same as the 'stream' option.
|
||||||
|
return "stream";
|
||||||
|
}
|
||||||
|
|
||||||
|
const cleanContent = contentType.split(";")[0]?.trim();
|
||||||
|
|
||||||
|
if (!cleanContent) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
cleanContent.startsWith("application/json") ||
|
||||||
|
cleanContent.endsWith("+json")
|
||||||
|
) {
|
||||||
|
return "json";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cleanContent === "multipart/form-data") {
|
||||||
|
return "formData";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
["application/", "audio/", "image/", "video/"].some((type) =>
|
||||||
|
cleanContent.startsWith(type),
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return "blob";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cleanContent.startsWith("text/")) {
|
||||||
|
return "text";
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setAuthParams = async ({
|
||||||
|
security,
|
||||||
|
...options
|
||||||
|
}: Pick<Required<RequestOptions>, "security"> &
|
||||||
|
Pick<RequestOptions, "auth" | "query"> & {
|
||||||
|
headers: Headers;
|
||||||
|
}) => {
|
||||||
|
for (const auth of security) {
|
||||||
|
const token = await getAuthToken(auth, options.auth);
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const name = auth.name ?? "Authorization";
|
||||||
|
|
||||||
|
switch (auth.in) {
|
||||||
|
case "query":
|
||||||
|
if (!options.query) {
|
||||||
|
options.query = {};
|
||||||
|
}
|
||||||
|
options.query[name] = token;
|
||||||
|
break;
|
||||||
|
case "cookie":
|
||||||
|
options.headers.append("Cookie", `${name}=${token}`);
|
||||||
|
break;
|
||||||
|
case "header":
|
||||||
|
default:
|
||||||
|
options.headers.set(name, token);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const buildUrl: Client["buildUrl"] = (options) => {
|
||||||
|
const url = getUrl({
|
||||||
|
baseUrl: options.baseUrl as string,
|
||||||
|
path: options.path,
|
||||||
|
query: options.query,
|
||||||
|
querySerializer:
|
||||||
|
typeof options.querySerializer === "function"
|
||||||
|
? options.querySerializer
|
||||||
|
: createQuerySerializer(options.querySerializer),
|
||||||
|
url: options.url,
|
||||||
|
});
|
||||||
|
return url;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getUrl = ({
|
||||||
|
baseUrl,
|
||||||
|
path,
|
||||||
|
query,
|
||||||
|
querySerializer,
|
||||||
|
url: _url,
|
||||||
|
}: {
|
||||||
|
baseUrl?: string;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
querySerializer: QuerySerializer;
|
||||||
|
url: string;
|
||||||
|
}) => {
|
||||||
|
const pathUrl = _url.startsWith("/") ? _url : `/${_url}`;
|
||||||
|
let url = (baseUrl ?? "") + pathUrl;
|
||||||
|
if (path) {
|
||||||
|
url = defaultPathSerializer({ path, url });
|
||||||
|
}
|
||||||
|
let search = query ? querySerializer(query) : "";
|
||||||
|
if (search.startsWith("?")) {
|
||||||
|
search = search.substring(1);
|
||||||
|
}
|
||||||
|
if (search) {
|
||||||
|
url += `?${search}`;
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const mergeConfigs = (a: Config, b: Config): Config => {
|
||||||
|
const config = { ...a, ...b };
|
||||||
|
if (config.baseUrl?.endsWith("/")) {
|
||||||
|
config.baseUrl = config.baseUrl.substring(0, config.baseUrl.length - 1);
|
||||||
|
}
|
||||||
|
config.headers = mergeHeaders(a.headers, b.headers);
|
||||||
|
return config;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const mergeHeaders = (
|
||||||
|
...headers: Array<Required<Config>["headers"] | undefined>
|
||||||
|
): Headers => {
|
||||||
|
const mergedHeaders = new Headers();
|
||||||
|
for (const header of headers) {
|
||||||
|
if (!header || typeof header !== "object") {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const iterator =
|
||||||
|
header instanceof Headers ? header.entries() : Object.entries(header);
|
||||||
|
|
||||||
|
for (const [key, value] of iterator) {
|
||||||
|
if (value === null) {
|
||||||
|
mergedHeaders.delete(key);
|
||||||
|
} else if (Array.isArray(value)) {
|
||||||
|
for (const v of value) {
|
||||||
|
mergedHeaders.append(key, v as string);
|
||||||
|
}
|
||||||
|
} else if (value !== undefined) {
|
||||||
|
// assume object headers are meant to be JSON stringified, i.e. their
|
||||||
|
// content value in OpenAPI specification is 'application/json'
|
||||||
|
mergedHeaders.set(
|
||||||
|
key,
|
||||||
|
typeof value === "object" ? JSON.stringify(value) : (value as string),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return mergedHeaders;
|
||||||
|
};
|
||||||
|
|
||||||
|
type ErrInterceptor<Err, Res, Req, Options> = (
|
||||||
|
error: Err,
|
||||||
|
response: Res,
|
||||||
|
request: Req,
|
||||||
|
options: Options,
|
||||||
|
) => Err | Promise<Err>;
|
||||||
|
|
||||||
|
type ReqInterceptor<Req, Options> = (
|
||||||
|
request: Req,
|
||||||
|
options: Options,
|
||||||
|
) => Req | Promise<Req>;
|
||||||
|
|
||||||
|
type ResInterceptor<Res, Req, Options> = (
|
||||||
|
response: Res,
|
||||||
|
request: Req,
|
||||||
|
options: Options,
|
||||||
|
) => Res | Promise<Res>;
|
||||||
|
|
||||||
|
class Interceptors<Interceptor> {
|
||||||
|
_fns: (Interceptor | null)[];
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this._fns = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
clear() {
|
||||||
|
this._fns = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
getInterceptorIndex(id: number | Interceptor): number {
|
||||||
|
if (typeof id === "number") {
|
||||||
|
return this._fns[id] ? id : -1;
|
||||||
|
} else {
|
||||||
|
return this._fns.indexOf(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
exists(id: number | Interceptor) {
|
||||||
|
const index = this.getInterceptorIndex(id);
|
||||||
|
return !!this._fns[index];
|
||||||
|
}
|
||||||
|
|
||||||
|
eject(id: number | Interceptor) {
|
||||||
|
const index = this.getInterceptorIndex(id);
|
||||||
|
if (this._fns[index]) {
|
||||||
|
this._fns[index] = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
update(id: number | Interceptor, fn: Interceptor) {
|
||||||
|
const index = this.getInterceptorIndex(id);
|
||||||
|
if (this._fns[index]) {
|
||||||
|
this._fns[index] = fn;
|
||||||
|
return id;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
use(fn: Interceptor) {
|
||||||
|
this._fns = [...this._fns, fn];
|
||||||
|
return this._fns.length - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// `createInterceptors()` response, meant for external use as it does not
|
||||||
|
// expose internals
|
||||||
|
export interface Middleware<Req, Res, Err, Options> {
|
||||||
|
error: Pick<
|
||||||
|
Interceptors<ErrInterceptor<Err, Res, Req, Options>>,
|
||||||
|
"eject" | "use"
|
||||||
|
>;
|
||||||
|
request: Pick<Interceptors<ReqInterceptor<Req, Options>>, "eject" | "use">;
|
||||||
|
response: Pick<
|
||||||
|
Interceptors<ResInterceptor<Res, Req, Options>>,
|
||||||
|
"eject" | "use"
|
||||||
|
>;
|
||||||
|
}
|
||||||
|
|
||||||
|
// do not add `Middleware` as return type so we can use _fns internally
|
||||||
|
export const createInterceptors = <Req, Res, Err, Options>() => ({
|
||||||
|
error: new Interceptors<ErrInterceptor<Err, Res, Req, Options>>(),
|
||||||
|
request: new Interceptors<ReqInterceptor<Req, Options>>(),
|
||||||
|
response: new Interceptors<ResInterceptor<Res, Req, Options>>(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const defaultQuerySerializer = createQuerySerializer({
|
||||||
|
allowReserved: false,
|
||||||
|
array: {
|
||||||
|
explode: true,
|
||||||
|
style: "form",
|
||||||
|
},
|
||||||
|
object: {
|
||||||
|
explode: true,
|
||||||
|
style: "deepObject",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const defaultHeaders = {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createConfig = <T extends ClientOptions = ClientOptions>(
|
||||||
|
override: Config<Omit<ClientOptions, keyof T> & T> = {},
|
||||||
|
): Config<Omit<ClientOptions, keyof T> & T> => ({
|
||||||
|
...jsonBodySerializer,
|
||||||
|
headers: defaultHeaders,
|
||||||
|
parseAs: "auto",
|
||||||
|
querySerializer: defaultQuerySerializer,
|
||||||
|
...override,
|
||||||
|
});
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
import type { ApiRequestOptions } from "./ApiRequestOptions";
|
|
||||||
import type { ApiResult } from "./ApiResult";
|
|
||||||
|
|
||||||
export class ApiError extends Error {
|
|
||||||
public readonly url: string;
|
|
||||||
public readonly status: number;
|
|
||||||
public readonly statusText: string;
|
|
||||||
public readonly body: unknown;
|
|
||||||
public readonly request: ApiRequestOptions;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
request: ApiRequestOptions,
|
|
||||||
response: ApiResult,
|
|
||||||
message: string,
|
|
||||||
) {
|
|
||||||
super(message);
|
|
||||||
|
|
||||||
this.name = "ApiError";
|
|
||||||
this.url = response.url;
|
|
||||||
this.status = response.status;
|
|
||||||
this.statusText = response.statusText;
|
|
||||||
this.body = response.body;
|
|
||||||
this.request = request;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
export type ApiRequestOptions<T = unknown> = {
|
|
||||||
readonly method:
|
|
||||||
| "GET"
|
|
||||||
| "PUT"
|
|
||||||
| "POST"
|
|
||||||
| "DELETE"
|
|
||||||
| "OPTIONS"
|
|
||||||
| "HEAD"
|
|
||||||
| "PATCH";
|
|
||||||
readonly url: string;
|
|
||||||
readonly path?: Record<string, unknown>;
|
|
||||||
readonly cookies?: Record<string, unknown>;
|
|
||||||
readonly headers?: Record<string, unknown>;
|
|
||||||
readonly query?: Record<string, unknown>;
|
|
||||||
readonly formData?: Record<string, unknown>;
|
|
||||||
readonly body?: any;
|
|
||||||
readonly mediaType?: string;
|
|
||||||
readonly responseHeader?: string;
|
|
||||||
readonly responseTransformer?: (data: unknown) => Promise<T>;
|
|
||||||
readonly errors?: Record<number | string, string>;
|
|
||||||
};
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
export type ApiResult<TData = any> = {
|
|
||||||
readonly body: TData;
|
|
||||||
readonly ok: boolean;
|
|
||||||
readonly status: number;
|
|
||||||
readonly statusText: string;
|
|
||||||
readonly url: string;
|
|
||||||
};
|
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
import type { ApiRequestOptions } from "./ApiRequestOptions";
|
|
||||||
import { BaseHttpRequest } from "./BaseHttpRequest";
|
|
||||||
import type { CancelablePromise } from "./CancelablePromise";
|
|
||||||
import type { OpenAPIConfig } from "./OpenAPI";
|
|
||||||
import { request as __request } from "./request";
|
|
||||||
|
|
||||||
export class AxiosHttpRequest extends BaseHttpRequest {
|
|
||||||
constructor(config: OpenAPIConfig) {
|
|
||||||
super(config);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request method
|
|
||||||
* @param options The request options from the service
|
|
||||||
* @returns CancelablePromise<T>
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public override request<T>(
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
): CancelablePromise<T> {
|
|
||||||
return __request(this.config, options);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
import type { ApiRequestOptions } from "./ApiRequestOptions";
|
|
||||||
import type { CancelablePromise } from "./CancelablePromise";
|
|
||||||
import type { OpenAPIConfig } from "./OpenAPI";
|
|
||||||
|
|
||||||
export abstract class BaseHttpRequest {
|
|
||||||
constructor(public readonly config: OpenAPIConfig) {}
|
|
||||||
|
|
||||||
public abstract request<T>(
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
): CancelablePromise<T>;
|
|
||||||
}
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
export class CancelError extends Error {
|
|
||||||
constructor(message: string) {
|
|
||||||
super(message);
|
|
||||||
this.name = "CancelError";
|
|
||||||
}
|
|
||||||
|
|
||||||
public get isCancelled(): boolean {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface OnCancel {
|
|
||||||
readonly isResolved: boolean;
|
|
||||||
readonly isRejected: boolean;
|
|
||||||
readonly isCancelled: boolean;
|
|
||||||
|
|
||||||
(cancelHandler: () => void): void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class CancelablePromise<T> implements Promise<T> {
|
|
||||||
private _isResolved: boolean;
|
|
||||||
private _isRejected: boolean;
|
|
||||||
private _isCancelled: boolean;
|
|
||||||
readonly cancelHandlers: (() => void)[];
|
|
||||||
readonly promise: Promise<T>;
|
|
||||||
private _resolve?: (value: T | PromiseLike<T>) => void;
|
|
||||||
private _reject?: (reason?: unknown) => void;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
executor: (
|
|
||||||
resolve: (value: T | PromiseLike<T>) => void,
|
|
||||||
reject: (reason?: unknown) => void,
|
|
||||||
onCancel: OnCancel,
|
|
||||||
) => void,
|
|
||||||
) {
|
|
||||||
this._isResolved = false;
|
|
||||||
this._isRejected = false;
|
|
||||||
this._isCancelled = false;
|
|
||||||
this.cancelHandlers = [];
|
|
||||||
this.promise = new Promise<T>((resolve, reject) => {
|
|
||||||
this._resolve = resolve;
|
|
||||||
this._reject = reject;
|
|
||||||
|
|
||||||
const onResolve = (value: T | PromiseLike<T>): void => {
|
|
||||||
if (this._isResolved || this._isRejected || this._isCancelled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this._isResolved = true;
|
|
||||||
if (this._resolve) this._resolve(value);
|
|
||||||
};
|
|
||||||
|
|
||||||
const onReject = (reason?: unknown): void => {
|
|
||||||
if (this._isResolved || this._isRejected || this._isCancelled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this._isRejected = true;
|
|
||||||
if (this._reject) this._reject(reason);
|
|
||||||
};
|
|
||||||
|
|
||||||
const onCancel = (cancelHandler: () => void): void => {
|
|
||||||
if (this._isResolved || this._isRejected || this._isCancelled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.cancelHandlers.push(cancelHandler);
|
|
||||||
};
|
|
||||||
|
|
||||||
Object.defineProperty(onCancel, "isResolved", {
|
|
||||||
get: (): boolean => this._isResolved,
|
|
||||||
});
|
|
||||||
|
|
||||||
Object.defineProperty(onCancel, "isRejected", {
|
|
||||||
get: (): boolean => this._isRejected,
|
|
||||||
});
|
|
||||||
|
|
||||||
Object.defineProperty(onCancel, "isCancelled", {
|
|
||||||
get: (): boolean => this._isCancelled,
|
|
||||||
});
|
|
||||||
|
|
||||||
return executor(onResolve, onReject, onCancel as OnCancel);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
get [Symbol.toStringTag]() {
|
|
||||||
return "Cancellable Promise";
|
|
||||||
}
|
|
||||||
|
|
||||||
public then<TResult1 = T, TResult2 = never>(
|
|
||||||
onFulfilled?: ((value: T) => TResult1 | PromiseLike<TResult1>) | null,
|
|
||||||
onRejected?: ((reason: unknown) => TResult2 | PromiseLike<TResult2>) | null,
|
|
||||||
): Promise<TResult1 | TResult2> {
|
|
||||||
return this.promise.then(onFulfilled, onRejected);
|
|
||||||
}
|
|
||||||
|
|
||||||
public catch<TResult = never>(
|
|
||||||
onRejected?: ((reason: unknown) => TResult | PromiseLike<TResult>) | null,
|
|
||||||
): Promise<T | TResult> {
|
|
||||||
return this.promise.catch(onRejected);
|
|
||||||
}
|
|
||||||
|
|
||||||
public finally(onFinally?: (() => void) | null): Promise<T> {
|
|
||||||
return this.promise.finally(onFinally);
|
|
||||||
}
|
|
||||||
|
|
||||||
public cancel(): void {
|
|
||||||
if (this._isResolved || this._isRejected || this._isCancelled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this._isCancelled = true;
|
|
||||||
if (this.cancelHandlers.length) {
|
|
||||||
try {
|
|
||||||
for (const cancelHandler of this.cancelHandlers) {
|
|
||||||
cancelHandler();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.warn("Cancellation threw an error", error);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.cancelHandlers.length = 0;
|
|
||||||
if (this._reject) this._reject(new CancelError("Request aborted"));
|
|
||||||
}
|
|
||||||
|
|
||||||
public get isCancelled(): boolean {
|
|
||||||
return this._isCancelled;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
import type { AxiosRequestConfig, AxiosResponse } from "axios";
|
|
||||||
import type { ApiRequestOptions } from "./ApiRequestOptions";
|
|
||||||
|
|
||||||
type Headers = Record<string, string>;
|
|
||||||
type Middleware<T> = (value: T) => T | Promise<T>;
|
|
||||||
type Resolver<T> = (options: ApiRequestOptions<T>) => Promise<T>;
|
|
||||||
|
|
||||||
export class Interceptors<T> {
|
|
||||||
_fns: Middleware<T>[];
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this._fns = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
eject(fn: Middleware<T>): void {
|
|
||||||
const index = this._fns.indexOf(fn);
|
|
||||||
if (index !== -1) {
|
|
||||||
this._fns = [...this._fns.slice(0, index), ...this._fns.slice(index + 1)];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
use(fn: Middleware<T>): void {
|
|
||||||
this._fns = [...this._fns, fn];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export type OpenAPIConfig = {
|
|
||||||
BASE: string;
|
|
||||||
CREDENTIALS: "include" | "omit" | "same-origin";
|
|
||||||
ENCODE_PATH?: ((path: string) => string) | undefined;
|
|
||||||
HEADERS?: Headers | Resolver<Headers> | undefined;
|
|
||||||
PASSWORD?: string | Resolver<string> | undefined;
|
|
||||||
TOKEN?: string | Resolver<string> | undefined;
|
|
||||||
USERNAME?: string | Resolver<string> | undefined;
|
|
||||||
VERSION: string;
|
|
||||||
WITH_CREDENTIALS: boolean;
|
|
||||||
interceptors: {
|
|
||||||
request: Interceptors<AxiosRequestConfig>;
|
|
||||||
response: Interceptors<AxiosResponse>;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export const OpenAPI: OpenAPIConfig = {
|
|
||||||
BASE: "",
|
|
||||||
CREDENTIALS: "include",
|
|
||||||
ENCODE_PATH: undefined,
|
|
||||||
HEADERS: undefined,
|
|
||||||
PASSWORD: undefined,
|
|
||||||
TOKEN: undefined,
|
|
||||||
USERNAME: undefined,
|
|
||||||
VERSION: "0.1.0",
|
|
||||||
WITH_CREDENTIALS: false,
|
|
||||||
interceptors: {
|
|
||||||
request: new Interceptors(),
|
|
||||||
response: new Interceptors(),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
40
www/app/api/core/auth.ts
Normal file
40
www/app/api/core/auth.ts
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
export type AuthToken = string | undefined;
|
||||||
|
|
||||||
|
export interface Auth {
|
||||||
|
/**
|
||||||
|
* Which part of the request do we use to send the auth?
|
||||||
|
*
|
||||||
|
* @default 'header'
|
||||||
|
*/
|
||||||
|
in?: "header" | "query" | "cookie";
|
||||||
|
/**
|
||||||
|
* Header or query parameter name.
|
||||||
|
*
|
||||||
|
* @default 'Authorization'
|
||||||
|
*/
|
||||||
|
name?: string;
|
||||||
|
scheme?: "basic" | "bearer";
|
||||||
|
type: "apiKey" | "http";
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getAuthToken = async (
|
||||||
|
auth: Auth,
|
||||||
|
callback: ((auth: Auth) => Promise<AuthToken> | AuthToken) | AuthToken,
|
||||||
|
): Promise<string | undefined> => {
|
||||||
|
const token =
|
||||||
|
typeof callback === "function" ? await callback(auth) : callback;
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auth.scheme === "bearer") {
|
||||||
|
return `Bearer ${token}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auth.scheme === "basic") {
|
||||||
|
return `Basic ${btoa(token)}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return token;
|
||||||
|
};
|
||||||
88
www/app/api/core/bodySerializer.ts
Normal file
88
www/app/api/core/bodySerializer.ts
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
import type {
|
||||||
|
ArrayStyle,
|
||||||
|
ObjectStyle,
|
||||||
|
SerializerOptions,
|
||||||
|
} from "./pathSerializer";
|
||||||
|
|
||||||
|
export type QuerySerializer = (query: Record<string, unknown>) => string;
|
||||||
|
|
||||||
|
export type BodySerializer = (body: any) => any;
|
||||||
|
|
||||||
|
export interface QuerySerializerOptions {
|
||||||
|
allowReserved?: boolean;
|
||||||
|
array?: SerializerOptions<ArrayStyle>;
|
||||||
|
object?: SerializerOptions<ObjectStyle>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const serializeFormDataPair = (
|
||||||
|
data: FormData,
|
||||||
|
key: string,
|
||||||
|
value: unknown,
|
||||||
|
): void => {
|
||||||
|
if (typeof value === "string" || value instanceof Blob) {
|
||||||
|
data.append(key, value);
|
||||||
|
} else {
|
||||||
|
data.append(key, JSON.stringify(value));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const serializeUrlSearchParamsPair = (
|
||||||
|
data: URLSearchParams,
|
||||||
|
key: string,
|
||||||
|
value: unknown,
|
||||||
|
): void => {
|
||||||
|
if (typeof value === "string") {
|
||||||
|
data.append(key, value);
|
||||||
|
} else {
|
||||||
|
data.append(key, JSON.stringify(value));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const formDataBodySerializer = {
|
||||||
|
bodySerializer: <T extends Record<string, any> | Array<Record<string, any>>>(
|
||||||
|
body: T,
|
||||||
|
): FormData => {
|
||||||
|
const data = new FormData();
|
||||||
|
|
||||||
|
Object.entries(body).forEach(([key, value]) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
value.forEach((v) => serializeFormDataPair(data, key, v));
|
||||||
|
} else {
|
||||||
|
serializeFormDataPair(data, key, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return data;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const jsonBodySerializer = {
|
||||||
|
bodySerializer: <T>(body: T): string =>
|
||||||
|
JSON.stringify(body, (_key, value) =>
|
||||||
|
typeof value === "bigint" ? value.toString() : value,
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
export const urlSearchParamsBodySerializer = {
|
||||||
|
bodySerializer: <T extends Record<string, any> | Array<Record<string, any>>>(
|
||||||
|
body: T,
|
||||||
|
): string => {
|
||||||
|
const data = new URLSearchParams();
|
||||||
|
|
||||||
|
Object.entries(body).forEach(([key, value]) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
value.forEach((v) => serializeUrlSearchParamsPair(data, key, v));
|
||||||
|
} else {
|
||||||
|
serializeUrlSearchParamsPair(data, key, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return data.toString();
|
||||||
|
},
|
||||||
|
};
|
||||||
151
www/app/api/core/params.ts
Normal file
151
www/app/api/core/params.ts
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
type Slot = "body" | "headers" | "path" | "query";
|
||||||
|
|
||||||
|
export type Field =
|
||||||
|
| {
|
||||||
|
in: Exclude<Slot, "body">;
|
||||||
|
/**
|
||||||
|
* Field name. This is the name we want the user to see and use.
|
||||||
|
*/
|
||||||
|
key: string;
|
||||||
|
/**
|
||||||
|
* Field mapped name. This is the name we want to use in the request.
|
||||||
|
* If omitted, we use the same value as `key`.
|
||||||
|
*/
|
||||||
|
map?: string;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
in: Extract<Slot, "body">;
|
||||||
|
/**
|
||||||
|
* Key isn't required for bodies.
|
||||||
|
*/
|
||||||
|
key?: string;
|
||||||
|
map?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface Fields {
|
||||||
|
allowExtra?: Partial<Record<Slot, boolean>>;
|
||||||
|
args?: ReadonlyArray<Field>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type FieldsConfig = ReadonlyArray<Field | Fields>;
|
||||||
|
|
||||||
|
const extraPrefixesMap: Record<string, Slot> = {
|
||||||
|
$body_: "body",
|
||||||
|
$headers_: "headers",
|
||||||
|
$path_: "path",
|
||||||
|
$query_: "query",
|
||||||
|
};
|
||||||
|
const extraPrefixes = Object.entries(extraPrefixesMap);
|
||||||
|
|
||||||
|
type KeyMap = Map<
|
||||||
|
string,
|
||||||
|
{
|
||||||
|
in: Slot;
|
||||||
|
map?: string;
|
||||||
|
}
|
||||||
|
>;
|
||||||
|
|
||||||
|
const buildKeyMap = (fields: FieldsConfig, map?: KeyMap): KeyMap => {
|
||||||
|
if (!map) {
|
||||||
|
map = new Map();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const config of fields) {
|
||||||
|
if ("in" in config) {
|
||||||
|
if (config.key) {
|
||||||
|
map.set(config.key, {
|
||||||
|
in: config.in,
|
||||||
|
map: config.map,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else if (config.args) {
|
||||||
|
buildKeyMap(config.args, map);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return map;
|
||||||
|
};
|
||||||
|
|
||||||
|
interface Params {
|
||||||
|
body: unknown;
|
||||||
|
headers: Record<string, unknown>;
|
||||||
|
path: Record<string, unknown>;
|
||||||
|
query: Record<string, unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const stripEmptySlots = (params: Params) => {
|
||||||
|
for (const [slot, value] of Object.entries(params)) {
|
||||||
|
if (value && typeof value === "object" && !Object.keys(value).length) {
|
||||||
|
delete params[slot as Slot];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const buildClientParams = (
|
||||||
|
args: ReadonlyArray<unknown>,
|
||||||
|
fields: FieldsConfig,
|
||||||
|
) => {
|
||||||
|
const params: Params = {
|
||||||
|
body: {},
|
||||||
|
headers: {},
|
||||||
|
path: {},
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
const map = buildKeyMap(fields);
|
||||||
|
|
||||||
|
let config: FieldsConfig[number] | undefined;
|
||||||
|
|
||||||
|
for (const [index, arg] of args.entries()) {
|
||||||
|
if (fields[index]) {
|
||||||
|
config = fields[index];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ("in" in config) {
|
||||||
|
if (config.key) {
|
||||||
|
const field = map.get(config.key)!;
|
||||||
|
const name = field.map || config.key;
|
||||||
|
(params[field.in] as Record<string, unknown>)[name] = arg;
|
||||||
|
} else {
|
||||||
|
params.body = arg;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (const [key, value] of Object.entries(arg ?? {})) {
|
||||||
|
const field = map.get(key);
|
||||||
|
|
||||||
|
if (field) {
|
||||||
|
const name = field.map || key;
|
||||||
|
(params[field.in] as Record<string, unknown>)[name] = value;
|
||||||
|
} else {
|
||||||
|
const extra = extraPrefixes.find(([prefix]) =>
|
||||||
|
key.startsWith(prefix),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (extra) {
|
||||||
|
const [prefix, slot] = extra;
|
||||||
|
(params[slot] as Record<string, unknown>)[
|
||||||
|
key.slice(prefix.length)
|
||||||
|
] = value;
|
||||||
|
} else {
|
||||||
|
for (const [slot, allowed] of Object.entries(
|
||||||
|
config.allowExtra ?? {},
|
||||||
|
)) {
|
||||||
|
if (allowed) {
|
||||||
|
(params[slot as Slot] as Record<string, unknown>)[key] = value;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stripEmptySlots(params);
|
||||||
|
|
||||||
|
return params;
|
||||||
|
};
|
||||||
179
www/app/api/core/pathSerializer.ts
Normal file
179
www/app/api/core/pathSerializer.ts
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
interface SerializeOptions<T>
|
||||||
|
extends SerializePrimitiveOptions,
|
||||||
|
SerializerOptions<T> {}
|
||||||
|
|
||||||
|
interface SerializePrimitiveOptions {
|
||||||
|
allowReserved?: boolean;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SerializerOptions<T> {
|
||||||
|
/**
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
explode: boolean;
|
||||||
|
style: T;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ArrayStyle = "form" | "spaceDelimited" | "pipeDelimited";
|
||||||
|
export type ArraySeparatorStyle = ArrayStyle | MatrixStyle;
|
||||||
|
type MatrixStyle = "label" | "matrix" | "simple";
|
||||||
|
export type ObjectStyle = "form" | "deepObject";
|
||||||
|
type ObjectSeparatorStyle = ObjectStyle | MatrixStyle;
|
||||||
|
|
||||||
|
interface SerializePrimitiveParam extends SerializePrimitiveOptions {
|
||||||
|
value: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const separatorArrayExplode = (style: ArraySeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case "label":
|
||||||
|
return ".";
|
||||||
|
case "matrix":
|
||||||
|
return ";";
|
||||||
|
case "simple":
|
||||||
|
return ",";
|
||||||
|
default:
|
||||||
|
return "&";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const separatorArrayNoExplode = (style: ArraySeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case "form":
|
||||||
|
return ",";
|
||||||
|
case "pipeDelimited":
|
||||||
|
return "|";
|
||||||
|
case "spaceDelimited":
|
||||||
|
return "%20";
|
||||||
|
default:
|
||||||
|
return ",";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const separatorObjectExplode = (style: ObjectSeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case "label":
|
||||||
|
return ".";
|
||||||
|
case "matrix":
|
||||||
|
return ";";
|
||||||
|
case "simple":
|
||||||
|
return ",";
|
||||||
|
default:
|
||||||
|
return "&";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializeArrayParam = ({
|
||||||
|
allowReserved,
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value,
|
||||||
|
}: SerializeOptions<ArraySeparatorStyle> & {
|
||||||
|
value: unknown[];
|
||||||
|
}) => {
|
||||||
|
if (!explode) {
|
||||||
|
const joinedValues = (
|
||||||
|
allowReserved ? value : value.map((v) => encodeURIComponent(v as string))
|
||||||
|
).join(separatorArrayNoExplode(style));
|
||||||
|
switch (style) {
|
||||||
|
case "label":
|
||||||
|
return `.${joinedValues}`;
|
||||||
|
case "matrix":
|
||||||
|
return `;${name}=${joinedValues}`;
|
||||||
|
case "simple":
|
||||||
|
return joinedValues;
|
||||||
|
default:
|
||||||
|
return `${name}=${joinedValues}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const separator = separatorArrayExplode(style);
|
||||||
|
const joinedValues = value
|
||||||
|
.map((v) => {
|
||||||
|
if (style === "label" || style === "simple") {
|
||||||
|
return allowReserved ? v : encodeURIComponent(v as string);
|
||||||
|
}
|
||||||
|
|
||||||
|
return serializePrimitiveParam({
|
||||||
|
allowReserved,
|
||||||
|
name,
|
||||||
|
value: v as string,
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.join(separator);
|
||||||
|
return style === "label" || style === "matrix"
|
||||||
|
? separator + joinedValues
|
||||||
|
: joinedValues;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializePrimitiveParam = ({
|
||||||
|
allowReserved,
|
||||||
|
name,
|
||||||
|
value,
|
||||||
|
}: SerializePrimitiveParam) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === "object") {
|
||||||
|
throw new Error(
|
||||||
|
"Deeply-nested arrays/objects aren’t supported. Provide your own `querySerializer()` to handle these.",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${name}=${allowReserved ? value : encodeURIComponent(value)}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializeObjectParam = ({
|
||||||
|
allowReserved,
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value,
|
||||||
|
valueOnly,
|
||||||
|
}: SerializeOptions<ObjectSeparatorStyle> & {
|
||||||
|
value: Record<string, unknown> | Date;
|
||||||
|
valueOnly?: boolean;
|
||||||
|
}) => {
|
||||||
|
if (value instanceof Date) {
|
||||||
|
return valueOnly ? value.toISOString() : `${name}=${value.toISOString()}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style !== "deepObject" && !explode) {
|
||||||
|
let values: string[] = [];
|
||||||
|
Object.entries(value).forEach(([key, v]) => {
|
||||||
|
values = [
|
||||||
|
...values,
|
||||||
|
key,
|
||||||
|
allowReserved ? (v as string) : encodeURIComponent(v as string),
|
||||||
|
];
|
||||||
|
});
|
||||||
|
const joinedValues = values.join(",");
|
||||||
|
switch (style) {
|
||||||
|
case "form":
|
||||||
|
return `${name}=${joinedValues}`;
|
||||||
|
case "label":
|
||||||
|
return `.${joinedValues}`;
|
||||||
|
case "matrix":
|
||||||
|
return `;${name}=${joinedValues}`;
|
||||||
|
default:
|
||||||
|
return joinedValues;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const separator = separatorObjectExplode(style);
|
||||||
|
const joinedValues = Object.entries(value)
|
||||||
|
.map(([key, v]) =>
|
||||||
|
serializePrimitiveParam({
|
||||||
|
allowReserved,
|
||||||
|
name: style === "deepObject" ? `${name}[${key}]` : key,
|
||||||
|
value: v as string,
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
.join(separator);
|
||||||
|
return style === "label" || style === "matrix"
|
||||||
|
? separator + joinedValues
|
||||||
|
: joinedValues;
|
||||||
|
};
|
||||||
@@ -1,387 +0,0 @@
|
|||||||
import axios from "axios";
|
|
||||||
import type {
|
|
||||||
AxiosError,
|
|
||||||
AxiosRequestConfig,
|
|
||||||
AxiosResponse,
|
|
||||||
AxiosInstance,
|
|
||||||
} from "axios";
|
|
||||||
|
|
||||||
import { ApiError } from "./ApiError";
|
|
||||||
import type { ApiRequestOptions } from "./ApiRequestOptions";
|
|
||||||
import type { ApiResult } from "./ApiResult";
|
|
||||||
import { CancelablePromise } from "./CancelablePromise";
|
|
||||||
import type { OnCancel } from "./CancelablePromise";
|
|
||||||
import type { OpenAPIConfig } from "./OpenAPI";
|
|
||||||
|
|
||||||
export const isString = (value: unknown): value is string => {
|
|
||||||
return typeof value === "string";
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isStringWithValue = (value: unknown): value is string => {
|
|
||||||
return isString(value) && value !== "";
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isBlob = (value: any): value is Blob => {
|
|
||||||
return value instanceof Blob;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isFormData = (value: unknown): value is FormData => {
|
|
||||||
return value instanceof FormData;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isSuccess = (status: number): boolean => {
|
|
||||||
return status >= 200 && status < 300;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const base64 = (str: string): string => {
|
|
||||||
try {
|
|
||||||
return btoa(str);
|
|
||||||
} catch (err) {
|
|
||||||
// @ts-ignore
|
|
||||||
return Buffer.from(str).toString("base64");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getQueryString = (params: Record<string, unknown>): string => {
|
|
||||||
const qs: string[] = [];
|
|
||||||
|
|
||||||
const append = (key: string, value: unknown) => {
|
|
||||||
qs.push(`${encodeURIComponent(key)}=${encodeURIComponent(String(value))}`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const encodePair = (key: string, value: unknown) => {
|
|
||||||
if (value === undefined || value === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value instanceof Date) {
|
|
||||||
append(key, value.toISOString());
|
|
||||||
} else if (Array.isArray(value)) {
|
|
||||||
value.forEach((v) => encodePair(key, v));
|
|
||||||
} else if (typeof value === "object") {
|
|
||||||
Object.entries(value).forEach(([k, v]) => encodePair(`${key}[${k}]`, v));
|
|
||||||
} else {
|
|
||||||
append(key, value);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Object.entries(params).forEach(([key, value]) => encodePair(key, value));
|
|
||||||
|
|
||||||
return qs.length ? `?${qs.join("&")}` : "";
|
|
||||||
};
|
|
||||||
|
|
||||||
const getUrl = (config: OpenAPIConfig, options: ApiRequestOptions): string => {
|
|
||||||
const encoder = config.ENCODE_PATH || encodeURI;
|
|
||||||
|
|
||||||
const path = options.url
|
|
||||||
.replace("{api-version}", config.VERSION)
|
|
||||||
.replace(/{(.*?)}/g, (substring: string, group: string) => {
|
|
||||||
if (options.path?.hasOwnProperty(group)) {
|
|
||||||
return encoder(String(options.path[group]));
|
|
||||||
}
|
|
||||||
return substring;
|
|
||||||
});
|
|
||||||
|
|
||||||
const url = config.BASE + path;
|
|
||||||
return options.query ? url + getQueryString(options.query) : url;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getFormData = (
|
|
||||||
options: ApiRequestOptions,
|
|
||||||
): FormData | undefined => {
|
|
||||||
if (options.formData) {
|
|
||||||
const formData = new FormData();
|
|
||||||
|
|
||||||
const process = (key: string, value: unknown) => {
|
|
||||||
if (isString(value) || isBlob(value)) {
|
|
||||||
formData.append(key, value);
|
|
||||||
} else {
|
|
||||||
formData.append(key, JSON.stringify(value));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Object.entries(options.formData)
|
|
||||||
.filter(([, value]) => value !== undefined && value !== null)
|
|
||||||
.forEach(([key, value]) => {
|
|
||||||
if (Array.isArray(value)) {
|
|
||||||
value.forEach((v) => process(key, v));
|
|
||||||
} else {
|
|
||||||
process(key, value);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return formData;
|
|
||||||
}
|
|
||||||
return undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Resolver<T> = (options: ApiRequestOptions<T>) => Promise<T>;
|
|
||||||
|
|
||||||
export const resolve = async <T>(
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
resolver?: T | Resolver<T>,
|
|
||||||
): Promise<T | undefined> => {
|
|
||||||
if (typeof resolver === "function") {
|
|
||||||
return (resolver as Resolver<T>)(options);
|
|
||||||
}
|
|
||||||
return resolver;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getHeaders = async <T>(
|
|
||||||
config: OpenAPIConfig,
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
): Promise<Record<string, string>> => {
|
|
||||||
const [token, username, password, additionalHeaders] = await Promise.all([
|
|
||||||
// @ts-ignore
|
|
||||||
resolve(options, config.TOKEN),
|
|
||||||
// @ts-ignore
|
|
||||||
resolve(options, config.USERNAME),
|
|
||||||
// @ts-ignore
|
|
||||||
resolve(options, config.PASSWORD),
|
|
||||||
// @ts-ignore
|
|
||||||
resolve(options, config.HEADERS),
|
|
||||||
]);
|
|
||||||
|
|
||||||
const headers = Object.entries({
|
|
||||||
Accept: "application/json",
|
|
||||||
...additionalHeaders,
|
|
||||||
...options.headers,
|
|
||||||
})
|
|
||||||
.filter(([, value]) => value !== undefined && value !== null)
|
|
||||||
.reduce(
|
|
||||||
(headers, [key, value]) => ({
|
|
||||||
...headers,
|
|
||||||
[key]: String(value),
|
|
||||||
}),
|
|
||||||
{} as Record<string, string>,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (isStringWithValue(token)) {
|
|
||||||
headers["Authorization"] = `Bearer ${token}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isStringWithValue(username) && isStringWithValue(password)) {
|
|
||||||
const credentials = base64(`${username}:${password}`);
|
|
||||||
headers["Authorization"] = `Basic ${credentials}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (options.body !== undefined) {
|
|
||||||
if (options.mediaType) {
|
|
||||||
headers["Content-Type"] = options.mediaType;
|
|
||||||
} else if (isBlob(options.body)) {
|
|
||||||
headers["Content-Type"] = options.body.type || "application/octet-stream";
|
|
||||||
} else if (isString(options.body)) {
|
|
||||||
headers["Content-Type"] = "text/plain";
|
|
||||||
} else if (!isFormData(options.body)) {
|
|
||||||
headers["Content-Type"] = "application/json";
|
|
||||||
}
|
|
||||||
} else if (options.formData !== undefined) {
|
|
||||||
if (options.mediaType) {
|
|
||||||
headers["Content-Type"] = options.mediaType;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return headers;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getRequestBody = (options: ApiRequestOptions): unknown => {
|
|
||||||
if (options.body) {
|
|
||||||
return options.body;
|
|
||||||
}
|
|
||||||
return undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const sendRequest = async <T>(
|
|
||||||
config: OpenAPIConfig,
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
url: string,
|
|
||||||
body: unknown,
|
|
||||||
formData: FormData | undefined,
|
|
||||||
headers: Record<string, string>,
|
|
||||||
onCancel: OnCancel,
|
|
||||||
axiosClient: AxiosInstance,
|
|
||||||
): Promise<AxiosResponse<T>> => {
|
|
||||||
const controller = new AbortController();
|
|
||||||
|
|
||||||
let requestConfig: AxiosRequestConfig = {
|
|
||||||
data: body ?? formData,
|
|
||||||
headers,
|
|
||||||
method: options.method,
|
|
||||||
signal: controller.signal,
|
|
||||||
url,
|
|
||||||
withCredentials: config.WITH_CREDENTIALS,
|
|
||||||
};
|
|
||||||
|
|
||||||
onCancel(() => controller.abort());
|
|
||||||
|
|
||||||
for (const fn of config.interceptors.request._fns) {
|
|
||||||
requestConfig = await fn(requestConfig);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
return await axiosClient.request(requestConfig);
|
|
||||||
} catch (error) {
|
|
||||||
const axiosError = error as AxiosError<T>;
|
|
||||||
if (axiosError.response) {
|
|
||||||
return axiosError.response;
|
|
||||||
}
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getResponseHeader = (
|
|
||||||
response: AxiosResponse<unknown>,
|
|
||||||
responseHeader?: string,
|
|
||||||
): string | undefined => {
|
|
||||||
if (responseHeader) {
|
|
||||||
const content = response.headers[responseHeader];
|
|
||||||
if (isString(content)) {
|
|
||||||
return content;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getResponseBody = (response: AxiosResponse<unknown>): unknown => {
|
|
||||||
if (response.status !== 204) {
|
|
||||||
return response.data;
|
|
||||||
}
|
|
||||||
return undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const catchErrorCodes = (
|
|
||||||
options: ApiRequestOptions,
|
|
||||||
result: ApiResult,
|
|
||||||
): void => {
|
|
||||||
const errors: Record<number, string> = {
|
|
||||||
400: "Bad Request",
|
|
||||||
401: "Unauthorized",
|
|
||||||
402: "Payment Required",
|
|
||||||
403: "Forbidden",
|
|
||||||
404: "Not Found",
|
|
||||||
405: "Method Not Allowed",
|
|
||||||
406: "Not Acceptable",
|
|
||||||
407: "Proxy Authentication Required",
|
|
||||||
408: "Request Timeout",
|
|
||||||
409: "Conflict",
|
|
||||||
410: "Gone",
|
|
||||||
411: "Length Required",
|
|
||||||
412: "Precondition Failed",
|
|
||||||
413: "Payload Too Large",
|
|
||||||
414: "URI Too Long",
|
|
||||||
415: "Unsupported Media Type",
|
|
||||||
416: "Range Not Satisfiable",
|
|
||||||
417: "Expectation Failed",
|
|
||||||
418: "Im a teapot",
|
|
||||||
421: "Misdirected Request",
|
|
||||||
422: "Unprocessable Content",
|
|
||||||
423: "Locked",
|
|
||||||
424: "Failed Dependency",
|
|
||||||
425: "Too Early",
|
|
||||||
426: "Upgrade Required",
|
|
||||||
428: "Precondition Required",
|
|
||||||
429: "Too Many Requests",
|
|
||||||
431: "Request Header Fields Too Large",
|
|
||||||
451: "Unavailable For Legal Reasons",
|
|
||||||
500: "Internal Server Error",
|
|
||||||
501: "Not Implemented",
|
|
||||||
502: "Bad Gateway",
|
|
||||||
503: "Service Unavailable",
|
|
||||||
504: "Gateway Timeout",
|
|
||||||
505: "HTTP Version Not Supported",
|
|
||||||
506: "Variant Also Negotiates",
|
|
||||||
507: "Insufficient Storage",
|
|
||||||
508: "Loop Detected",
|
|
||||||
510: "Not Extended",
|
|
||||||
511: "Network Authentication Required",
|
|
||||||
...options.errors,
|
|
||||||
};
|
|
||||||
|
|
||||||
const error = errors[result.status];
|
|
||||||
if (error) {
|
|
||||||
throw new ApiError(options, result, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!result.ok) {
|
|
||||||
const errorStatus = result.status ?? "unknown";
|
|
||||||
const errorStatusText = result.statusText ?? "unknown";
|
|
||||||
const errorBody = (() => {
|
|
||||||
try {
|
|
||||||
return JSON.stringify(result.body, null, 2);
|
|
||||||
} catch (e) {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
|
|
||||||
throw new ApiError(
|
|
||||||
options,
|
|
||||||
result,
|
|
||||||
`Generic Error: status: ${errorStatus}; status text: ${errorStatusText}; body: ${errorBody}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request method
|
|
||||||
* @param config The OpenAPI configuration object
|
|
||||||
* @param options The request options from the service
|
|
||||||
* @param axiosClient The axios client instance to use
|
|
||||||
* @returns CancelablePromise<T>
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
export const request = <T>(
|
|
||||||
config: OpenAPIConfig,
|
|
||||||
options: ApiRequestOptions<T>,
|
|
||||||
axiosClient: AxiosInstance = axios,
|
|
||||||
): CancelablePromise<T> => {
|
|
||||||
return new CancelablePromise(async (resolve, reject, onCancel) => {
|
|
||||||
try {
|
|
||||||
const url = getUrl(config, options);
|
|
||||||
const formData = getFormData(options);
|
|
||||||
const body = getRequestBody(options);
|
|
||||||
const headers = await getHeaders(config, options);
|
|
||||||
|
|
||||||
if (!onCancel.isCancelled) {
|
|
||||||
let response = await sendRequest<T>(
|
|
||||||
config,
|
|
||||||
options,
|
|
||||||
url,
|
|
||||||
body,
|
|
||||||
formData,
|
|
||||||
headers,
|
|
||||||
onCancel,
|
|
||||||
axiosClient,
|
|
||||||
);
|
|
||||||
|
|
||||||
for (const fn of config.interceptors.response._fns) {
|
|
||||||
response = await fn(response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const responseBody = getResponseBody(response);
|
|
||||||
const responseHeader = getResponseHeader(
|
|
||||||
response,
|
|
||||||
options.responseHeader,
|
|
||||||
);
|
|
||||||
|
|
||||||
let transformedBody = responseBody;
|
|
||||||
if (options.responseTransformer && isSuccess(response.status)) {
|
|
||||||
transformedBody = await options.responseTransformer(responseBody);
|
|
||||||
}
|
|
||||||
|
|
||||||
const result: ApiResult = {
|
|
||||||
url,
|
|
||||||
ok: isSuccess(response.status),
|
|
||||||
status: response.status,
|
|
||||||
statusText: response.statusText,
|
|
||||||
body: responseHeader ?? transformedBody,
|
|
||||||
};
|
|
||||||
|
|
||||||
catchErrorCodes(options, result);
|
|
||||||
|
|
||||||
resolve(result.body);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
reject(error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
118
www/app/api/core/types.ts
Normal file
118
www/app/api/core/types.ts
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
import type { Auth, AuthToken } from "./auth";
|
||||||
|
import type {
|
||||||
|
BodySerializer,
|
||||||
|
QuerySerializer,
|
||||||
|
QuerySerializerOptions,
|
||||||
|
} from "./bodySerializer";
|
||||||
|
|
||||||
|
export interface Client<
|
||||||
|
RequestFn = never,
|
||||||
|
Config = unknown,
|
||||||
|
MethodFn = never,
|
||||||
|
BuildUrlFn = never,
|
||||||
|
> {
|
||||||
|
/**
|
||||||
|
* Returns the final request URL.
|
||||||
|
*/
|
||||||
|
buildUrl: BuildUrlFn;
|
||||||
|
connect: MethodFn;
|
||||||
|
delete: MethodFn;
|
||||||
|
get: MethodFn;
|
||||||
|
getConfig: () => Config;
|
||||||
|
head: MethodFn;
|
||||||
|
options: MethodFn;
|
||||||
|
patch: MethodFn;
|
||||||
|
post: MethodFn;
|
||||||
|
put: MethodFn;
|
||||||
|
request: RequestFn;
|
||||||
|
setConfig: (config: Config) => Config;
|
||||||
|
trace: MethodFn;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Config {
|
||||||
|
/**
|
||||||
|
* Auth token or a function returning auth token. The resolved value will be
|
||||||
|
* added to the request payload as defined by its `security` array.
|
||||||
|
*/
|
||||||
|
auth?: ((auth: Auth) => Promise<AuthToken> | AuthToken) | AuthToken;
|
||||||
|
/**
|
||||||
|
* A function for serializing request body parameter. By default,
|
||||||
|
* {@link JSON.stringify()} will be used.
|
||||||
|
*/
|
||||||
|
bodySerializer?: BodySerializer | null;
|
||||||
|
/**
|
||||||
|
* An object containing any HTTP headers that you want to pre-populate your
|
||||||
|
* `Headers` object with.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/Headers/Headers#init See more}
|
||||||
|
*/
|
||||||
|
headers?:
|
||||||
|
| RequestInit["headers"]
|
||||||
|
| Record<
|
||||||
|
string,
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| (string | number | boolean)[]
|
||||||
|
| null
|
||||||
|
| undefined
|
||||||
|
| unknown
|
||||||
|
>;
|
||||||
|
/**
|
||||||
|
* The request method.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/fetch#method See more}
|
||||||
|
*/
|
||||||
|
method?:
|
||||||
|
| "CONNECT"
|
||||||
|
| "DELETE"
|
||||||
|
| "GET"
|
||||||
|
| "HEAD"
|
||||||
|
| "OPTIONS"
|
||||||
|
| "PATCH"
|
||||||
|
| "POST"
|
||||||
|
| "PUT"
|
||||||
|
| "TRACE";
|
||||||
|
/**
|
||||||
|
* A function for serializing request query parameters. By default, arrays
|
||||||
|
* will be exploded in form style, objects will be exploded in deepObject
|
||||||
|
* style, and reserved characters are percent-encoded.
|
||||||
|
*
|
||||||
|
* This method will have no effect if the native `paramsSerializer()` Axios
|
||||||
|
* API function is used.
|
||||||
|
*
|
||||||
|
* {@link https://swagger.io/docs/specification/serialization/#query View examples}
|
||||||
|
*/
|
||||||
|
querySerializer?: QuerySerializer | QuerySerializerOptions;
|
||||||
|
/**
|
||||||
|
* A function validating request data. This is useful if you want to ensure
|
||||||
|
* the request conforms to the desired shape, so it can be safely sent to
|
||||||
|
* the server.
|
||||||
|
*/
|
||||||
|
requestValidator?: (data: unknown) => Promise<unknown>;
|
||||||
|
/**
|
||||||
|
* A function transforming response data before it's returned. This is useful
|
||||||
|
* for post-processing data, e.g. converting ISO strings into Date objects.
|
||||||
|
*/
|
||||||
|
responseTransformer?: (data: unknown) => Promise<unknown>;
|
||||||
|
/**
|
||||||
|
* A function validating response data. This is useful if you want to ensure
|
||||||
|
* the response conforms to the desired shape, so it can be safely passed to
|
||||||
|
* the transformers and returned to the user.
|
||||||
|
*/
|
||||||
|
responseValidator?: (data: unknown) => Promise<unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
type IsExactlyNeverOrNeverUndefined<T> = [T] extends [never]
|
||||||
|
? true
|
||||||
|
: [T] extends [never | undefined]
|
||||||
|
? [undefined] extends [T]
|
||||||
|
? false
|
||||||
|
: true
|
||||||
|
: false;
|
||||||
|
|
||||||
|
export type OmitNever<T extends Record<string, unknown>> = {
|
||||||
|
[K in keyof T as IsExactlyNeverOrNeverUndefined<T[K]> extends true
|
||||||
|
? never
|
||||||
|
: K]: T[K];
|
||||||
|
};
|
||||||
@@ -1,9 +1,3 @@
|
|||||||
// This file is auto-generated by @hey-api/openapi-ts
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
export { OpenApi } from "./OpenApi";
|
|
||||||
export { ApiError } from "./core/ApiError";
|
|
||||||
export { BaseHttpRequest } from "./core/BaseHttpRequest";
|
|
||||||
export { CancelablePromise, CancelError } from "./core/CancelablePromise";
|
|
||||||
export { OpenAPI, type OpenAPIConfig } from "./core/OpenAPI";
|
|
||||||
export * from "./schemas.gen";
|
|
||||||
export * from "./services.gen";
|
|
||||||
export * from "./types.gen";
|
export * from "./types.gen";
|
||||||
|
export * from "./sdk.gen";
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
927
www/app/api/sdk.gen.ts
Normal file
927
www/app/api/sdk.gen.ts
Normal file
@@ -0,0 +1,927 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import {
|
||||||
|
type Options as ClientOptions,
|
||||||
|
type TDataShape,
|
||||||
|
type Client,
|
||||||
|
formDataBodySerializer,
|
||||||
|
} from "./client";
|
||||||
|
import type {
|
||||||
|
MetricsData,
|
||||||
|
MetricsResponses,
|
||||||
|
V1MeetingAudioConsentData,
|
||||||
|
V1MeetingAudioConsentResponses,
|
||||||
|
V1MeetingAudioConsentErrors,
|
||||||
|
V1RoomsListData,
|
||||||
|
V1RoomsListResponses,
|
||||||
|
V1RoomsListErrors,
|
||||||
|
V1RoomsCreateData,
|
||||||
|
V1RoomsCreateResponses,
|
||||||
|
V1RoomsCreateErrors,
|
||||||
|
V1RoomsDeleteData,
|
||||||
|
V1RoomsDeleteResponses,
|
||||||
|
V1RoomsDeleteErrors,
|
||||||
|
V1RoomsUpdateData,
|
||||||
|
V1RoomsUpdateResponses,
|
||||||
|
V1RoomsUpdateErrors,
|
||||||
|
V1RoomsCreateMeetingData,
|
||||||
|
V1RoomsCreateMeetingResponses,
|
||||||
|
V1RoomsCreateMeetingErrors,
|
||||||
|
V1TranscriptsListData,
|
||||||
|
V1TranscriptsListResponses,
|
||||||
|
V1TranscriptsListErrors,
|
||||||
|
V1TranscriptsCreateData,
|
||||||
|
V1TranscriptsCreateResponses,
|
||||||
|
V1TranscriptsCreateErrors,
|
||||||
|
V1TranscriptDeleteData,
|
||||||
|
V1TranscriptDeleteResponses,
|
||||||
|
V1TranscriptDeleteErrors,
|
||||||
|
V1TranscriptGetData,
|
||||||
|
V1TranscriptGetResponses,
|
||||||
|
V1TranscriptGetErrors,
|
||||||
|
V1TranscriptUpdateData,
|
||||||
|
V1TranscriptUpdateResponses,
|
||||||
|
V1TranscriptUpdateErrors,
|
||||||
|
V1TranscriptGetTopicsData,
|
||||||
|
V1TranscriptGetTopicsResponses,
|
||||||
|
V1TranscriptGetTopicsErrors,
|
||||||
|
V1TranscriptGetTopicsWithWordsData,
|
||||||
|
V1TranscriptGetTopicsWithWordsResponses,
|
||||||
|
V1TranscriptGetTopicsWithWordsErrors,
|
||||||
|
V1TranscriptGetTopicsWithWordsPerSpeakerData,
|
||||||
|
V1TranscriptGetTopicsWithWordsPerSpeakerResponses,
|
||||||
|
V1TranscriptGetTopicsWithWordsPerSpeakerErrors,
|
||||||
|
V1TranscriptPostToZulipData,
|
||||||
|
V1TranscriptPostToZulipResponses,
|
||||||
|
V1TranscriptPostToZulipErrors,
|
||||||
|
V1TranscriptGetAudioMp3Data,
|
||||||
|
V1TranscriptGetAudioMp3Responses,
|
||||||
|
V1TranscriptGetAudioMp3Errors,
|
||||||
|
V1TranscriptHeadAudioMp3Data,
|
||||||
|
V1TranscriptHeadAudioMp3Responses,
|
||||||
|
V1TranscriptHeadAudioMp3Errors,
|
||||||
|
V1TranscriptGetAudioWaveformData,
|
||||||
|
V1TranscriptGetAudioWaveformResponses,
|
||||||
|
V1TranscriptGetAudioWaveformErrors,
|
||||||
|
V1TranscriptGetParticipantsData,
|
||||||
|
V1TranscriptGetParticipantsResponses,
|
||||||
|
V1TranscriptGetParticipantsErrors,
|
||||||
|
V1TranscriptAddParticipantData,
|
||||||
|
V1TranscriptAddParticipantResponses,
|
||||||
|
V1TranscriptAddParticipantErrors,
|
||||||
|
V1TranscriptDeleteParticipantData,
|
||||||
|
V1TranscriptDeleteParticipantResponses,
|
||||||
|
V1TranscriptDeleteParticipantErrors,
|
||||||
|
V1TranscriptGetParticipantData,
|
||||||
|
V1TranscriptGetParticipantResponses,
|
||||||
|
V1TranscriptGetParticipantErrors,
|
||||||
|
V1TranscriptUpdateParticipantData,
|
||||||
|
V1TranscriptUpdateParticipantResponses,
|
||||||
|
V1TranscriptUpdateParticipantErrors,
|
||||||
|
V1TranscriptAssignSpeakerData,
|
||||||
|
V1TranscriptAssignSpeakerResponses,
|
||||||
|
V1TranscriptAssignSpeakerErrors,
|
||||||
|
V1TranscriptMergeSpeakerData,
|
||||||
|
V1TranscriptMergeSpeakerResponses,
|
||||||
|
V1TranscriptMergeSpeakerErrors,
|
||||||
|
V1TranscriptRecordUploadData,
|
||||||
|
V1TranscriptRecordUploadResponses,
|
||||||
|
V1TranscriptRecordUploadErrors,
|
||||||
|
V1TranscriptGetWebsocketEventsData,
|
||||||
|
V1TranscriptGetWebsocketEventsResponses,
|
||||||
|
V1TranscriptGetWebsocketEventsErrors,
|
||||||
|
V1TranscriptRecordWebrtcData,
|
||||||
|
V1TranscriptRecordWebrtcResponses,
|
||||||
|
V1TranscriptRecordWebrtcErrors,
|
||||||
|
V1TranscriptProcessData,
|
||||||
|
V1TranscriptProcessResponses,
|
||||||
|
V1TranscriptProcessErrors,
|
||||||
|
V1UserMeData,
|
||||||
|
V1UserMeResponses,
|
||||||
|
V1ZulipGetStreamsData,
|
||||||
|
V1ZulipGetStreamsResponses,
|
||||||
|
V1ZulipGetTopicsData,
|
||||||
|
V1ZulipGetTopicsResponses,
|
||||||
|
V1ZulipGetTopicsErrors,
|
||||||
|
V1WherebyWebhookData,
|
||||||
|
V1WherebyWebhookResponses,
|
||||||
|
V1WherebyWebhookErrors,
|
||||||
|
} from "./types.gen";
|
||||||
|
import { client as _heyApiClient } from "./client.gen";
|
||||||
|
|
||||||
|
export type Options<
|
||||||
|
TData extends TDataShape = TDataShape,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
> = ClientOptions<TData, ThrowOnError> & {
|
||||||
|
/**
|
||||||
|
* You can provide a client instance returned by `createClient()` instead of
|
||||||
|
* individual options. This might be also useful if you want to implement a
|
||||||
|
* custom client.
|
||||||
|
*/
|
||||||
|
client?: Client;
|
||||||
|
/**
|
||||||
|
* You can pass arbitrary values through the `meta` object. This can be
|
||||||
|
* used to access values that aren't defined as part of the SDK function.
|
||||||
|
*/
|
||||||
|
meta?: Record<string, unknown>;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Metrics
|
||||||
|
* Endpoint that serves Prometheus metrics.
|
||||||
|
*/
|
||||||
|
export const metrics = <ThrowOnError extends boolean = false>(
|
||||||
|
options?: Options<MetricsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options?.client ?? _heyApiClient).get<
|
||||||
|
MetricsResponses,
|
||||||
|
unknown,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
url: "/metrics",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Meeting Audio Consent
|
||||||
|
*/
|
||||||
|
export const v1MeetingAudioConsent = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1MeetingAudioConsentData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1MeetingAudioConsentResponses,
|
||||||
|
V1MeetingAudioConsentErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/meetings/{meeting_id}/consent",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rooms List
|
||||||
|
*/
|
||||||
|
export const v1RoomsList = <ThrowOnError extends boolean = false>(
|
||||||
|
options?: Options<V1RoomsListData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options?.client ?? _heyApiClient).get<
|
||||||
|
V1RoomsListResponses,
|
||||||
|
V1RoomsListErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/rooms",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rooms Create
|
||||||
|
*/
|
||||||
|
export const v1RoomsCreate = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1RoomsCreateData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1RoomsCreateResponses,
|
||||||
|
V1RoomsCreateErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/rooms",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rooms Delete
|
||||||
|
*/
|
||||||
|
export const v1RoomsDelete = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1RoomsDeleteData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).delete<
|
||||||
|
V1RoomsDeleteResponses,
|
||||||
|
V1RoomsDeleteErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/rooms/{room_id}",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rooms Update
|
||||||
|
*/
|
||||||
|
export const v1RoomsUpdate = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1RoomsUpdateData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).patch<
|
||||||
|
V1RoomsUpdateResponses,
|
||||||
|
V1RoomsUpdateErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/rooms/{room_id}",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rooms Create Meeting
|
||||||
|
*/
|
||||||
|
export const v1RoomsCreateMeeting = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1RoomsCreateMeetingData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1RoomsCreateMeetingResponses,
|
||||||
|
V1RoomsCreateMeetingErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/rooms/{room_name}/meeting",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcripts List
|
||||||
|
*/
|
||||||
|
export const v1TranscriptsList = <ThrowOnError extends boolean = false>(
|
||||||
|
options?: Options<V1TranscriptsListData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options?.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptsListResponses,
|
||||||
|
V1TranscriptsListErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcripts Create
|
||||||
|
*/
|
||||||
|
export const v1TranscriptsCreate = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptsCreateData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptsCreateResponses,
|
||||||
|
V1TranscriptsCreateErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Delete
|
||||||
|
*/
|
||||||
|
export const v1TranscriptDelete = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptDeleteData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).delete<
|
||||||
|
V1TranscriptDeleteResponses,
|
||||||
|
V1TranscriptDeleteErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGet = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptGetData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetResponses,
|
||||||
|
V1TranscriptGetErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Update
|
||||||
|
*/
|
||||||
|
export const v1TranscriptUpdate = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptUpdateData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).patch<
|
||||||
|
V1TranscriptUpdateResponses,
|
||||||
|
V1TranscriptUpdateErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Topics
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetTopics = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptGetTopicsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetTopicsResponses,
|
||||||
|
V1TranscriptGetTopicsErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/topics",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Topics With Words
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetTopicsWithWords = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetTopicsWithWordsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetTopicsWithWordsResponses,
|
||||||
|
V1TranscriptGetTopicsWithWordsErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/topics/with-words",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Topics With Words Per Speaker
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetTopicsWithWordsPerSpeaker = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetTopicsWithWordsPerSpeakerData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetTopicsWithWordsPerSpeakerResponses,
|
||||||
|
V1TranscriptGetTopicsWithWordsPerSpeakerErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/topics/{topic_id}/words-per-speaker",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Post To Zulip
|
||||||
|
*/
|
||||||
|
export const v1TranscriptPostToZulip = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptPostToZulipData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptPostToZulipResponses,
|
||||||
|
V1TranscriptPostToZulipErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/zulip",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Audio Mp3
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetAudioMp3 = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptGetAudioMp3Data, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetAudioMp3Responses,
|
||||||
|
V1TranscriptGetAudioMp3Errors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/audio/mp3",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Audio Mp3
|
||||||
|
*/
|
||||||
|
export const v1TranscriptHeadAudioMp3 = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptHeadAudioMp3Data, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).head<
|
||||||
|
V1TranscriptHeadAudioMp3Responses,
|
||||||
|
V1TranscriptHeadAudioMp3Errors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/audio/mp3",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Audio Waveform
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetAudioWaveform = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetAudioWaveformData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetAudioWaveformResponses,
|
||||||
|
V1TranscriptGetAudioWaveformErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/audio/waveform",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Participants
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetParticipants = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetParticipantsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetParticipantsResponses,
|
||||||
|
V1TranscriptGetParticipantsErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/participants",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Add Participant
|
||||||
|
*/
|
||||||
|
export const v1TranscriptAddParticipant = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptAddParticipantData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptAddParticipantResponses,
|
||||||
|
V1TranscriptAddParticipantErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/participants",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Delete Participant
|
||||||
|
*/
|
||||||
|
export const v1TranscriptDeleteParticipant = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptDeleteParticipantData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).delete<
|
||||||
|
V1TranscriptDeleteParticipantResponses,
|
||||||
|
V1TranscriptDeleteParticipantErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Participant
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetParticipant = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetParticipantData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetParticipantResponses,
|
||||||
|
V1TranscriptGetParticipantErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Update Participant
|
||||||
|
*/
|
||||||
|
export const v1TranscriptUpdateParticipant = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptUpdateParticipantData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).patch<
|
||||||
|
V1TranscriptUpdateParticipantResponses,
|
||||||
|
V1TranscriptUpdateParticipantErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Assign Speaker
|
||||||
|
*/
|
||||||
|
export const v1TranscriptAssignSpeaker = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptAssignSpeakerData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).patch<
|
||||||
|
V1TranscriptAssignSpeakerResponses,
|
||||||
|
V1TranscriptAssignSpeakerErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/speaker/assign",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Merge Speaker
|
||||||
|
*/
|
||||||
|
export const v1TranscriptMergeSpeaker = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptMergeSpeakerData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).patch<
|
||||||
|
V1TranscriptMergeSpeakerResponses,
|
||||||
|
V1TranscriptMergeSpeakerErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/speaker/merge",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Record Upload
|
||||||
|
*/
|
||||||
|
export const v1TranscriptRecordUpload = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptRecordUploadData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptRecordUploadResponses,
|
||||||
|
V1TranscriptRecordUploadErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
...formDataBodySerializer,
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/record/upload",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": null,
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Get Websocket Events
|
||||||
|
*/
|
||||||
|
export const v1TranscriptGetWebsocketEvents = <
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Options<V1TranscriptGetWebsocketEventsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1TranscriptGetWebsocketEventsResponses,
|
||||||
|
V1TranscriptGetWebsocketEventsErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
url: "/v1/transcripts/{transcript_id}/events",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Record Webrtc
|
||||||
|
*/
|
||||||
|
export const v1TranscriptRecordWebrtc = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptRecordWebrtcData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptRecordWebrtcResponses,
|
||||||
|
V1TranscriptRecordWebrtcErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/record/webrtc",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transcript Process
|
||||||
|
*/
|
||||||
|
export const v1TranscriptProcess = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1TranscriptProcessData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1TranscriptProcessResponses,
|
||||||
|
V1TranscriptProcessErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/transcripts/{transcript_id}/process",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* User Me
|
||||||
|
*/
|
||||||
|
export const v1UserMe = <ThrowOnError extends boolean = false>(
|
||||||
|
options?: Options<V1UserMeData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options?.client ?? _heyApiClient).get<
|
||||||
|
V1UserMeResponses,
|
||||||
|
unknown,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/me",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zulip Get Streams
|
||||||
|
* Get all Zulip streams.
|
||||||
|
*/
|
||||||
|
export const v1ZulipGetStreams = <ThrowOnError extends boolean = false>(
|
||||||
|
options?: Options<V1ZulipGetStreamsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options?.client ?? _heyApiClient).get<
|
||||||
|
V1ZulipGetStreamsResponses,
|
||||||
|
unknown,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/zulip/streams",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zulip Get Topics
|
||||||
|
* Get all topics for a specific Zulip stream.
|
||||||
|
*/
|
||||||
|
export const v1ZulipGetTopics = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1ZulipGetTopicsData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).get<
|
||||||
|
V1ZulipGetTopicsResponses,
|
||||||
|
V1ZulipGetTopicsErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: "bearer",
|
||||||
|
type: "http",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
url: "/v1/zulip/streams/{stream_id}/topics",
|
||||||
|
...options,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whereby Webhook
|
||||||
|
*/
|
||||||
|
export const v1WherebyWebhook = <ThrowOnError extends boolean = false>(
|
||||||
|
options: Options<V1WherebyWebhookData, ThrowOnError>,
|
||||||
|
) => {
|
||||||
|
return (options.client ?? _heyApiClient).post<
|
||||||
|
V1WherebyWebhookResponses,
|
||||||
|
V1WherebyWebhookErrors,
|
||||||
|
ThrowOnError
|
||||||
|
>({
|
||||||
|
url: "/v1/whereby",
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -1,860 +0,0 @@
|
|||||||
// This file is auto-generated by @hey-api/openapi-ts
|
|
||||||
|
|
||||||
import type { CancelablePromise } from "./core/CancelablePromise";
|
|
||||||
import type { BaseHttpRequest } from "./core/BaseHttpRequest";
|
|
||||||
import type {
|
|
||||||
MetricsResponse,
|
|
||||||
V1MeetingAudioConsentData,
|
|
||||||
V1MeetingAudioConsentResponse,
|
|
||||||
V1RoomsListData,
|
|
||||||
V1RoomsListResponse,
|
|
||||||
V1RoomsCreateData,
|
|
||||||
V1RoomsCreateResponse,
|
|
||||||
V1RoomsUpdateData,
|
|
||||||
V1RoomsUpdateResponse,
|
|
||||||
V1RoomsDeleteData,
|
|
||||||
V1RoomsDeleteResponse,
|
|
||||||
V1RoomsCreateMeetingData,
|
|
||||||
V1RoomsCreateMeetingResponse,
|
|
||||||
V1TranscriptsListData,
|
|
||||||
V1TranscriptsListResponse,
|
|
||||||
V1TranscriptsCreateData,
|
|
||||||
V1TranscriptsCreateResponse,
|
|
||||||
V1TranscriptGetData,
|
|
||||||
V1TranscriptGetResponse,
|
|
||||||
V1TranscriptUpdateData,
|
|
||||||
V1TranscriptUpdateResponse,
|
|
||||||
V1TranscriptDeleteData,
|
|
||||||
V1TranscriptDeleteResponse,
|
|
||||||
V1TranscriptGetTopicsData,
|
|
||||||
V1TranscriptGetTopicsResponse,
|
|
||||||
V1TranscriptGetTopicsWithWordsData,
|
|
||||||
V1TranscriptGetTopicsWithWordsResponse,
|
|
||||||
V1TranscriptGetTopicsWithWordsPerSpeakerData,
|
|
||||||
V1TranscriptGetTopicsWithWordsPerSpeakerResponse,
|
|
||||||
V1TranscriptPostToZulipData,
|
|
||||||
V1TranscriptPostToZulipResponse,
|
|
||||||
V1TranscriptHeadAudioMp3Data,
|
|
||||||
V1TranscriptHeadAudioMp3Response,
|
|
||||||
V1TranscriptGetAudioMp3Data,
|
|
||||||
V1TranscriptGetAudioMp3Response,
|
|
||||||
V1TranscriptGetAudioWaveformData,
|
|
||||||
V1TranscriptGetAudioWaveformResponse,
|
|
||||||
V1TranscriptGetParticipantsData,
|
|
||||||
V1TranscriptGetParticipantsResponse,
|
|
||||||
V1TranscriptAddParticipantData,
|
|
||||||
V1TranscriptAddParticipantResponse,
|
|
||||||
V1TranscriptGetParticipantData,
|
|
||||||
V1TranscriptGetParticipantResponse,
|
|
||||||
V1TranscriptUpdateParticipantData,
|
|
||||||
V1TranscriptUpdateParticipantResponse,
|
|
||||||
V1TranscriptDeleteParticipantData,
|
|
||||||
V1TranscriptDeleteParticipantResponse,
|
|
||||||
V1TranscriptAssignSpeakerData,
|
|
||||||
V1TranscriptAssignSpeakerResponse,
|
|
||||||
V1TranscriptMergeSpeakerData,
|
|
||||||
V1TranscriptMergeSpeakerResponse,
|
|
||||||
V1TranscriptRecordUploadData,
|
|
||||||
V1TranscriptRecordUploadResponse,
|
|
||||||
V1TranscriptGetWebsocketEventsData,
|
|
||||||
V1TranscriptGetWebsocketEventsResponse,
|
|
||||||
V1TranscriptRecordWebrtcData,
|
|
||||||
V1TranscriptRecordWebrtcResponse,
|
|
||||||
V1TranscriptProcessData,
|
|
||||||
V1TranscriptProcessResponse,
|
|
||||||
V1UserMeResponse,
|
|
||||||
V1ZulipGetStreamsResponse,
|
|
||||||
V1ZulipGetTopicsData,
|
|
||||||
V1ZulipGetTopicsResponse,
|
|
||||||
V1WherebyWebhookData,
|
|
||||||
V1WherebyWebhookResponse,
|
|
||||||
} from "./types.gen";
|
|
||||||
|
|
||||||
export class DefaultService {
|
|
||||||
constructor(public readonly httpRequest: BaseHttpRequest) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Metrics
|
|
||||||
* Endpoint that serves Prometheus metrics.
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public metrics(): CancelablePromise<MetricsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/metrics",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Meeting Audio Consent
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.meetingId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1MeetingAudioConsent(
|
|
||||||
data: V1MeetingAudioConsentData,
|
|
||||||
): CancelablePromise<V1MeetingAudioConsentResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/meetings/{meeting_id}/consent",
|
|
||||||
path: {
|
|
||||||
meeting_id: data.meetingId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rooms List
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.page Page number
|
|
||||||
* @param data.size Page size
|
|
||||||
* @returns Page_Room_ Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1RoomsList(
|
|
||||||
data: V1RoomsListData = {},
|
|
||||||
): CancelablePromise<V1RoomsListResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/rooms",
|
|
||||||
query: {
|
|
||||||
page: data.page,
|
|
||||||
size: data.size,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rooms Create
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns Room Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1RoomsCreate(
|
|
||||||
data: V1RoomsCreateData,
|
|
||||||
): CancelablePromise<V1RoomsCreateResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/rooms",
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rooms Update
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.roomId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns Room Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1RoomsUpdate(
|
|
||||||
data: V1RoomsUpdateData,
|
|
||||||
): CancelablePromise<V1RoomsUpdateResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "PATCH",
|
|
||||||
url: "/v1/rooms/{room_id}",
|
|
||||||
path: {
|
|
||||||
room_id: data.roomId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rooms Delete
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.roomId
|
|
||||||
* @returns DeletionStatus Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1RoomsDelete(
|
|
||||||
data: V1RoomsDeleteData,
|
|
||||||
): CancelablePromise<V1RoomsDeleteResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "DELETE",
|
|
||||||
url: "/v1/rooms/{room_id}",
|
|
||||||
path: {
|
|
||||||
room_id: data.roomId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rooms Create Meeting
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.roomName
|
|
||||||
* @returns Meeting Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1RoomsCreateMeeting(
|
|
||||||
data: V1RoomsCreateMeetingData,
|
|
||||||
): CancelablePromise<V1RoomsCreateMeetingResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/rooms/{room_name}/meeting",
|
|
||||||
path: {
|
|
||||||
room_name: data.roomName,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcripts List
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.sourceKind
|
|
||||||
* @param data.roomId
|
|
||||||
* @param data.searchTerm
|
|
||||||
* @param data.page Page number
|
|
||||||
* @param data.size Page size
|
|
||||||
* @returns Page_GetTranscriptMinimal_ Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptsList(
|
|
||||||
data: V1TranscriptsListData = {},
|
|
||||||
): CancelablePromise<V1TranscriptsListResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts",
|
|
||||||
query: {
|
|
||||||
source_kind: data.sourceKind,
|
|
||||||
room_id: data.roomId,
|
|
||||||
search_term: data.searchTerm,
|
|
||||||
page: data.page,
|
|
||||||
size: data.size,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcripts Create
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns GetTranscript Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptsCreate(
|
|
||||||
data: V1TranscriptsCreateData,
|
|
||||||
): CancelablePromise<V1TranscriptsCreateResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts",
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns GetTranscript Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGet(
|
|
||||||
data: V1TranscriptGetData,
|
|
||||||
): CancelablePromise<V1TranscriptGetResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Update
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns GetTranscript Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptUpdate(
|
|
||||||
data: V1TranscriptUpdateData,
|
|
||||||
): CancelablePromise<V1TranscriptUpdateResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "PATCH",
|
|
||||||
url: "/v1/transcripts/{transcript_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Delete
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns DeletionStatus Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptDelete(
|
|
||||||
data: V1TranscriptDeleteData,
|
|
||||||
): CancelablePromise<V1TranscriptDeleteResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "DELETE",
|
|
||||||
url: "/v1/transcripts/{transcript_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Topics
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns GetTranscriptTopic Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetTopics(
|
|
||||||
data: V1TranscriptGetTopicsData,
|
|
||||||
): CancelablePromise<V1TranscriptGetTopicsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/topics",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Topics With Words
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns GetTranscriptTopicWithWords Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetTopicsWithWords(
|
|
||||||
data: V1TranscriptGetTopicsWithWordsData,
|
|
||||||
): CancelablePromise<V1TranscriptGetTopicsWithWordsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/topics/with-words",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Topics With Words Per Speaker
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.topicId
|
|
||||||
* @returns GetTranscriptTopicWithWordsPerSpeaker Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetTopicsWithWordsPerSpeaker(
|
|
||||||
data: V1TranscriptGetTopicsWithWordsPerSpeakerData,
|
|
||||||
): CancelablePromise<V1TranscriptGetTopicsWithWordsPerSpeakerResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/topics/{topic_id}/words-per-speaker",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
topic_id: data.topicId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Post To Zulip
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.stream
|
|
||||||
* @param data.topic
|
|
||||||
* @param data.includeTopics
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptPostToZulip(
|
|
||||||
data: V1TranscriptPostToZulipData,
|
|
||||||
): CancelablePromise<V1TranscriptPostToZulipResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/zulip",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
stream: data.stream,
|
|
||||||
topic: data.topic,
|
|
||||||
include_topics: data.includeTopics,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Audio Mp3
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.token
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptHeadAudioMp3(
|
|
||||||
data: V1TranscriptHeadAudioMp3Data,
|
|
||||||
): CancelablePromise<V1TranscriptHeadAudioMp3Response> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "HEAD",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/audio/mp3",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
token: data.token,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Audio Mp3
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.token
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetAudioMp3(
|
|
||||||
data: V1TranscriptGetAudioMp3Data,
|
|
||||||
): CancelablePromise<V1TranscriptGetAudioMp3Response> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/audio/mp3",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
token: data.token,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Audio Waveform
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns AudioWaveform Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetAudioWaveform(
|
|
||||||
data: V1TranscriptGetAudioWaveformData,
|
|
||||||
): CancelablePromise<V1TranscriptGetAudioWaveformResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/audio/waveform",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Participants
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns Participant Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetParticipants(
|
|
||||||
data: V1TranscriptGetParticipantsData,
|
|
||||||
): CancelablePromise<V1TranscriptGetParticipantsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/participants",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Add Participant
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns Participant Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptAddParticipant(
|
|
||||||
data: V1TranscriptAddParticipantData,
|
|
||||||
): CancelablePromise<V1TranscriptAddParticipantResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/participants",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Participant
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.participantId
|
|
||||||
* @returns Participant Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetParticipant(
|
|
||||||
data: V1TranscriptGetParticipantData,
|
|
||||||
): CancelablePromise<V1TranscriptGetParticipantResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
participant_id: data.participantId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Update Participant
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.participantId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns Participant Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptUpdateParticipant(
|
|
||||||
data: V1TranscriptUpdateParticipantData,
|
|
||||||
): CancelablePromise<V1TranscriptUpdateParticipantResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "PATCH",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
participant_id: data.participantId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Delete Participant
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.participantId
|
|
||||||
* @returns DeletionStatus Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptDeleteParticipant(
|
|
||||||
data: V1TranscriptDeleteParticipantData,
|
|
||||||
): CancelablePromise<V1TranscriptDeleteParticipantResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "DELETE",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/participants/{participant_id}",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
participant_id: data.participantId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Assign Speaker
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns SpeakerAssignmentStatus Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptAssignSpeaker(
|
|
||||||
data: V1TranscriptAssignSpeakerData,
|
|
||||||
): CancelablePromise<V1TranscriptAssignSpeakerResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "PATCH",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/speaker/assign",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Merge Speaker
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns SpeakerAssignmentStatus Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptMergeSpeaker(
|
|
||||||
data: V1TranscriptMergeSpeakerData,
|
|
||||||
): CancelablePromise<V1TranscriptMergeSpeakerResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "PATCH",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/speaker/merge",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Record Upload
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.chunkNumber
|
|
||||||
* @param data.totalChunks
|
|
||||||
* @param data.formData
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptRecordUpload(
|
|
||||||
data: V1TranscriptRecordUploadData,
|
|
||||||
): CancelablePromise<V1TranscriptRecordUploadResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/record/upload",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
chunk_number: data.chunkNumber,
|
|
||||||
total_chunks: data.totalChunks,
|
|
||||||
},
|
|
||||||
formData: data.formData,
|
|
||||||
mediaType: "multipart/form-data",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Get Websocket Events
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptGetWebsocketEvents(
|
|
||||||
data: V1TranscriptGetWebsocketEventsData,
|
|
||||||
): CancelablePromise<V1TranscriptGetWebsocketEventsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/events",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Record Webrtc
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptRecordWebrtc(
|
|
||||||
data: V1TranscriptRecordWebrtcData,
|
|
||||||
): CancelablePromise<V1TranscriptRecordWebrtcResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/record/webrtc",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transcript Process
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.transcriptId
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1TranscriptProcess(
|
|
||||||
data: V1TranscriptProcessData,
|
|
||||||
): CancelablePromise<V1TranscriptProcessResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/transcripts/{transcript_id}/process",
|
|
||||||
path: {
|
|
||||||
transcript_id: data.transcriptId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* User Me
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1UserMe(): CancelablePromise<V1UserMeResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/me",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Zulip Get Streams
|
|
||||||
* Get all Zulip streams.
|
|
||||||
* @returns Stream Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1ZulipGetStreams(): CancelablePromise<V1ZulipGetStreamsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/zulip/streams",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Zulip Get Topics
|
|
||||||
* Get all topics for a specific Zulip stream.
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.streamId
|
|
||||||
* @returns Topic Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1ZulipGetTopics(
|
|
||||||
data: V1ZulipGetTopicsData,
|
|
||||||
): CancelablePromise<V1ZulipGetTopicsResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "GET",
|
|
||||||
url: "/v1/zulip/streams/{stream_id}/topics",
|
|
||||||
path: {
|
|
||||||
stream_id: data.streamId,
|
|
||||||
},
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Whereby Webhook
|
|
||||||
* @param data The data for the request.
|
|
||||||
* @param data.requestBody
|
|
||||||
* @returns unknown Successful Response
|
|
||||||
* @throws ApiError
|
|
||||||
*/
|
|
||||||
public v1WherebyWebhook(
|
|
||||||
data: V1WherebyWebhookData,
|
|
||||||
): CancelablePromise<V1WherebyWebhookResponse> {
|
|
||||||
return this.httpRequest.request({
|
|
||||||
method: "POST",
|
|
||||||
url: "/v1/whereby",
|
|
||||||
body: data.requestBody,
|
|
||||||
mediaType: "application/json",
|
|
||||||
errors: {
|
|
||||||
422: "Validation Error",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,7 +1,14 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
// Simple toaster implementation for migration
|
import {
|
||||||
// This is a temporary solution until we properly configure Chakra UI v3 toasts
|
createContext,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
useEffect,
|
||||||
|
useCallback,
|
||||||
|
} from "react";
|
||||||
|
import { createPortal } from "react-dom";
|
||||||
|
import { Box } from "@chakra-ui/react";
|
||||||
|
|
||||||
interface ToastOptions {
|
interface ToastOptions {
|
||||||
placement?: string;
|
placement?: string;
|
||||||
@@ -9,41 +16,162 @@ interface ToastOptions {
|
|||||||
render: (props: { dismiss: () => void }) => React.ReactNode;
|
render: (props: { dismiss: () => void }) => React.ReactNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface Toast extends ToastOptions {
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ToasterContextType {
|
||||||
|
toasts: Toast[];
|
||||||
|
addToast: (options: ToastOptions) => string;
|
||||||
|
removeToast: (id: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ToasterContext = createContext<ToasterContextType | null>(null);
|
||||||
|
|
||||||
|
export const ToasterProvider = ({
|
||||||
|
children,
|
||||||
|
}: {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}) => {
|
||||||
|
const [toasts, setToasts] = useState<Toast[]>([]);
|
||||||
|
|
||||||
|
const addToast = useCallback((options: ToastOptions) => {
|
||||||
|
const id = String(Date.now() + Math.random());
|
||||||
|
setToasts((prev) => [...prev, { ...options, id }]);
|
||||||
|
|
||||||
|
if (options.duration !== null) {
|
||||||
|
setTimeout(() => {
|
||||||
|
removeToast(id);
|
||||||
|
}, options.duration || 5000);
|
||||||
|
}
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const removeToast = useCallback((id: string) => {
|
||||||
|
setToasts((prev) => prev.filter((toast) => toast.id !== id));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ToasterContext.Provider value={{ toasts, addToast, removeToast }}>
|
||||||
|
{children}
|
||||||
|
<ToastContainer />
|
||||||
|
</ToasterContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const ToastContainer = () => {
|
||||||
|
const context = useContext(ToasterContext);
|
||||||
|
const [mounted, setMounted] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setMounted(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!context || !mounted) return null;
|
||||||
|
|
||||||
|
return createPortal(
|
||||||
|
<Box
|
||||||
|
position="fixed"
|
||||||
|
top="20px"
|
||||||
|
left="50%"
|
||||||
|
transform="translateX(-50%)"
|
||||||
|
zIndex={9999}
|
||||||
|
pointerEvents="none"
|
||||||
|
>
|
||||||
|
{context.toasts.map((toast) => (
|
||||||
|
<Box key={toast.id} mb={3} pointerEvents="auto">
|
||||||
|
{toast.render({ dismiss: () => context.removeToast(toast.id) })}
|
||||||
|
</Box>
|
||||||
|
))}
|
||||||
|
</Box>,
|
||||||
|
document.body,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
class ToasterClass {
|
class ToasterClass {
|
||||||
private toasts: Map<string, any> = new Map();
|
private listeners: ((action: { type: string; payload: any }) => void)[] = [];
|
||||||
private nextId = 1;
|
private nextId = 1;
|
||||||
|
private toastsMap: Map<string, boolean> = new Map();
|
||||||
|
|
||||||
|
subscribe(listener: (action: { type: string; payload: any }) => void) {
|
||||||
|
this.listeners.push(listener);
|
||||||
|
return () => {
|
||||||
|
this.listeners = this.listeners.filter((l) => l !== listener);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private notify(action: { type: string; payload: any }) {
|
||||||
|
this.listeners.forEach((listener) => listener(action));
|
||||||
|
}
|
||||||
|
|
||||||
create(options: ToastOptions): Promise<string> {
|
create(options: ToastOptions): Promise<string> {
|
||||||
const id = String(this.nextId++);
|
const id = String(this.nextId++);
|
||||||
this.toasts.set(id, options);
|
this.toastsMap.set(id, true);
|
||||||
|
this.notify({ type: "ADD_TOAST", payload: { ...options, id } });
|
||||||
|
|
||||||
// For now, we'll render toasts using a portal or modal
|
|
||||||
// This is a simplified implementation
|
|
||||||
if (typeof window !== "undefined") {
|
|
||||||
console.log("Toast created:", id, options);
|
|
||||||
|
|
||||||
// Auto-dismiss after duration if specified
|
|
||||||
if (options.duration !== null) {
|
if (options.duration !== null) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.dismiss(id);
|
this.dismiss(id);
|
||||||
}, options.duration || 5000);
|
}, options.duration || 5000);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.resolve(id);
|
return Promise.resolve(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
dismiss(id: string) {
|
dismiss(id: string) {
|
||||||
this.toasts.delete(id);
|
this.toastsMap.delete(id);
|
||||||
console.log("Toast dismissed:", id);
|
this.notify({ type: "REMOVE_TOAST", payload: id });
|
||||||
}
|
}
|
||||||
|
|
||||||
isActive(id: string): boolean {
|
isActive(id: string): boolean {
|
||||||
return this.toasts.has(id);
|
return this.toastsMap.has(id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const toaster = new ToasterClass();
|
export const toaster = new ToasterClass();
|
||||||
|
|
||||||
// Empty Toaster component for now
|
// Bridge component to connect the class-based API with React
|
||||||
export const Toaster = () => null;
|
export const Toaster = () => {
|
||||||
|
const [toasts, setToasts] = useState<Toast[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const unsubscribe = toaster.subscribe((action) => {
|
||||||
|
if (action.type === "ADD_TOAST") {
|
||||||
|
setToasts((prev) => [...prev, action.payload]);
|
||||||
|
} else if (action.type === "REMOVE_TOAST") {
|
||||||
|
setToasts((prev) =>
|
||||||
|
prev.filter((toast) => toast.id !== action.payload),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return unsubscribe;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const [mounted, setMounted] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setMounted(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!mounted) return null;
|
||||||
|
|
||||||
|
return createPortal(
|
||||||
|
<Box
|
||||||
|
position="fixed"
|
||||||
|
top="20px"
|
||||||
|
left="50%"
|
||||||
|
transform="translateX(-50%)"
|
||||||
|
zIndex={9999}
|
||||||
|
pointerEvents="none"
|
||||||
|
>
|
||||||
|
{toasts.map((toast) => (
|
||||||
|
<Box key={toast.id} mb={3} pointerEvents="auto">
|
||||||
|
{toast.render({ dismiss: () => toaster.dismiss(toast.id) })}
|
||||||
|
</Box>
|
||||||
|
))}
|
||||||
|
</Box>,
|
||||||
|
document.body,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|||||||
@@ -4,11 +4,15 @@ import { ChakraProvider } from "@chakra-ui/react";
|
|||||||
import system from "./styles/theme";
|
import system from "./styles/theme";
|
||||||
|
|
||||||
import { WherebyProvider } from "@whereby.com/browser-sdk/react";
|
import { WherebyProvider } from "@whereby.com/browser-sdk/react";
|
||||||
|
import { Toaster } from "./components/ui/toaster";
|
||||||
|
|
||||||
export function Providers({ children }: { children: React.ReactNode }) {
|
export function Providers({ children }: { children: React.ReactNode }) {
|
||||||
return (
|
return (
|
||||||
<ChakraProvider value={system}>
|
<ChakraProvider value={system}>
|
||||||
<WherebyProvider>{children}</WherebyProvider>
|
<WherebyProvider>
|
||||||
|
{children}
|
||||||
|
<Toaster />
|
||||||
|
</WherebyProvider>
|
||||||
</ChakraProvider>
|
</ChakraProvider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -55,7 +55,7 @@
|
|||||||
"author": "Andreas <andreas@monadical.com>",
|
"author": "Andreas <andreas@monadical.com>",
|
||||||
"license": "All Rights Reserved",
|
"license": "All Rights Reserved",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@hey-api/openapi-ts": "^0.48.0",
|
"@hey-api/openapi-ts": "^0.80.1",
|
||||||
"@types/react": "18.2.20",
|
"@types/react": "18.2.20",
|
||||||
"prettier": "^3.0.0",
|
"prettier": "^3.0.0",
|
||||||
"vercel": "^37.3.0"
|
"vercel": "^37.3.0"
|
||||||
|
|||||||
218
www/yarn.lock
218
www/yarn.lock
@@ -12,15 +12,6 @@
|
|||||||
resolved "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz"
|
resolved "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz"
|
||||||
integrity sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==
|
integrity sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==
|
||||||
|
|
||||||
"@apidevtools/json-schema-ref-parser@11.6.4":
|
|
||||||
version "11.6.4"
|
|
||||||
resolved "https://registry.yarnpkg.com/@apidevtools/json-schema-ref-parser/-/json-schema-ref-parser-11.6.4.tgz#0f3e02302f646471d621a8850e6a346d63c8ebd4"
|
|
||||||
integrity sha512-9K6xOqeevacvweLGik6LnZCb1fBtCOSIWQs8d096XGeqoLKC33UVMGz9+77Gw44KvbH4pKcQPWo4ZpxkXYj05w==
|
|
||||||
dependencies:
|
|
||||||
"@jsdevtools/ono" "^7.1.3"
|
|
||||||
"@types/json-schema" "^7.0.15"
|
|
||||||
js-yaml "^4.1.0"
|
|
||||||
|
|
||||||
"@ark-ui/react@5.16.1":
|
"@ark-ui/react@5.16.1":
|
||||||
version "5.16.1"
|
version "5.16.1"
|
||||||
resolved "https://registry.yarnpkg.com/@ark-ui/react/-/react-5.16.1.tgz#bfeb3c5d145e013fe4694e050e37f88f34ae4d56"
|
resolved "https://registry.yarnpkg.com/@ark-ui/react/-/react-5.16.1.tgz#bfeb3c5d145e013fe4694e050e37f88f34ae4d56"
|
||||||
@@ -517,16 +508,29 @@
|
|||||||
dependencies:
|
dependencies:
|
||||||
prop-types "^15.8.1"
|
prop-types "^15.8.1"
|
||||||
|
|
||||||
"@hey-api/openapi-ts@^0.48.0":
|
"@hey-api/json-schema-ref-parser@1.0.6":
|
||||||
version "0.48.1"
|
version "1.0.6"
|
||||||
resolved "https://registry.yarnpkg.com/@hey-api/openapi-ts/-/openapi-ts-0.48.1.tgz#a80e2372e7550057817c3dfb3742d87c395036a9"
|
resolved "https://registry.yarnpkg.com/@hey-api/json-schema-ref-parser/-/json-schema-ref-parser-1.0.6.tgz#3c415fc265846aff8455039bb90e4283e0cc30c0"
|
||||||
integrity sha512-iZBEmS12EWn4yl/nYMui+PA3hprjFR9z+9p+p+s3f0VRXPw+uZWO0yuIfCcsAw1n0isikw2uJEY4qxwlnI07AQ==
|
integrity sha512-yktiFZoWPtEW8QKS65eqKwA5MTKp88CyiL8q72WynrBs/73SAaxlSWlA2zW/DZlywZ5hX1OYzrCC0wFdvO9c2w==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@apidevtools/json-schema-ref-parser" "11.6.4"
|
"@jsdevtools/ono" "^7.1.3"
|
||||||
c12 "1.11.1"
|
"@types/json-schema" "^7.0.15"
|
||||||
camelcase "8.0.0"
|
js-yaml "^4.1.0"
|
||||||
commander "12.1.0"
|
lodash "^4.17.21"
|
||||||
|
|
||||||
|
"@hey-api/openapi-ts@^0.80.1":
|
||||||
|
version "0.80.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/@hey-api/openapi-ts/-/openapi-ts-0.80.1.tgz#3f787bc19d875d5b4e47282a33ba3362d02c8723"
|
||||||
|
integrity sha512-AC478kg36vmmrseLZNFonZ/cmXXmDzW5yWz4PVg1S8ebJsRtVRJ/QU+mtnXfzf9avN2P0pz/AO4WAe4jyFY2gA==
|
||||||
|
dependencies:
|
||||||
|
"@hey-api/json-schema-ref-parser" "1.0.6"
|
||||||
|
ansi-colors "4.1.3"
|
||||||
|
c12 "2.0.1"
|
||||||
|
color-support "1.1.3"
|
||||||
|
commander "13.0.0"
|
||||||
handlebars "4.7.8"
|
handlebars "4.7.8"
|
||||||
|
open "10.1.2"
|
||||||
|
semver "7.7.2"
|
||||||
|
|
||||||
"@humanwhocodes/module-importer@^1.0.1":
|
"@humanwhocodes/module-importer@^1.0.1":
|
||||||
version "1.0.1"
|
version "1.0.1"
|
||||||
@@ -2350,6 +2354,11 @@ acorn@^8.11.0, acorn@^8.11.3, acorn@^8.12.0, acorn@^8.4.1, acorn@^8.6.0:
|
|||||||
resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.12.1.tgz#71616bdccbe25e27a54439e0046e89ca76df2248"
|
resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.12.1.tgz#71616bdccbe25e27a54439e0046e89ca76df2248"
|
||||||
integrity sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==
|
integrity sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==
|
||||||
|
|
||||||
|
acorn@^8.14.0:
|
||||||
|
version "8.15.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.15.0.tgz#a360898bc415edaac46c8241f6383975b930b816"
|
||||||
|
integrity sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==
|
||||||
|
|
||||||
agent-base@6:
|
agent-base@6:
|
||||||
version "6.0.2"
|
version "6.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77"
|
resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77"
|
||||||
@@ -2377,6 +2386,11 @@ ajv@^6.0.0, ajv@^6.12.4:
|
|||||||
json-schema-traverse "^0.4.1"
|
json-schema-traverse "^0.4.1"
|
||||||
uri-js "^4.2.2"
|
uri-js "^4.2.2"
|
||||||
|
|
||||||
|
ansi-colors@4.1.3:
|
||||||
|
version "4.1.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.3.tgz#37611340eb2243e70cc604cad35d63270d48781b"
|
||||||
|
integrity sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==
|
||||||
|
|
||||||
ansi-regex@^5.0.1:
|
ansi-regex@^5.0.1:
|
||||||
version "5.0.1"
|
version "5.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304"
|
resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304"
|
||||||
@@ -2742,6 +2756,13 @@ buffer@^6.0.3:
|
|||||||
base64-js "^1.3.1"
|
base64-js "^1.3.1"
|
||||||
ieee754 "^1.2.1"
|
ieee754 "^1.2.1"
|
||||||
|
|
||||||
|
bundle-name@^4.1.0:
|
||||||
|
version "4.1.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/bundle-name/-/bundle-name-4.1.0.tgz#f3b96b34160d6431a19d7688135af7cfb8797889"
|
||||||
|
integrity sha512-tjwM5exMg6BGRI+kNmTntNsvdZS1X8BFYS6tnJ2hdH0kVxM6/eVZ2xy+FqStSWvYmtfFMDLIxurorHwDKfDz5Q==
|
||||||
|
dependencies:
|
||||||
|
run-applescript "^7.0.0"
|
||||||
|
|
||||||
busboy@1.6.0:
|
busboy@1.6.0:
|
||||||
version "1.6.0"
|
version "1.6.0"
|
||||||
resolved "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz"
|
resolved "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz"
|
||||||
@@ -2754,22 +2775,22 @@ bytes@3.1.0:
|
|||||||
resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6"
|
resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6"
|
||||||
integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==
|
integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==
|
||||||
|
|
||||||
c12@1.11.1:
|
c12@2.0.1:
|
||||||
version "1.11.1"
|
version "2.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/c12/-/c12-1.11.1.tgz#d5244e95407af450a523e44eb57e5b87b82f8677"
|
resolved "https://registry.yarnpkg.com/c12/-/c12-2.0.1.tgz#5702d280b31a08abba39833494c9b1202f0f5aec"
|
||||||
integrity sha512-KDU0TvSvVdaYcQKQ6iPHATGz/7p/KiVjPg4vQrB6Jg/wX9R0yl5RZxWm9IoZqaIHD2+6PZd81+KMGwRr/lRIUg==
|
integrity sha512-Z4JgsKXHG37C6PYUtIxCfLJZvo6FyhHJoClwwb9ftUkLpPSkuYqn6Tr+vnaN8hymm0kIbcg6Ey3kv/Q71k5w/A==
|
||||||
dependencies:
|
dependencies:
|
||||||
chokidar "^3.6.0"
|
chokidar "^4.0.1"
|
||||||
confbox "^0.1.7"
|
confbox "^0.1.7"
|
||||||
defu "^6.1.4"
|
defu "^6.1.4"
|
||||||
dotenv "^16.4.5"
|
dotenv "^16.4.5"
|
||||||
giget "^1.2.3"
|
giget "^1.2.3"
|
||||||
jiti "^1.21.6"
|
jiti "^2.3.0"
|
||||||
mlly "^1.7.1"
|
mlly "^1.7.1"
|
||||||
ohash "^1.1.3"
|
ohash "^1.1.4"
|
||||||
pathe "^1.1.2"
|
pathe "^1.1.2"
|
||||||
perfect-debounce "^1.0.0"
|
perfect-debounce "^1.0.0"
|
||||||
pkg-types "^1.1.1"
|
pkg-types "^1.2.0"
|
||||||
rc9 "^2.1.2"
|
rc9 "^2.1.2"
|
||||||
|
|
||||||
call-bind@^1.0.0:
|
call-bind@^1.0.0:
|
||||||
@@ -2799,11 +2820,6 @@ camelcase-css@^2.0.1:
|
|||||||
resolved "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz"
|
resolved "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz"
|
||||||
integrity sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==
|
integrity sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==
|
||||||
|
|
||||||
camelcase@8.0.0:
|
|
||||||
version "8.0.0"
|
|
||||||
resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-8.0.0.tgz#c0d36d418753fb6ad9c5e0437579745c1c14a534"
|
|
||||||
integrity sha512-8WB3Jcas3swSvjIeA2yvCJ+Miyz5l1ZmB6HFb9R1317dt9LCQoswg/BGrmAmkWVEszSrrg4RwmO46qIm2OEnSA==
|
|
||||||
|
|
||||||
caniuse-lite@^1.0.30001579, caniuse-lite@^1.0.30001646:
|
caniuse-lite@^1.0.30001579, caniuse-lite@^1.0.30001646:
|
||||||
version "1.0.30001655"
|
version "1.0.30001655"
|
||||||
resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001655.tgz#0ce881f5a19a2dcfda2ecd927df4d5c1684b982f"
|
resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001655.tgz#0ce881f5a19a2dcfda2ecd927df4d5c1684b982f"
|
||||||
@@ -2883,20 +2899,12 @@ chokidar@3.3.1:
|
|||||||
optionalDependencies:
|
optionalDependencies:
|
||||||
fsevents "~2.3.2"
|
fsevents "~2.3.2"
|
||||||
|
|
||||||
chokidar@^3.6.0:
|
chokidar@^4.0.1:
|
||||||
version "3.6.0"
|
version "4.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.6.0.tgz#197c6cc669ef2a8dc5e7b4d97ee4e092c3eb0d5b"
|
resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-4.0.3.tgz#7be37a4c03c9aee1ecfe862a4a23b2c70c205d30"
|
||||||
integrity sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==
|
integrity sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==
|
||||||
dependencies:
|
dependencies:
|
||||||
anymatch "~3.1.2"
|
readdirp "^4.0.1"
|
||||||
braces "~3.0.2"
|
|
||||||
glob-parent "~5.1.2"
|
|
||||||
is-binary-path "~2.1.0"
|
|
||||||
is-glob "~4.0.1"
|
|
||||||
normalize-path "~3.0.0"
|
|
||||||
readdirp "~3.6.0"
|
|
||||||
optionalDependencies:
|
|
||||||
fsevents "~2.3.2"
|
|
||||||
|
|
||||||
chownr@^1.1.4:
|
chownr@^1.1.4:
|
||||||
version "1.1.4"
|
version "1.1.4"
|
||||||
@@ -2974,7 +2982,7 @@ color-name@~1.1.4:
|
|||||||
resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
|
resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
|
||||||
integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
|
integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
|
||||||
|
|
||||||
color-support@^1.1.2:
|
color-support@1.1.3, color-support@^1.1.2:
|
||||||
version "1.1.3"
|
version "1.1.3"
|
||||||
resolved "https://registry.yarnpkg.com/color-support/-/color-support-1.1.3.tgz#93834379a1cc9a0c61f82f52f0d04322251bd5a2"
|
resolved "https://registry.yarnpkg.com/color-support/-/color-support-1.1.3.tgz#93834379a1cc9a0c61f82f52f0d04322251bd5a2"
|
||||||
integrity sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==
|
integrity sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==
|
||||||
@@ -2991,10 +2999,10 @@ comma-separated-tokens@^2.0.0:
|
|||||||
resolved "https://registry.yarnpkg.com/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz#4e89c9458acb61bc8fef19f4529973b2392839ee"
|
resolved "https://registry.yarnpkg.com/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz#4e89c9458acb61bc8fef19f4529973b2392839ee"
|
||||||
integrity sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==
|
integrity sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==
|
||||||
|
|
||||||
commander@12.1.0:
|
commander@13.0.0:
|
||||||
version "12.1.0"
|
version "13.0.0"
|
||||||
resolved "https://registry.yarnpkg.com/commander/-/commander-12.1.0.tgz#01423b36f501259fdaac4d0e4d60c96c991585d3"
|
resolved "https://registry.yarnpkg.com/commander/-/commander-13.0.0.tgz#1b161f60ee3ceb8074583a0f95359a4f8701845c"
|
||||||
integrity sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==
|
integrity sha512-oPYleIY8wmTVzkvQq10AEok6YcTC4sRUBl8F9gVuwchGVUCTbl/vhLTaQqutuuySYOsu8YTgV+OxKc/8Yvx+mQ==
|
||||||
|
|
||||||
commander@^4.0.0:
|
commander@^4.0.0:
|
||||||
version "4.1.1"
|
version "4.1.1"
|
||||||
@@ -3021,6 +3029,11 @@ confbox@^0.1.7:
|
|||||||
resolved "https://registry.yarnpkg.com/confbox/-/confbox-0.1.7.tgz#ccfc0a2bcae36a84838e83a3b7f770fb17d6c579"
|
resolved "https://registry.yarnpkg.com/confbox/-/confbox-0.1.7.tgz#ccfc0a2bcae36a84838e83a3b7f770fb17d6c579"
|
||||||
integrity sha512-uJcB/FKZtBMCJpK8MQji6bJHgu1tixKPxRLeGkNzBoOZzpnZUJm0jm2/sBDWcuBx1dYgxV4JU+g5hmNxCyAmdA==
|
integrity sha512-uJcB/FKZtBMCJpK8MQji6bJHgu1tixKPxRLeGkNzBoOZzpnZUJm0jm2/sBDWcuBx1dYgxV4JU+g5hmNxCyAmdA==
|
||||||
|
|
||||||
|
confbox@^0.1.8:
|
||||||
|
version "0.1.8"
|
||||||
|
resolved "https://registry.yarnpkg.com/confbox/-/confbox-0.1.8.tgz#820d73d3b3c82d9bd910652c5d4d599ef8ff8b06"
|
||||||
|
integrity sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w==
|
||||||
|
|
||||||
consola@^3.2.3:
|
consola@^3.2.3:
|
||||||
version "3.2.3"
|
version "3.2.3"
|
||||||
resolved "https://registry.yarnpkg.com/consola/-/consola-3.2.3.tgz#0741857aa88cfa0d6fd53f1cff0375136e98502f"
|
resolved "https://registry.yarnpkg.com/consola/-/consola-3.2.3.tgz#0741857aa88cfa0d6fd53f1cff0375136e98502f"
|
||||||
@@ -3153,6 +3166,19 @@ deep-is@^0.1.3:
|
|||||||
resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831"
|
resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831"
|
||||||
integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==
|
integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==
|
||||||
|
|
||||||
|
default-browser-id@^5.0.0:
|
||||||
|
version "5.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/default-browser-id/-/default-browser-id-5.0.0.tgz#a1d98bf960c15082d8a3fa69e83150ccccc3af26"
|
||||||
|
integrity sha512-A6p/pu/6fyBcA1TRz/GqWYPViplrftcW2gZC9q79ngNCKAeR/X3gcEdXQHl4KNXV+3wgIJ1CPkJQ3IHM6lcsyA==
|
||||||
|
|
||||||
|
default-browser@^5.2.1:
|
||||||
|
version "5.2.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/default-browser/-/default-browser-5.2.1.tgz#7b7ba61204ff3e425b556869ae6d3e9d9f1712cf"
|
||||||
|
integrity sha512-WY/3TUME0x3KPYdRRxEJJvXRHV4PyPoUsxtZa78lwItwRQRHhd2U9xOscaT/YTf8uCXIAjeJOFBVEh/7FtD8Xg==
|
||||||
|
dependencies:
|
||||||
|
bundle-name "^4.1.0"
|
||||||
|
default-browser-id "^5.0.0"
|
||||||
|
|
||||||
define-data-property@^1.0.1, define-data-property@^1.1.1:
|
define-data-property@^1.0.1, define-data-property@^1.1.1:
|
||||||
version "1.1.1"
|
version "1.1.1"
|
||||||
resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.1.tgz#c35f7cd0ab09883480d12ac5cb213715587800b3"
|
resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.1.tgz#c35f7cd0ab09883480d12ac5cb213715587800b3"
|
||||||
@@ -3162,6 +3188,11 @@ define-data-property@^1.0.1, define-data-property@^1.1.1:
|
|||||||
gopd "^1.0.1"
|
gopd "^1.0.1"
|
||||||
has-property-descriptors "^1.0.0"
|
has-property-descriptors "^1.0.0"
|
||||||
|
|
||||||
|
define-lazy-prop@^3.0.0:
|
||||||
|
version "3.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/define-lazy-prop/-/define-lazy-prop-3.0.0.tgz#dbb19adfb746d7fc6d734a06b72f4a00d021255f"
|
||||||
|
integrity sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg==
|
||||||
|
|
||||||
define-properties@^1.1.3, define-properties@^1.2.0, define-properties@^1.2.1:
|
define-properties@^1.1.3, define-properties@^1.2.0, define-properties@^1.2.1:
|
||||||
version "1.2.1"
|
version "1.2.1"
|
||||||
resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.1.tgz#10781cc616eb951a80a034bafcaa7377f6af2b6c"
|
resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.1.tgz#10781cc616eb951a80a034bafcaa7377f6af2b6c"
|
||||||
@@ -4668,6 +4699,11 @@ is-date-object@^1.0.1, is-date-object@^1.0.5:
|
|||||||
dependencies:
|
dependencies:
|
||||||
has-tostringtag "^1.0.0"
|
has-tostringtag "^1.0.0"
|
||||||
|
|
||||||
|
is-docker@^3.0.0:
|
||||||
|
version "3.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-3.0.0.tgz#90093aa3106277d8a77a5910dbae71747e15a200"
|
||||||
|
integrity sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ==
|
||||||
|
|
||||||
is-extglob@^2.1.1:
|
is-extglob@^2.1.1:
|
||||||
version "2.1.1"
|
version "2.1.1"
|
||||||
resolved "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz"
|
resolved "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz"
|
||||||
@@ -4699,6 +4735,13 @@ is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3, is-glob@~4.0.1:
|
|||||||
dependencies:
|
dependencies:
|
||||||
is-extglob "^2.1.1"
|
is-extglob "^2.1.1"
|
||||||
|
|
||||||
|
is-inside-container@^1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/is-inside-container/-/is-inside-container-1.0.0.tgz#e81fba699662eb31dbdaf26766a61d4814717ea4"
|
||||||
|
integrity sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA==
|
||||||
|
dependencies:
|
||||||
|
is-docker "^3.0.0"
|
||||||
|
|
||||||
is-map@^2.0.1:
|
is-map@^2.0.1:
|
||||||
version "2.0.2"
|
version "2.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/is-map/-/is-map-2.0.2.tgz#00922db8c9bf73e81b7a335827bc2a43f2b91127"
|
resolved "https://registry.yarnpkg.com/is-map/-/is-map-2.0.2.tgz#00922db8c9bf73e81b7a335827bc2a43f2b91127"
|
||||||
@@ -4809,6 +4852,13 @@ is-weakset@^2.0.1:
|
|||||||
call-bind "^1.0.2"
|
call-bind "^1.0.2"
|
||||||
get-intrinsic "^1.1.1"
|
get-intrinsic "^1.1.1"
|
||||||
|
|
||||||
|
is-wsl@^3.1.0:
|
||||||
|
version "3.1.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-3.1.0.tgz#e1c657e39c10090afcbedec61720f6b924c3cbd2"
|
||||||
|
integrity sha512-UcVfVfaK4Sc4m7X3dUSoHoozQGBEFeDC+zVo06t98xe8CzHSZZBekNXH+tu0NalHolcJ/QAGqS46Hef7QXBIMw==
|
||||||
|
dependencies:
|
||||||
|
is-inside-container "^1.0.0"
|
||||||
|
|
||||||
isarray@0.0.1:
|
isarray@0.0.1:
|
||||||
version "0.0.1"
|
version "0.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/isarray/-/isarray-0.0.1.tgz#8a18acfca9a8f4177e09abfc6038939b05d1eedf"
|
resolved "https://registry.yarnpkg.com/isarray/-/isarray-0.0.1.tgz#8a18acfca9a8f4177e09abfc6038939b05d1eedf"
|
||||||
@@ -4871,10 +4921,10 @@ jiti@^1.18.2:
|
|||||||
resolved "https://registry.npmjs.org/jiti/-/jiti-1.19.1.tgz"
|
resolved "https://registry.npmjs.org/jiti/-/jiti-1.19.1.tgz"
|
||||||
integrity sha512-oVhqoRDaBXf7sjkll95LHVS6Myyyb1zaunVwk4Z0+WPSW4gjS0pl01zYKHScTuyEhQsFxV5L4DR5r+YqSyqyyg==
|
integrity sha512-oVhqoRDaBXf7sjkll95LHVS6Myyyb1zaunVwk4Z0+WPSW4gjS0pl01zYKHScTuyEhQsFxV5L4DR5r+YqSyqyyg==
|
||||||
|
|
||||||
jiti@^1.21.6:
|
jiti@^2.3.0:
|
||||||
version "1.21.6"
|
version "2.5.1"
|
||||||
resolved "https://registry.yarnpkg.com/jiti/-/jiti-1.21.6.tgz#6c7f7398dd4b3142767f9a168af2f317a428d268"
|
resolved "https://registry.yarnpkg.com/jiti/-/jiti-2.5.1.tgz#bd099c1c2be1c59bbea4e5adcd127363446759d0"
|
||||||
integrity sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w==
|
integrity sha512-twQoecYPiVA5K/h6SxtORw/Bs3ar+mLUtoPSc7iMXzQzK8d7eJ/R09wmTwAjiamETn1cXYPGfNnu7DMoHgu12w==
|
||||||
|
|
||||||
jose@^4.15.5:
|
jose@^4.15.5:
|
||||||
version "4.15.9"
|
version "4.15.9"
|
||||||
@@ -5052,6 +5102,11 @@ lodash.merge@^4.6.2:
|
|||||||
resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a"
|
resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a"
|
||||||
integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==
|
integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==
|
||||||
|
|
||||||
|
lodash@^4.17.21:
|
||||||
|
version "4.17.21"
|
||||||
|
resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
|
||||||
|
integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
|
||||||
|
|
||||||
loose-envify@^1.0.0, loose-envify@^1.1.0, loose-envify@^1.4.0:
|
loose-envify@^1.0.0, loose-envify@^1.1.0, loose-envify@^1.4.0:
|
||||||
version "1.4.0"
|
version "1.4.0"
|
||||||
resolved "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz"
|
resolved "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz"
|
||||||
@@ -5512,6 +5567,16 @@ mlly@^1.7.1:
|
|||||||
pkg-types "^1.1.1"
|
pkg-types "^1.1.1"
|
||||||
ufo "^1.5.3"
|
ufo "^1.5.3"
|
||||||
|
|
||||||
|
mlly@^1.7.4:
|
||||||
|
version "1.7.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/mlly/-/mlly-1.7.4.tgz#3d7295ea2358ec7a271eaa5d000a0f84febe100f"
|
||||||
|
integrity sha512-qmdSIPC4bDJXgZTCR7XosJiNKySV7O215tsPtDN9iEO/7q/76b/ijtgRu/+epFXSJhijtTCCGp3DWS549P3xKw==
|
||||||
|
dependencies:
|
||||||
|
acorn "^8.14.0"
|
||||||
|
pathe "^2.0.1"
|
||||||
|
pkg-types "^1.3.0"
|
||||||
|
ufo "^1.5.4"
|
||||||
|
|
||||||
mri@1.2.0:
|
mri@1.2.0:
|
||||||
version "1.2.0"
|
version "1.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/mri/-/mri-1.2.0.tgz#6721480fec2a11a4889861115a48b6cbe7cc8f0b"
|
resolved "https://registry.yarnpkg.com/mri/-/mri-1.2.0.tgz#6721480fec2a11a4889861115a48b6cbe7cc8f0b"
|
||||||
@@ -5793,6 +5858,11 @@ ohash@^1.1.3:
|
|||||||
resolved "https://registry.yarnpkg.com/ohash/-/ohash-1.1.3.tgz#f12c3c50bfe7271ce3fd1097d42568122ccdcf07"
|
resolved "https://registry.yarnpkg.com/ohash/-/ohash-1.1.3.tgz#f12c3c50bfe7271ce3fd1097d42568122ccdcf07"
|
||||||
integrity sha512-zuHHiGTYTA1sYJ/wZN+t5HKZaH23i4yI1HMwbuXm24Nid7Dv0KcuRlKoNKS9UNfAVSBlnGLcuQrnOKWOZoEGaw==
|
integrity sha512-zuHHiGTYTA1sYJ/wZN+t5HKZaH23i4yI1HMwbuXm24Nid7Dv0KcuRlKoNKS9UNfAVSBlnGLcuQrnOKWOZoEGaw==
|
||||||
|
|
||||||
|
ohash@^1.1.4:
|
||||||
|
version "1.1.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/ohash/-/ohash-1.1.6.tgz#9ff7b0271d7076290794537d68ec2b40a60d133e"
|
||||||
|
integrity sha512-TBu7PtV8YkAZn0tSxobKY2n2aAQva936lhRrj6957aDaCf9IEtqsKbgMzXE/F/sjqYOwmrukeORHNLe5glk7Cg==
|
||||||
|
|
||||||
oidc-token-hash@^5.0.3:
|
oidc-token-hash@^5.0.3:
|
||||||
version "5.0.3"
|
version "5.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/oidc-token-hash/-/oidc-token-hash-5.0.3.tgz#9a229f0a1ce9d4fc89bcaee5478c97a889e7b7b6"
|
resolved "https://registry.yarnpkg.com/oidc-token-hash/-/oidc-token-hash-5.0.3.tgz#9a229f0a1ce9d4fc89bcaee5478c97a889e7b7b6"
|
||||||
@@ -5826,6 +5896,16 @@ onetime@^6.0.0:
|
|||||||
dependencies:
|
dependencies:
|
||||||
mimic-fn "^4.0.0"
|
mimic-fn "^4.0.0"
|
||||||
|
|
||||||
|
open@10.1.2:
|
||||||
|
version "10.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/open/-/open-10.1.2.tgz#d5df40984755c9a9c3c93df8156a12467e882925"
|
||||||
|
integrity sha512-cxN6aIDPz6rm8hbebcP7vrQNhvRcveZoJU72Y7vskh4oIm+BZwBECnx5nTmrlres1Qapvx27Qo1Auukpf8PKXw==
|
||||||
|
dependencies:
|
||||||
|
default-browser "^5.2.1"
|
||||||
|
define-lazy-prop "^3.0.0"
|
||||||
|
is-inside-container "^1.0.0"
|
||||||
|
is-wsl "^3.1.0"
|
||||||
|
|
||||||
openid-client@^5.4.0:
|
openid-client@^5.4.0:
|
||||||
version "5.6.5"
|
version "5.6.5"
|
||||||
resolved "https://registry.yarnpkg.com/openid-client/-/openid-client-5.6.5.tgz#c149ad07b9c399476dc347097e297bbe288b8b00"
|
resolved "https://registry.yarnpkg.com/openid-client/-/openid-client-5.6.5.tgz#c149ad07b9c399476dc347097e297bbe288b8b00"
|
||||||
@@ -5967,6 +6047,11 @@ pathe@^1.1.2:
|
|||||||
resolved "https://registry.yarnpkg.com/pathe/-/pathe-1.1.2.tgz#6c4cb47a945692e48a1ddd6e4094d170516437ec"
|
resolved "https://registry.yarnpkg.com/pathe/-/pathe-1.1.2.tgz#6c4cb47a945692e48a1ddd6e4094d170516437ec"
|
||||||
integrity sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==
|
integrity sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==
|
||||||
|
|
||||||
|
pathe@^2.0.1:
|
||||||
|
version "2.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/pathe/-/pathe-2.0.3.tgz#3ecbec55421685b70a9da872b2cff3e1cbed1716"
|
||||||
|
integrity sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==
|
||||||
|
|
||||||
pend@~1.2.0:
|
pend@~1.2.0:
|
||||||
version "1.2.0"
|
version "1.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50"
|
resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50"
|
||||||
@@ -6016,6 +6101,15 @@ pkg-types@^1.1.1:
|
|||||||
mlly "^1.7.1"
|
mlly "^1.7.1"
|
||||||
pathe "^1.1.2"
|
pathe "^1.1.2"
|
||||||
|
|
||||||
|
pkg-types@^1.2.0, pkg-types@^1.3.0:
|
||||||
|
version "1.3.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/pkg-types/-/pkg-types-1.3.1.tgz#bd7cc70881192777eef5326c19deb46e890917df"
|
||||||
|
integrity sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ==
|
||||||
|
dependencies:
|
||||||
|
confbox "^0.1.8"
|
||||||
|
mlly "^1.7.4"
|
||||||
|
pathe "^2.0.1"
|
||||||
|
|
||||||
postcss-import@^15.1.0:
|
postcss-import@^15.1.0:
|
||||||
version "15.1.0"
|
version "15.1.0"
|
||||||
resolved "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz"
|
resolved "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz"
|
||||||
@@ -6347,6 +6441,11 @@ readable-stream@^3.6.0:
|
|||||||
string_decoder "^1.1.1"
|
string_decoder "^1.1.1"
|
||||||
util-deprecate "^1.0.1"
|
util-deprecate "^1.0.1"
|
||||||
|
|
||||||
|
readdirp@^4.0.1:
|
||||||
|
version "4.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-4.1.2.tgz#eb85801435fbf2a7ee58f19e0921b068fc69948d"
|
||||||
|
integrity sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg==
|
||||||
|
|
||||||
readdirp@~3.3.0:
|
readdirp@~3.3.0:
|
||||||
version "3.3.0"
|
version "3.3.0"
|
||||||
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.3.0.tgz#984458d13a1e42e2e9f5841b129e162f369aff17"
|
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.3.0.tgz#984458d13a1e42e2e9f5841b129e162f369aff17"
|
||||||
@@ -6508,6 +6607,11 @@ rollup@2.78.0:
|
|||||||
version "5.4.0"
|
version "5.4.0"
|
||||||
resolved "https://codeload.github.com/whereby/rtcstats/tar.gz/6f6623b4b53e9f6b41f530339793de227f34ea51"
|
resolved "https://codeload.github.com/whereby/rtcstats/tar.gz/6f6623b4b53e9f6b41f530339793de227f34ea51"
|
||||||
|
|
||||||
|
run-applescript@^7.0.0:
|
||||||
|
version "7.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/run-applescript/-/run-applescript-7.0.0.tgz#e5a553c2bffd620e169d276c1cd8f1b64778fbeb"
|
||||||
|
integrity sha512-9by4Ij99JUr/MCFBUkDKLWK3G9HVXmabKz9U5MlIAIuvuzkiOicRYs8XJLxX+xahD+mLiiCYDqF9dKAgtzKP1A==
|
||||||
|
|
||||||
run-parallel@^1.1.9:
|
run-parallel@^1.1.9:
|
||||||
version "1.2.0"
|
version "1.2.0"
|
||||||
resolved "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz"
|
resolved "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz"
|
||||||
@@ -6587,6 +6691,11 @@ semver@7.3.5:
|
|||||||
dependencies:
|
dependencies:
|
||||||
lru-cache "^6.0.0"
|
lru-cache "^6.0.0"
|
||||||
|
|
||||||
|
semver@7.7.2:
|
||||||
|
version "7.7.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/semver/-/semver-7.7.2.tgz#67d99fdcd35cec21e6f8b87a7fd515a33f982b58"
|
||||||
|
integrity sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==
|
||||||
|
|
||||||
semver@^7.3.5:
|
semver@^7.3.5:
|
||||||
version "7.6.3"
|
version "7.6.3"
|
||||||
resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143"
|
resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143"
|
||||||
@@ -7244,6 +7353,11 @@ ufo@^1.5.3:
|
|||||||
resolved "https://registry.yarnpkg.com/ufo/-/ufo-1.5.3.tgz#3325bd3c977b6c6cd3160bf4ff52989adc9d3344"
|
resolved "https://registry.yarnpkg.com/ufo/-/ufo-1.5.3.tgz#3325bd3c977b6c6cd3160bf4ff52989adc9d3344"
|
||||||
integrity sha512-Y7HYmWaFwPUmkoQCUIAYpKqkOf+SbVj/2fJJZ4RJMCfZp0rTGwRbzQD+HghfnhKOjL9E01okqz+ncJskGYfBNw==
|
integrity sha512-Y7HYmWaFwPUmkoQCUIAYpKqkOf+SbVj/2fJJZ4RJMCfZp0rTGwRbzQD+HghfnhKOjL9E01okqz+ncJskGYfBNw==
|
||||||
|
|
||||||
|
ufo@^1.5.4:
|
||||||
|
version "1.6.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/ufo/-/ufo-1.6.1.tgz#ac2db1d54614d1b22c1d603e3aef44a85d8f146b"
|
||||||
|
integrity sha512-9a4/uxlTWJ4+a5i0ooc1rU7C7YOw3wT+UGqdeNNHWnOF9qcMBgLRS+4IYUqbczewFx4mLEig6gawh7X6mFlEkA==
|
||||||
|
|
||||||
uglify-js@^3.1.4:
|
uglify-js@^3.1.4:
|
||||||
version "3.17.4"
|
version "3.17.4"
|
||||||
resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-3.17.4.tgz#61678cf5fa3f5b7eb789bb345df29afb8257c22c"
|
resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-3.17.4.tgz#61678cf5fa3f5b7eb789bb345df29afb8257c22c"
|
||||||
|
|||||||
Reference in New Issue
Block a user