Skip to content

web

Module which contains the web server related function FastAPI routes/classes etc.

ApplicationState ¤

Class that holds the application state.

Client(client_id) ¤

Holds the client's state.

  • websocket: The WebSocket instance.
  • client_id: The client's ID.
  • downloader_settings: The downloader settings.
Source code in spotdl/utils/web.py
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
def __init__(
    self,
    # websocket: WebSocket,
    client_id: str,
    # update_callback: Optional[Callable] = None,
):
    """
    Initialize the WebSocket handler.
    ### Arguments
    - websocket: The WebSocket instance.
    - client_id: The client's ID.
    - downloader_settings: The downloader settings.
    """

    self.downloader_settings = DownloaderOptions(
        **create_settings_type(
            Namespace(config=False),
            dict(app_state.downloader_settings),
            DOWNLOADER_OPTIONS,
        )  # type: ignore
    )

    # self.websocket = websocket
    self.client_id = client_id
    # self.update_callback = update_callback
    self.downloader = Downloader(
        settings=self.downloader_settings, loop=app_state.loop
    )

    self.downloader.progress_handler = ProgressHandler(
        simple_tui=True,
        update_callback=self.song_update,
        web_ui=True,
    )

    self.disconnect_timer = None

connect() async ¤

Called when a new client connects to the websocket.

Source code in spotdl/utils/web.py
121
122
123
124
125
126
127
128
129
130
131
132
133
async def connect(self):
    """
    Called when a new client connects to the websocket.
    """

    # await self.websocket.accept()

    # Add the connection to the list of connections
    if self.disconnect_timer and self.disconnect_timer.is_alive():
        self.disconnect_timer.cancel()
        self.disconnect_timer = None
    app_state.clients[self.client_id] = self
    app_state.logger.info("Client %s connected", self.client_id)

disconnect() async ¤

Called when a client disconnects from the websocket.

Source code in spotdl/utils/web.py
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
async def disconnect(self):
    """
    Called when a client disconnects from the websocket.
    """

    # If the disconnect timer is running, cancel it
    if self.disconnect_timer and self.disconnect_timer.is_alive():
        self.disconnect_timer.cancel()

    # Schedule the disconnect now
    app_state.logger.info(
        "Client %s will disconnect in 15 seconds of inactivity", self.client_id
    )
    self.disconnect_timer = threading.Timer(15, self.disconnect_now)
    self.disconnect_timer.start()

disconnect_now() ¤

Disconnect the client.

Source code in spotdl/utils/web.py
151
152
153
154
155
156
157
158
159
160
161
162
163
164
def disconnect_now(self):
    """
    Disconnect the client.
    """
    # Remove the connection from the list of connections
    if self.client_id in app_state.clients:
        # app_state.clients.pop(client_id, None)
        del app_state.clients[self.client_id]
        app_state.logger.info("Client %s disconnected", self.client_id)
    else:
        app_state.logger.warning(
            "Client %s not found on disconnect", self.client_id
        )
    self.disconnect_timer = None

get_instance(client_id) classmethod ¤

Get the WebSocket instance for a client.

Arguments¤
  • client_id: The client's ID.
Returns¤
  • returns the WebSocket instance.
Source code in spotdl/utils/web.py
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
@classmethod
def get_instance(cls, client_id: str) -> Optional["Client"]:
    """
    Get the WebSocket instance for a client.

    ### Arguments
    - client_id: The client's ID.

    ### Returns
    - returns the WebSocket instance.
    """

    instance = app_state.clients.get(client_id)
    if instance:
        return instance

    app_state.logger.error("Client %s not found", client_id)

    return None

send_update(update) async ¤

Send an update to the client.

Arguments¤
  • update: The update to send.
Source code in spotdl/utils/web.py
166
167
168
169
170
171
172
async def send_update(self, update: Dict[str, Any]):
    """
    Send an update to the client.

    ### Arguments
    - update: The update to send.
    """

song_update(progress_handler, message) ¤

Called when a song updates.

Arguments¤
  • progress_handler: The progress handler.
  • message: The message to send.
Source code in spotdl/utils/web.py
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
def song_update(self, progress_handler: SongTracker, message: str):
    """
    Called when a song updates.

    ### Arguments
    - progress_handler: The progress handler.
    - message: The message to send.
    """

    update_message = {
        "song": progress_handler.song.json,
        "progress": progress_handler.progress,
        "message": message,
    }

    asyncio.run_coroutine_threadsafe(
        self.send_update(update_message), app_state.loop
    )

SPAStaticFiles ¤

Bases: StaticFiles

Override the static files to serve the index.html and other assets.

get_response(path, scope) async ¤

Serve static files from the SPA.

Arguments¤
  • path: The path to the file.
  • scope: The scope of the request.
Returns¤
  • returns the response.
Source code in spotdl/utils/web.py
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
async def get_response(self, path: str, scope: Scope) -> Response:
    """
    Serve static files from the SPA.

    ### Arguments
    - path: The path to the file.
    - scope: The scope of the request.

    ### Returns
    - returns the response.
    """

    response = await super().get_response(path, scope)
    if response.status_code == 404:
        response = await super().get_response(".", scope)

    response.headers.setdefault(
        "Cache-Control", "max-age=0, no-cache, no-store, , must-revalidate"
    )
    response.headers.setdefault("Pragma", "no-cache")
    response.headers.setdefault("Expires", "0")

    return response

fix_mime_types() ¤

Fix incorrect entries in the mimetypes registry. On Windows, the Python standard library's mimetypes reads in mappings from file extension to MIME type from the Windows registry. Other applications can and do write incorrect values to this registry, which causes mimetypes.guess_type to return incorrect values, which causes spotDL to fail to render on the frontend. This method hard-codes the correct mappings for certain MIME types that are known to be either used by TensorBoard or problematic in general.

Source code in spotdl/utils/web.py
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
def fix_mime_types():
    """Fix incorrect entries in the `mimetypes` registry.
    On Windows, the Python standard library's `mimetypes` reads in
    mappings from file extension to MIME type from the Windows
    registry. Other applications can and do write incorrect values
    to this registry, which causes `mimetypes.guess_type` to return
    incorrect values, which causes spotDL to fail to render on
    the frontend.
    This method hard-codes the correct mappings for certain MIME
    types that are known to be either used by TensorBoard or
    problematic in general.
    """

    # Known to be problematic when Visual Studio is installed:
    # https://github.com/tensorflow/tensorboard/issues/3120
    # https://github.com/spotDL/spotify-downloader/issues/1540
    mimetypes.add_type("application/javascript", ".js")

    # Not known to be problematic, but used by spotDL:
    mimetypes.add_type("text/css", ".css")
    mimetypes.add_type("image/svg+xml", ".svg")
    mimetypes.add_type("text/html", ".html")

get_client(client_id=Query(default=None)) ¤

Get the client's state.

Arguments¤
  • client_id: The client's ID.
Returns¤
  • returns the client's state.
Source code in spotdl/utils/web.py
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
def get_client(client_id: Union[str, None] = Query(default=None)) -> Client:
    """
    Get the client's state.

    ### Arguments
    - client_id: The client's ID.

    ### Returns
    - returns the client's state.
    """

    if client_id is None:
        raise HTTPException(status_code=400, detail="client_id is required")

    instance = Client.get_instance(client_id)
    if instance is None:
        raise HTTPException(status_code=404, detail="client not found")

    return instance

get_current_state() ¤

Get the current state of the application.

Returns¤
  • returns the application state.
Source code in spotdl/utils/web.py
237
238
239
240
241
242
243
244
245
def get_current_state() -> ApplicationState:
    """
    Get the current state of the application.

    ### Returns
    - returns the application state.
    """

    return app_state

validate_search_term(search_term) ¤

Validate the search term to check if it is a valid Spotify URL.

Arguments¤
  • search_term: The search term to validate.
Returns¤
  • True if the search term is valid, False otherwise.
Source code in spotdl/utils/web.py
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
def validate_search_term(search_term: str) -> bool:
    """
    Validate the search term to check if it is a valid Spotify URL.

    ### Arguments
    - search_term: The search term to validate.

    ### Returns
    - True if the search term is valid, False otherwise.
    """
    return search_term != "" and (
        "://open.spotify.com/track/" in search_term
        or "://open.spotify.com/album/" in search_term
        or "://open.spotify.com/playlist/" in search_term
        or "://open.spotify.com/artist/" in search_term
    )