Compare commits
21 Commits
d3b599738e
...
7-create-p
Author | SHA1 | Date | |
---|---|---|---|
7f616b4c7d | |||
2e7d2004af
|
|||
b2e6d20d27 | |||
64d7538ff4
|
|||
77e483d637
|
|||
b0465a020d
|
|||
a0757ea3ff
|
|||
a8b6a309f0
|
|||
317f4d7fba
|
|||
b7a930c69a
|
|||
3c6e742e43
|
|||
7296582b0d
|
|||
66f09cf5a3
|
|||
d39ccba927
|
|||
498b1d82d9
|
|||
79405cd08c
|
|||
39c9689af4
|
|||
ad7c8af9de
|
|||
f1cb0b2159
|
|||
049cd86ae0
|
|||
aab4433a55
|
@ -1,7 +1,11 @@
|
|||||||
HASHTAG_FILTER = ichlausche,music,musik,nowplaying,tunetuesday,nowlistening
|
HASHTAG_FILTER = ichlausche,music,musik,nowplaying,tunetuesday,nowlistening
|
||||||
YOUTUBE_API_KEY = CHANGE_ME
|
YOUTUBE_API_KEY = CHANGE_ME
|
||||||
|
YOUTUBE_PLAYLIST_ID = CHANGE_ME
|
||||||
|
YOUTUBE_CLIENT_ID = CHANGE_ME
|
||||||
|
YOUTUBE_CLIENT_SECRET = CHANGE_ME
|
||||||
ODESLI_API_KEY = CHANGE_ME
|
ODESLI_API_KEY = CHANGE_ME
|
||||||
MASTODON_INSTANCE = 'metalhead.club'
|
MASTODON_INSTANCE = 'metalhead.club'
|
||||||
|
MASTODON_ACCESS_TOKEN = 'YOUR_ACCESS_TOKEN_HERE'
|
||||||
BASE_URL = 'https://moshingmammut.phlaym.net'
|
BASE_URL = 'https://moshingmammut.phlaym.net'
|
||||||
VERBOSE = false
|
VERBOSE = false
|
||||||
IGNORE_USERS = @moshhead@metalhead.club
|
IGNORE_USERS = @moshhead@metalhead.club
|
||||||
@ -9,3 +13,4 @@ WEBSUB_HUB = 'http://pubsubhubbub.superfeedr.com'
|
|||||||
|
|
||||||
PUBLIC_REFRESH_INTERVAL = 10000
|
PUBLIC_REFRESH_INTERVAL = 10000
|
||||||
PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME = 'Metalhead.club'
|
PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME = 'Metalhead.club'
|
||||||
|
PORT = 3001
|
@ -1,16 +1,25 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
root: true,
|
root: true,
|
||||||
parser: '@typescript-eslint/parser',
|
parser: '@typescript-eslint/parser',
|
||||||
extends: ['eslint:recommended', 'plugin:@typescript-eslint/recommended', 'prettier'],
|
extends: ['plugin:svelte/recommended', 'plugin:@typescript-eslint/recommended', 'prettier'],
|
||||||
plugins: ['svelte3', '@typescript-eslint'],
|
plugins: ['@typescript-eslint'],
|
||||||
ignorePatterns: ['*.cjs'],
|
ignorePatterns: ['*.cjs'],
|
||||||
overrides: [{ files: ['*.svelte'], processor: 'svelte3/svelte3' }],
|
overrides: [
|
||||||
|
{
|
||||||
|
files: ['*.svelte'],
|
||||||
|
parser: 'svelte-eslint-parser',
|
||||||
|
parserOptions: {
|
||||||
|
parser: '@typescript-eslint/parser'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
settings: {
|
settings: {
|
||||||
'svelte3/typescript': () => require('typescript')
|
'svelte3/typescript': () => require('typescript')
|
||||||
},
|
},
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
sourceType: 'module',
|
sourceType: 'module',
|
||||||
ecmaVersion: 2020
|
ecmaVersion: 2020,
|
||||||
|
extraFileExtensions: ['.svelte']
|
||||||
},
|
},
|
||||||
env: {
|
env: {
|
||||||
browser: true,
|
browser: true,
|
||||||
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -1,3 +1,5 @@
|
|||||||
|
yt_auth_token
|
||||||
|
spotify_auth_token
|
||||||
*.db
|
*.db
|
||||||
feed.xml
|
feed.xml
|
||||||
playbook.yml
|
playbook.yml
|
||||||
|
19
README.md
19
README.md
@ -65,7 +65,7 @@ Set up NVM:
|
|||||||
|
|
||||||
```
|
```
|
||||||
$ cd
|
$ cd
|
||||||
$ curl https://raw.github.com/creationix/nvm/master/install.sh | sh
|
$ curl https://raw.githubusercontent.com/nvm-sh/nvm/refs/heads/master/install.sh | bash
|
||||||
$ source ~/.nvm/nvm.sh
|
$ source ~/.nvm/nvm.sh
|
||||||
$ nvm install --lts
|
$ nvm install --lts
|
||||||
```
|
```
|
||||||
@ -98,11 +98,23 @@ and set your `User`, `Group`, `ExecStart` and `WorkingDirectory` accordingly.
|
|||||||
Copy `.env.EXAMPLE` to `.env` and add your `YOUTUBE_API_KEY` and `ODESLI_API_KEY`.
|
Copy `.env.EXAMPLE` to `.env` and add your `YOUTUBE_API_KEY` and `ODESLI_API_KEY`.
|
||||||
To obtain one follow [YouTube's guide](https://developers.google.com/youtube/registering_an_application) to create an
|
To obtain one follow [YouTube's guide](https://developers.google.com/youtube/registering_an_application) to create an
|
||||||
_API key_.
|
_API key_.
|
||||||
If `YOUTUBE_API_KEY` is unset, no playlist will be updated. Also, _all_ YouTube links will be treated as music videos,
|
If `YOUTUBE_API_KEY` is unset _all_ YouTube links will be treated as music videos,
|
||||||
because the API is the only way to check if a YouTube link leads to music or something else.
|
because the API is the only way to check if a YouTube link leads to music or something else.
|
||||||
|
|
||||||
If `ODESLI_API_KEY` is unset, your rate limit to the song.link API will be lower.
|
If `ODESLI_API_KEY` is unset, your rate limit to the song.link API will be lower.
|
||||||
|
|
||||||
|
Add `MASTODON_ACCESS_TOKEN` as well, see [Creating our application
|
||||||
|
|
||||||
|
](https://docs.joinmastodon.org/client/token/#app) in the Mastodon documentation.
|
||||||
|
`read:statuses` and `read:search` the only required scope. An access token will be displayed in your settings. Use that!
|
||||||
|
|
||||||
|
There are currently no plans to implement an actual authentication flow.
|
||||||
|
|
||||||
|
If you want the app to save the songs it encounters into a playlist, YouTube requires OAuth 2.0 credentials.
|
||||||
|
Once again, follow [YouTube's guide](https://developers.google.com/youtube/registering_an_application) and the OAuth 2.0 described there
|
||||||
|
to obtain a clientId and clientSecret. Add the values as `YOUTUBE_CLIENT_ID` and `YOUTUBE_CLIENT_SECRET`.
|
||||||
|
Create a playlist and configure its ID as `YOUTUBE_PLAYLIST_ID`.
|
||||||
|
|
||||||
Run `npm run build` and copy the output folder, usually `build` to `$APP_DIR` on your server.
|
Run `npm run build` and copy the output folder, usually `build` to `$APP_DIR` on your server.
|
||||||
|
|
||||||
#### On your server again
|
#### On your server again
|
||||||
@ -118,6 +130,9 @@ Verify that everything is okay with `service moshing-mammut status`.
|
|||||||
|
|
||||||
The app should now be reachable on http://localhost:3000 or whatever you configured your domain to be!
|
The app should now be reachable on http://localhost:3000 or whatever you configured your domain to be!
|
||||||
|
|
||||||
|
If you want to add the songs available on YouTube to a playlist and have configured the environment variables to do so,
|
||||||
|
you now need to visit `/ytauth`, e.g. `http://localhost:3000/ytauth`. This will obtain the necessary access tokens from Google.
|
||||||
|
|
||||||
# Icons
|
# Icons
|
||||||
|
|
||||||
Favicon is a combination of [speaker-line by remix icon](https://remixicon.com/icon/speaker-line)
|
Favicon is a combination of [speaker-line by remix icon](https://remixicon.com/icon/speaker-line)
|
||||||
|
5543
package-lock.json
generated
5543
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
49
package.json
49
package.json
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "moshing-mammut",
|
"name": "moshing-mammut",
|
||||||
"version": "1.3.1",
|
"version": "2.0.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "LGPL-3.0-or-later",
|
"license": "LGPL-3.0-or-later",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -14,34 +14,35 @@
|
|||||||
"format": "prettier --plugin-search-dir . --write ."
|
"format": "prettier --plugin-search-dir . --write ."
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@sveltejs/adapter-node": "^1.2.3",
|
"@sveltejs/adapter-node": "^5.2.12",
|
||||||
"@sveltejs/kit": "^1.5.0",
|
"@sveltejs/kit": "^2.22.2",
|
||||||
"@types/node": "^18.16.3",
|
"@sveltejs/vite-plugin-svelte": "^5.1.0",
|
||||||
"@types/sqlite3": "^3.1.8",
|
"@types/node": "^22.9.0",
|
||||||
"@types/ws": "^8.5.4",
|
"@types/sqlite3": "^3.0.0",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.45.0",
|
"@types/ws": "^8.5.0",
|
||||||
"@typescript-eslint/parser": "^5.45.0",
|
"@typescript-eslint/eslint-plugin": "^8.0.0",
|
||||||
|
"@typescript-eslint/parser": "^8.0.0",
|
||||||
"@zerodevx/svelte-toast": "^0.9.3",
|
"@zerodevx/svelte-toast": "^0.9.3",
|
||||||
"eslint": "^8.28.0",
|
"eslint": "^9.11.1",
|
||||||
"eslint-config-prettier": "^8.5.0",
|
"eslint-config-prettier": "^10.0.0",
|
||||||
"eslint-plugin-svelte3": "^4.0.0",
|
"eslint-plugin-svelte": "^3.0.0",
|
||||||
"prettier": "^2.8.0",
|
"prettier": "^3.1.0",
|
||||||
"prettier-plugin-svelte": "^2.8.1",
|
"prettier-plugin-svelte": "^3.2.6",
|
||||||
"svelte": "^3.54.0",
|
"svelte": "^5",
|
||||||
"svelte-check": "^3.0.1",
|
"svelte-check": "^4.0.0",
|
||||||
"tslib": "^2.4.1",
|
"tslib": "^2.0.0",
|
||||||
"typescript": "^4.9.3",
|
"typescript": "^5.0.0",
|
||||||
"vite": "^4.0.0"
|
"vite": "^6.0.0"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"dotenv": "^16.0.3",
|
"dotenv": "^17.0.0",
|
||||||
"feed": "^4.2.2",
|
"feed": "^5.1.0",
|
||||||
"sharp": "^0.32.0",
|
"sharp": "^0.34.2",
|
||||||
"sqlite3": "^5.1.6",
|
"sqlite3": "^5.0.0",
|
||||||
"ws": "^8.13.0"
|
"ws": "^8.18.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18.0.0"
|
"node": ">=20.0.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
21
src/app.html
21
src/app.html
@ -1,4 +1,4 @@
|
|||||||
<!DOCTYPE html>
|
<!doctype html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
@ -54,9 +54,22 @@
|
|||||||
|
|
||||||
color: var(--color-text);
|
color: var(--color-text);
|
||||||
background-color: var(--color-bg);
|
background-color: var(--color-bg);
|
||||||
font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen-Sans,
|
font-family:
|
||||||
Ubuntu, Cantarell, 'Helvetica Neue', Helvetica, Arial, sans-serif, 'Apple Color Emoji',
|
system-ui,
|
||||||
'Segoe UI Emoji', 'Segoe UI Symbol';
|
-apple-system,
|
||||||
|
BlinkMacSystemFont,
|
||||||
|
'Segoe UI',
|
||||||
|
Roboto,
|
||||||
|
Oxygen-Sans,
|
||||||
|
Ubuntu,
|
||||||
|
Cantarell,
|
||||||
|
'Helvetica Neue',
|
||||||
|
Helvetica,
|
||||||
|
Arial,
|
||||||
|
sans-serif,
|
||||||
|
'Apple Color Emoji',
|
||||||
|
'Segoe UI Emoji',
|
||||||
|
'Segoe UI Symbol';
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
|
@ -1,14 +1,17 @@
|
|||||||
import { log } from '$lib/log';
|
import { Logger } from '$lib/log';
|
||||||
import { TimelineReader } from '$lib/server/timeline';
|
import { TimelineReader } from '$lib/server/timeline';
|
||||||
import type { HandleServerError } from '@sveltejs/kit';
|
import type { Handle, HandleServerError } from '@sveltejs/kit';
|
||||||
import { error } from '@sveltejs/kit';
|
import { error } from '@sveltejs/kit';
|
||||||
import fs from 'fs/promises';
|
import fs from 'fs/promises';
|
||||||
|
|
||||||
|
const logger = new Logger('App');
|
||||||
|
|
||||||
|
logger.log('App startup');
|
||||||
TimelineReader.init();
|
TimelineReader.init();
|
||||||
|
|
||||||
export const handleError = (({ error }) => {
|
export const handleError = (({ error }) => {
|
||||||
if (error instanceof Error) {
|
if (error instanceof Error) {
|
||||||
log.error('Something went wrong: ', error.name, error.message);
|
logger.error('Something went wrong: ', error.name, error.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@ -16,9 +19,13 @@ export const handleError = (({ error }) => {
|
|||||||
};
|
};
|
||||||
}) satisfies HandleServerError;
|
}) satisfies HandleServerError;
|
||||||
|
|
||||||
import type { Handle } from '@sveltejs/kit';
|
|
||||||
|
|
||||||
export const handle = (async ({ event, resolve }) => {
|
export const handle = (async ({ event, resolve }) => {
|
||||||
|
const searchParams = event.url.searchParams;
|
||||||
|
const authCode = searchParams.get('code');
|
||||||
|
if (authCode) {
|
||||||
|
logger.debug('received GET hook', event.url.searchParams);
|
||||||
|
}
|
||||||
|
|
||||||
// Reeder *insists* on checking /feed instead of /feed.xml
|
// Reeder *insists* on checking /feed instead of /feed.xml
|
||||||
if (event.url.pathname === '/feed') {
|
if (event.url.pathname === '/feed') {
|
||||||
return new Response('', { status: 301, headers: { Location: '/feed.xml' } });
|
return new Response('', { status: 301, headers: { Location: '/feed.xml' } });
|
||||||
@ -40,7 +47,7 @@ export const handle = (async ({ event, resolve }) => {
|
|||||||
const readStream = fd
|
const readStream = fd
|
||||||
.readableWebStream()
|
.readableWebStream()
|
||||||
.getReader({ mode: 'byob' }) as ReadableStream<Uint8Array>;
|
.getReader({ mode: 'byob' }) as ReadableStream<Uint8Array>;
|
||||||
log.info('sending. size: ', stat.size);
|
logger.info('sending. size: ', stat.size);
|
||||||
return new Response(readStream, {
|
return new Response(readStream, {
|
||||||
headers: [
|
headers: [
|
||||||
['Content-Type', 'image/' + suffix],
|
['Content-Type', 'image/' + suffix],
|
||||||
@ -52,8 +59,8 @@ export const handle = (async ({ event, resolve }) => {
|
|||||||
const f = await fs.readFile('avatars/' + fileName);
|
const f = await fs.readFile('avatars/' + fileName);
|
||||||
return new Response(f, { headers: [['Content-Type', 'image/' + suffix]] });
|
return new Response(f, { headers: [['Content-Type', 'image/' + suffix]] });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log.error('no stream', e);
|
logger.error('no stream', e);
|
||||||
throw error(404);
|
error(404);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,11 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { Account } from '$lib/mastodon/response';
|
import type { Account } from '$lib/mastodon/response';
|
||||||
|
|
||||||
export let account: Account;
|
interface Props {
|
||||||
|
account: Account;
|
||||||
|
}
|
||||||
|
|
||||||
|
let { account }: Props = $props();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<a href={account.url} target="_blank">{account.display_name} @{account.acct}</a>
|
<a href={account.url} target="_blank">{account.display_name} @{account.acct}</a>
|
||||||
|
@ -1,11 +1,13 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { Account } from '$lib/mastodon/response';
|
import type { Account } from '$lib/mastodon/response';
|
||||||
|
|
||||||
export let account: Account;
|
interface Props {
|
||||||
let avatarDescription: string;
|
account: Account;
|
||||||
let sourceSetHtml: string;
|
}
|
||||||
$: avatarDescription = `Avatar for ${account.acct}`;
|
|
||||||
$: {
|
let { account }: Props = $props();
|
||||||
|
let avatarDescription: string = $derived(`Avatar for ${account.acct}`);
|
||||||
|
let sourceSetHtml: string = $derived.by(() => {
|
||||||
// Sort thumbnails by file type. This is important, because the order of the srcset entries matter.
|
// Sort thumbnails by file type. This is important, because the order of the srcset entries matter.
|
||||||
// We need the best format to be first
|
// We need the best format to be first
|
||||||
const formatPriority = new Map<string, number>([
|
const formatPriority = new Map<string, number>([
|
||||||
@ -14,7 +16,7 @@
|
|||||||
['jpg', 99],
|
['jpg', 99],
|
||||||
['jpeg', 99]
|
['jpeg', 99]
|
||||||
]);
|
]);
|
||||||
const resizedAvatars = (account.resizedAvatars ?? []).sort((a, b) => {
|
const resizedAvatars = (account.resizedAvatars ?? []).toSorted((a, b) => {
|
||||||
const extensionA = a.file.split('.').pop() ?? '';
|
const extensionA = a.file.split('.').pop() ?? '';
|
||||||
const extensionB = b.file.split('.').pop() ?? '';
|
const extensionB = b.file.split('.').pop() ?? '';
|
||||||
const prioA = formatPriority.get(extensionA) ?? 3;
|
const prioA = formatPriority.get(extensionA) ?? 3;
|
||||||
@ -33,8 +35,8 @@
|
|||||||
const srcset = entry[1].join(', ');
|
const srcset = entry[1].join(', ');
|
||||||
html += `<source srcset="${srcset}" type="${entry[0]}" />`;
|
html += `<source srcset="${srcset}" type="${entry[0]}" />`;
|
||||||
}
|
}
|
||||||
sourceSetHtml = html;
|
return html;
|
||||||
}
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<picture>
|
<picture>
|
||||||
|
@ -1,31 +1,34 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { createEventDispatcher } from 'svelte';
|
|
||||||
import LoadingSpinnerComponent from '$lib/components/LoadingSpinnerComponent.svelte';
|
import LoadingSpinnerComponent from '$lib/components/LoadingSpinnerComponent.svelte';
|
||||||
|
|
||||||
export let moreAvailable = false;
|
interface Props {
|
||||||
export let isLoading = false;
|
moreAvailable?: boolean;
|
||||||
let displayText = '';
|
isLoading?: boolean;
|
||||||
let title = '';
|
loadOlderPosts: any;
|
||||||
let disabled: boolean;
|
|
||||||
|
|
||||||
$: if (isLoading) {
|
|
||||||
displayText = 'Loading...';
|
|
||||||
} else if (!moreAvailable) {
|
|
||||||
displayText = 'You reached the end';
|
|
||||||
} else {
|
|
||||||
displayText = 'Load More';
|
|
||||||
}
|
}
|
||||||
$: disabled = !moreAvailable || isLoading;
|
|
||||||
$: title = moreAvailable ? 'Load More' : 'There be dragons!';
|
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
let { moreAvailable = false, isLoading = false, loadOlderPosts }: Props = $props();
|
||||||
|
let displayText = $derived.by(() => {
|
||||||
|
if (isLoading) {
|
||||||
|
return 'Loading...';
|
||||||
|
} else if (!moreAvailable) {
|
||||||
|
return 'You reached the end';
|
||||||
|
}
|
||||||
|
return 'Load More';
|
||||||
|
});
|
||||||
|
let title = $derived(moreAvailable ? 'Load More' : 'There be dragons!');
|
||||||
|
let disabled: boolean = $derived(!moreAvailable || isLoading);
|
||||||
|
|
||||||
|
/*const dispatch = createEventDispatcher<{
|
||||||
|
loadOlderPosts: string;
|
||||||
|
}>();
|
||||||
|
|
||||||
function loadOlderPosts() {
|
function loadOlderPosts() {
|
||||||
dispatch('loadOlderPosts');
|
dispatch('loadOlderPosts');
|
||||||
}
|
}*/
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<button on:click={loadOlderPosts} {disabled} {title}>
|
<button onclick={() => loadOlderPosts()} {disabled} {title}>
|
||||||
<div class="loading" class:collapsed={!isLoading}>
|
<div class="loading" class:collapsed={!isLoading}>
|
||||||
<LoadingSpinnerComponent size="0.5em" thickness="6px" />
|
<LoadingSpinnerComponent size="0.5em" thickness="6px" />
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,9 +1,13 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
export let size = '64px';
|
interface Props {
|
||||||
export let thickness = '6px';
|
size?: string;
|
||||||
|
thickness?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
let { size = '64px', thickness = '6px' }: Props = $props();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="lds-dual-ring" style="--size: {size}; --thickness: {thickness}" />
|
<div class="lds-dual-ring" style="--size: {size}; --thickness: {thickness}"></div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.lds-dual-ring {
|
.lds-dual-ring {
|
||||||
|
@ -6,15 +6,21 @@
|
|||||||
import { secondsSince, relativeTime } from '$lib/relativeTime';
|
import { secondsSince, relativeTime } from '$lib/relativeTime';
|
||||||
import { onMount } from 'svelte';
|
import { onMount } from 'svelte';
|
||||||
|
|
||||||
export let post: Post;
|
interface Props {
|
||||||
let displayRelativeTime = false;
|
post: Post;
|
||||||
const absoluteDate = new Date(post.created_at).toLocaleString();
|
|
||||||
let dateCreated = absoluteDate;
|
|
||||||
const timePassed = secondsSince(new Date(post.created_at));
|
|
||||||
$: if (displayRelativeTime) {
|
|
||||||
dateCreated = relativeTime($timePassed) ?? absoluteDate;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let { post }: Props = $props();
|
||||||
|
let displayRelativeTime = $state(false);
|
||||||
|
const absoluteDate = new Date(post.created_at).toLocaleString();
|
||||||
|
const timePassed = secondsSince(new Date(post.created_at));
|
||||||
|
let dateCreated = $derived.by(() => {
|
||||||
|
if (displayRelativeTime) {
|
||||||
|
return relativeTime($timePassed) ?? absoluteDate;
|
||||||
|
}
|
||||||
|
return absoluteDate;
|
||||||
|
});
|
||||||
|
|
||||||
const songs = filterDuplicates(post.songs ?? []);
|
const songs = filterDuplicates(post.songs ?? []);
|
||||||
|
|
||||||
function filterDuplicates(songs: SongInfo[]): SongInfo[] {
|
function filterDuplicates(songs: SongInfo[]): SongInfo[] {
|
||||||
@ -59,7 +65,7 @@
|
|||||||
['jpg', 99],
|
['jpg', 99],
|
||||||
['jpeg', 99]
|
['jpeg', 99]
|
||||||
]);
|
]);
|
||||||
const thumbs = (song.resizedThumbnails ?? []).sort((a, b) => {
|
const thumbs = (song.resizedThumbnails ?? []).toSorted((a, b) => {
|
||||||
const extensionA = a.file.split('.').pop() ?? '';
|
const extensionA = a.file.split('.').pop() ?? '';
|
||||||
const extensionB = b.file.split('.').pop() ?? '';
|
const extensionB = b.file.split('.').pop() ?? '';
|
||||||
const prioA = formatPriority.get(extensionA) ?? 3;
|
const prioA = formatPriority.get(extensionA) ?? 3;
|
||||||
|
@ -4,6 +4,9 @@ const { DEV } = import.meta.env;
|
|||||||
|
|
||||||
export const enableVerboseLog = isTruthy(env.VERBOSE);
|
export const enableVerboseLog = isTruthy(env.VERBOSE);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use the new {@link Logger} class instead.
|
||||||
|
*/
|
||||||
export const log = {
|
export const log = {
|
||||||
verbose: (...params: any[]) => {
|
verbose: (...params: any[]) => {
|
||||||
if (!enableVerboseLog) {
|
if (!enableVerboseLog) {
|
||||||
@ -12,7 +15,7 @@ export const log = {
|
|||||||
console.debug(new Date().toISOString(), ...params);
|
console.debug(new Date().toISOString(), ...params);
|
||||||
},
|
},
|
||||||
debug: (...params: any[]) => {
|
debug: (...params: any[]) => {
|
||||||
if (!DEV) {
|
if (!log.isDebugEnabled()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
console.debug(new Date().toISOString(), ...params);
|
console.debug(new Date().toISOString(), ...params);
|
||||||
@ -28,5 +31,59 @@ export const log = {
|
|||||||
},
|
},
|
||||||
error: (...params: any[]) => {
|
error: (...params: any[]) => {
|
||||||
console.error(new Date().toISOString(), ...params);
|
console.error(new Date().toISOString(), ...params);
|
||||||
|
},
|
||||||
|
isDebugEnabled: (): boolean => {
|
||||||
|
return DEV;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export class Logger {
|
||||||
|
public constructor(private name: string) {}
|
||||||
|
|
||||||
|
public static isDebugEnabled(): boolean {
|
||||||
|
return DEV;
|
||||||
|
}
|
||||||
|
public verbose(...params: any[]) {
|
||||||
|
if (!enableVerboseLog) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.debug(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
public debug(...params: any[]) {
|
||||||
|
if (!Logger.isDebugEnabled()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.debug(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
public log(...params: any[]) {
|
||||||
|
console.log(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
public info(...params: any[]) {
|
||||||
|
console.info(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
public warn(...params: any[]) {
|
||||||
|
console.warn(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
public error(...params: any[]) {
|
||||||
|
console.error(new Date().toISOString(), `- ${this.name} -`, ...params);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static error(...params: any[]) {
|
||||||
|
console.error(new Date().toISOString(), ...params);
|
||||||
|
}
|
||||||
|
public static debug(...params: any[]) {
|
||||||
|
if (!Logger.isDebugEnabled()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.debug(new Date().toISOString(), ...params);
|
||||||
|
}
|
||||||
|
public static log(...params: any[]) {
|
||||||
|
console.log(new Date().toISOString(), ...params);
|
||||||
|
}
|
||||||
|
public static info(...params: any[]) {
|
||||||
|
console.info(new Date().toISOString(), ...params);
|
||||||
|
}
|
||||||
|
public static warn(...params: any[]) {
|
||||||
|
console.warn(new Date().toISOString(), ...params);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -16,6 +16,17 @@ export interface Post {
|
|||||||
songs?: SongInfo[];
|
songs?: SongInfo[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface OauthResponse {
|
||||||
|
access_token: string;
|
||||||
|
expires_in: number;
|
||||||
|
expires?: Date;
|
||||||
|
refresh_token?: string;
|
||||||
|
refresh_token_expires_in?: number;
|
||||||
|
scope: string;
|
||||||
|
token_type: string;
|
||||||
|
error?: any;
|
||||||
|
}
|
||||||
|
|
||||||
export interface PreviewCard {
|
export interface PreviewCard {
|
||||||
url: string;
|
url: string;
|
||||||
title: string;
|
title: string;
|
||||||
|
@ -3,6 +3,8 @@ import type { SongThumbnailImage } from '$lib/mastodon/response';
|
|||||||
export type SongInfo = {
|
export type SongInfo = {
|
||||||
pageUrl: string;
|
pageUrl: string;
|
||||||
youtubeUrl?: string;
|
youtubeUrl?: string;
|
||||||
|
spotifyUrl?: string;
|
||||||
|
spotifyUri?: string;
|
||||||
type: 'song' | 'album';
|
type: 'song' | 'album';
|
||||||
title?: string;
|
title?: string;
|
||||||
artistName?: string;
|
artistName?: string;
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
import { IGNORE_USERS, MASTODON_INSTANCE } from '$env/static/private';
|
import { IGNORE_USERS, MASTODON_INSTANCE } from '$env/static/private';
|
||||||
import { enableVerboseLog, log } from '$lib/log';
|
import { enableVerboseLog, Logger } from '$lib/log';
|
||||||
import type { Account, AccountAvatar, Post, SongThumbnailImage, Tag } from '$lib/mastodon/response';
|
import type { Account, AccountAvatar, Post, SongThumbnailImage, Tag } from '$lib/mastodon/response';
|
||||||
import type { SongInfo } from '$lib/odesliResponse';
|
import type { SongInfo } from '$lib/odesliResponse';
|
||||||
import { TimelineReader } from '$lib/server/timeline';
|
import { TimelineReader } from '$lib/server/timeline';
|
||||||
import sqlite3 from 'sqlite3';
|
import sqlite3 from 'sqlite3';
|
||||||
|
|
||||||
|
const logger = new Logger('Database');
|
||||||
|
|
||||||
type FilterParameter = {
|
type FilterParameter = {
|
||||||
$limit?: number | undefined | null;
|
$limit?: number | undefined | null;
|
||||||
$since?: string | undefined | null;
|
$since?: string | undefined | null;
|
||||||
@ -37,6 +39,8 @@ type SongRow = {
|
|||||||
overviewUrl?: string;
|
overviewUrl?: string;
|
||||||
type: 'album' | 'song';
|
type: 'album' | 'song';
|
||||||
youtubeUrl?: string;
|
youtubeUrl?: string;
|
||||||
|
spotifyUrl?: string;
|
||||||
|
spotifyUri?: string;
|
||||||
title?: string;
|
title?: string;
|
||||||
artistName?: string;
|
artistName?: string;
|
||||||
thumbnailUrl?: string;
|
thumbnailUrl?: string;
|
||||||
@ -81,15 +85,15 @@ let databaseReady = false;
|
|||||||
if (enableVerboseLog) {
|
if (enableVerboseLog) {
|
||||||
sqlite3.verbose();
|
sqlite3.verbose();
|
||||||
db.on('change', (t, d, table, rowid) => {
|
db.on('change', (t, d, table, rowid) => {
|
||||||
log.verbose('DB change event', t, d, table, rowid);
|
logger.verbose('DB change event', t, d, table, rowid);
|
||||||
});
|
});
|
||||||
|
|
||||||
db.on('trace', (sql) => {
|
db.on('trace', (sql) => {
|
||||||
log.verbose('Running', sql);
|
logger.verbose('Running', sql);
|
||||||
});
|
});
|
||||||
|
|
||||||
db.on('profile', (sql) => {
|
db.on('profile', (sql) => {
|
||||||
log.verbose('Finished', sql);
|
logger.verbose('Finished', sql);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,7 +101,7 @@ function applyDbMigration(migration: Migration): Promise<void> {
|
|||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
db.exec(migration.statement, (err) => {
|
db.exec(migration.statement, (err) => {
|
||||||
if (err !== null) {
|
if (err !== null) {
|
||||||
log.error(`Failed to apply migration ${migration.name}`, err);
|
logger.error(`Failed to apply migration ${migration.name}`, err);
|
||||||
reject(err);
|
reject(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -118,31 +122,31 @@ async function applyMigration(migration: Migration) {
|
|||||||
if (post.songs && post.songs.length) {
|
if (post.songs && post.songs.length) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
log.info(
|
logger.info(
|
||||||
`Fetching songs for existing post ${current.toString().padStart(4, '0')} of ${total}`,
|
`Fetching songs for existing post ${current.toString().padStart(4, '0')} of ${total}`,
|
||||||
post.url
|
post.url
|
||||||
);
|
);
|
||||||
const songs = await TimelineReader.getSongInfoInPost(post);
|
const songs = await TimelineReader.getSongInfoInPost(post);
|
||||||
await saveSongInfoData(post.url, songs);
|
await saveSongInfoData(post.url, songs);
|
||||||
log.debug(`Fetched ${songs.length} songs for existing post`, post.url);
|
logger.debug(`Fetched ${songs.length} songs for existing post`, post.url);
|
||||||
}
|
}
|
||||||
log.debug(`Finished fetching songs`);
|
logger.debug(`Finished fetching songs`);
|
||||||
} else {
|
} else {
|
||||||
await applyDbMigration(migration);
|
await applyDbMigration(migration);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
db.on('open', () => {
|
db.on('open', () => {
|
||||||
log.info('Opened database');
|
logger.info('Opened database');
|
||||||
db.serialize();
|
db.serialize();
|
||||||
db.run('CREATE TABLE IF NOT EXISTS "migrations" ("id" integer,"name" TEXT, PRIMARY KEY (id))');
|
db.run('CREATE TABLE IF NOT EXISTS "migrations" ("id" integer,"name" TEXT, PRIMARY KEY (id))');
|
||||||
db.all('SELECT id FROM migrations', (err, rows: Migration[]) => {
|
db.all('SELECT id FROM migrations', (err, rows: Migration[]) => {
|
||||||
if (err !== null) {
|
if (err !== null) {
|
||||||
log.error('Could not fetch existing migrations', err);
|
logger.error('Could not fetch existing migrations', err);
|
||||||
databaseReady = true;
|
databaseReady = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
log.debug('Already applied migrations', rows);
|
logger.debug('Already applied migrations', rows);
|
||||||
const appliedMigrations: Set<number> = new Set(rows.map((row) => row['id']));
|
const appliedMigrations: Set<number> = new Set(rows.map((row) => row['id']));
|
||||||
const toApply = getMigrations().filter((m) => !appliedMigrations.has(m.id));
|
const toApply = getMigrations().filter((m) => !appliedMigrations.has(m.id));
|
||||||
let remaining = toApply.length;
|
let remaining = toApply.length;
|
||||||
@ -159,7 +163,7 @@ db.on('open', () => {
|
|||||||
databaseReady = true;
|
databaseReady = true;
|
||||||
}
|
}
|
||||||
if (err !== null) {
|
if (err !== null) {
|
||||||
log.error(`Failed to apply migration ${migration.name}`, err);
|
logger.error(`Failed to apply migration ${migration.name}`, err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
db.run(
|
db.run(
|
||||||
@ -167,10 +171,10 @@ db.on('open', () => {
|
|||||||
[migration.id, migration.name],
|
[migration.id, migration.name],
|
||||||
(e: Error) => {
|
(e: Error) => {
|
||||||
if (e !== null) {
|
if (e !== null) {
|
||||||
log.error(`Failed to mark migration ${migration.name} as applied`, e);
|
logger.error(`Failed to mark migration ${migration.name} as applied`, e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
log.info(`Applied migration ${migration.name}`);
|
logger.info(`Applied migration ${migration.name}`);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
@ -178,7 +182,7 @@ db.on('open', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
db.on('error', (err) => {
|
db.on('error', (err) => {
|
||||||
log.error('Error opening database', err);
|
logger.error('Error opening database', err);
|
||||||
});
|
});
|
||||||
|
|
||||||
function getMigrations(): Migration[] {
|
function getMigrations(): Migration[] {
|
||||||
@ -313,6 +317,13 @@ function getMigrations(): Migration[] {
|
|||||||
statement: `
|
statement: `
|
||||||
ALTER TABLE songs ADD COLUMN thumbnailWidth INTEGER NULL;
|
ALTER TABLE songs ADD COLUMN thumbnailWidth INTEGER NULL;
|
||||||
ALTER TABLE songs ADD COLUMN thumbnailHeight INTEGER NULL;`
|
ALTER TABLE songs ADD COLUMN thumbnailHeight INTEGER NULL;`
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 8,
|
||||||
|
name: 'song spotify url/uri',
|
||||||
|
statement: `
|
||||||
|
ALTER TABLE songs ADD COLUMN spotifyUrl TEXT NULL;
|
||||||
|
ALTER TABLE songs ADD COLUMN spotifyUri TEXT NULL;`
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@ -321,9 +332,9 @@ async function waitReady(): Promise<void> {
|
|||||||
// Simpler than a semaphore and is really only needed on startup
|
// Simpler than a semaphore and is really only needed on startup
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
const interval = setInterval(() => {
|
const interval = setInterval(() => {
|
||||||
log.verbose('Waiting for database to be ready');
|
logger.verbose('Waiting for database to be ready');
|
||||||
if (databaseReady) {
|
if (databaseReady) {
|
||||||
log.verbose('DB is ready');
|
logger.verbose('DB is ready');
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
resolve();
|
resolve();
|
||||||
}
|
}
|
||||||
@ -354,7 +365,7 @@ function saveAccountData(account: Account): Promise<void> {
|
|||||||
],
|
],
|
||||||
(err) => {
|
(err) => {
|
||||||
if (err !== null) {
|
if (err !== null) {
|
||||||
log.error(`Could not insert/update account ${account.id}`, err);
|
logger.error(`Could not insert/update account ${account.id}`, err);
|
||||||
reject(err);
|
reject(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -377,7 +388,7 @@ function savePostData(post: Post): Promise<void> {
|
|||||||
[post.id, post.content, post.created_at, post.url, post.account.url],
|
[post.id, post.content, post.created_at, post.url, post.account.url],
|
||||||
(postErr) => {
|
(postErr) => {
|
||||||
if (postErr !== null) {
|
if (postErr !== null) {
|
||||||
log.error(`Could not insert post ${post.url}`, postErr);
|
logger.error(`Could not insert post ${post.url}`, postErr);
|
||||||
reject(postErr);
|
reject(postErr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -405,7 +416,7 @@ function savePostTagData(post: Post): Promise<void> {
|
|||||||
[tag.url, tag.name],
|
[tag.url, tag.name],
|
||||||
(tagErr) => {
|
(tagErr) => {
|
||||||
if (tagErr !== null) {
|
if (tagErr !== null) {
|
||||||
log.error(`Could not insert/update tag ${tag.url}`, tagErr);
|
logger.error(`Could not insert/update tag ${tag.url}`, tagErr);
|
||||||
reject(tagErr);
|
reject(tagErr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -414,7 +425,7 @@ function savePostTagData(post: Post): Promise<void> {
|
|||||||
[post.url, tag.url],
|
[post.url, tag.url],
|
||||||
(posttagserr) => {
|
(posttagserr) => {
|
||||||
if (posttagserr !== null) {
|
if (posttagserr !== null) {
|
||||||
log.error(`Could not insert poststags ${tag.url}, ${post.url}`, posttagserr);
|
logger.error(`Could not insert poststags ${tag.url}, ${post.url}`, posttagserr);
|
||||||
reject(posttagserr);
|
reject(posttagserr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -444,14 +455,16 @@ function saveSongInfoData(postUrl: string, songs: SongInfo[]): Promise<void> {
|
|||||||
for (const song of songs) {
|
for (const song of songs) {
|
||||||
db.run(
|
db.run(
|
||||||
`
|
`
|
||||||
INSERT INTO songs (postedUrl, overviewUrl, type, youtubeUrl, title, artistName, thumbnailUrl, post_url, thumbnailWidth, thumbnailHeight)
|
INSERT INTO songs (postedUrl, overviewUrl, type, youtubeUrl, spotifyUrl, spotifyUri, title, artistName, thumbnailUrl, post_url, thumbnailWidth, thumbnailHeight)
|
||||||
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
||||||
`,
|
`,
|
||||||
[
|
[
|
||||||
song.postedUrl,
|
song.postedUrl,
|
||||||
song.pageUrl,
|
song.pageUrl,
|
||||||
song.type,
|
song.type,
|
||||||
song.youtubeUrl,
|
song.youtubeUrl,
|
||||||
|
song.spotifyUrl,
|
||||||
|
song.spotifyUri,
|
||||||
song.title,
|
song.title,
|
||||||
song.artistName,
|
song.artistName,
|
||||||
song.thumbnailUrl,
|
song.thumbnailUrl,
|
||||||
@ -461,7 +474,7 @@ function saveSongInfoData(postUrl: string, songs: SongInfo[]): Promise<void> {
|
|||||||
],
|
],
|
||||||
(songErr) => {
|
(songErr) => {
|
||||||
if (songErr !== null) {
|
if (songErr !== null) {
|
||||||
log.error(`Could not insert song ${song.postedUrl}`, songErr);
|
logger.error(`Could not insert song ${song.postedUrl}`, songErr);
|
||||||
reject(songErr);
|
reject(songErr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -479,20 +492,20 @@ function saveSongInfoData(postUrl: string, songs: SongInfo[]): Promise<void> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function savePost(post: Post, songs: SongInfo[]) {
|
export async function savePost(post: Post, songs: SongInfo[]) {
|
||||||
|
logger.debug(`Saving post ${post.url}`);
|
||||||
if (!databaseReady) {
|
if (!databaseReady) {
|
||||||
await waitReady();
|
await waitReady();
|
||||||
}
|
}
|
||||||
|
|
||||||
log.debug(`Saving post ${post.url}`);
|
|
||||||
const account = post.account;
|
const account = post.account;
|
||||||
await saveAccountData(account);
|
await saveAccountData(account);
|
||||||
log.debug(`Saved account data ${post.url}`);
|
logger.debug(`Saved account data ${post.url}`);
|
||||||
await savePostData(post);
|
await savePostData(post);
|
||||||
log.debug(`Saved post data ${post.url}`);
|
logger.debug(`Saved post data ${post.url}`);
|
||||||
await savePostTagData(post);
|
await savePostTagData(post);
|
||||||
log.debug(`Saved ${post.tags.length} tag data ${post.url}`);
|
logger.debug(`Saved ${post.tags.length} tag data ${post.url}`);
|
||||||
await saveSongInfoData(post.url, songs);
|
await saveSongInfoData(post.url, songs);
|
||||||
log.debug(
|
logger.debug(
|
||||||
`Saved ${songs.length} song info data ${post.url}`,
|
`Saved ${songs.length} song info data ${post.url}`,
|
||||||
songs.map((s) => s.thumbnailHeight)
|
songs.map((s) => s.thumbnailHeight)
|
||||||
);
|
);
|
||||||
@ -511,7 +524,7 @@ function getPostData(filterQuery: string, params: FilterParameter): Promise<Post
|
|||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
db.all(sql, params, (err, rows: PostRow[]) => {
|
db.all(sql, params, (err, rows: PostRow[]) => {
|
||||||
if (err != null) {
|
if (err != null) {
|
||||||
log.error('Error loading posts', err);
|
logger.error('Error loading posts', err);
|
||||||
reject(err);
|
reject(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -530,7 +543,7 @@ function getTagData(postIdsParams: string, postIds: string[]): Promise<Map<strin
|
|||||||
postIds,
|
postIds,
|
||||||
(tagErr, tagRows: PostTagRow[]) => {
|
(tagErr, tagRows: PostTagRow[]) => {
|
||||||
if (tagErr != null) {
|
if (tagErr != null) {
|
||||||
log.error('Error loading post tags', tagErr);
|
logger.error('Error loading post tags', tagErr);
|
||||||
reject(tagErr);
|
reject(tagErr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -551,14 +564,14 @@ function getTagData(postIdsParams: string, postIds: string[]): Promise<Map<strin
|
|||||||
function getSongData(postIdsParams: string, postIds: string[]): Promise<Map<string, SongInfo[]>> {
|
function getSongData(postIdsParams: string, postIds: string[]): Promise<Map<string, SongInfo[]>> {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
db.all(
|
db.all(
|
||||||
`SELECT post_url, songs.postedUrl, songs.overviewUrl, songs.type, songs.youtubeUrl,
|
`SELECT post_url, songs.postedUrl, songs.overviewUrl, songs.type, songs.youtubeUrl, songs.spotifyUri, songs.spotifyUri,
|
||||||
songs.title, songs.artistName, songs.thumbnailUrl, songs.post_url, songs.thumbnailWidth, songs.thumbnailHeight
|
songs.title, songs.artistName, songs.thumbnailUrl, songs.post_url, songs.thumbnailWidth, songs.thumbnailHeight
|
||||||
FROM songs
|
FROM songs
|
||||||
WHERE post_url IN (${postIdsParams});`,
|
WHERE post_url IN (${postIdsParams});`,
|
||||||
postIds,
|
postIds,
|
||||||
(tagErr, tagRows: SongRow[]) => {
|
(tagErr, tagRows: SongRow[]) => {
|
||||||
if (tagErr != null) {
|
if (tagErr != null) {
|
||||||
log.error('Error loading post songs', tagErr);
|
logger.error('Error loading post songs', tagErr);
|
||||||
reject(tagErr);
|
reject(tagErr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -567,6 +580,8 @@ function getSongData(postIdsParams: string, postIds: string[]): Promise<Map<stri
|
|||||||
const info = {
|
const info = {
|
||||||
pageUrl: item.overviewUrl,
|
pageUrl: item.overviewUrl,
|
||||||
youtubeUrl: item.youtubeUrl,
|
youtubeUrl: item.youtubeUrl,
|
||||||
|
spotifyUrl: item.spotifyUrl,
|
||||||
|
spotifyUri: item.spotifyUri,
|
||||||
type: item.type,
|
type: item.type,
|
||||||
title: item.title,
|
title: item.title,
|
||||||
artistName: item.artistName,
|
artistName: item.artistName,
|
||||||
@ -580,7 +595,7 @@ function getSongData(postIdsParams: string, postIds: string[]): Promise<Map<stri
|
|||||||
},
|
},
|
||||||
new Map()
|
new Map()
|
||||||
);
|
);
|
||||||
log.verbose('songMap', songMap);
|
logger.verbose('songMap', songMap);
|
||||||
resolve(songMap);
|
resolve(songMap);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -599,7 +614,7 @@ function getAvatarData(
|
|||||||
accountUrls,
|
accountUrls,
|
||||||
(err, rows: AccountAvatarRow[]) => {
|
(err, rows: AccountAvatarRow[]) => {
|
||||||
if (err != null) {
|
if (err != null) {
|
||||||
log.error('Error loading avatars', err);
|
logger.error('Error loading avatars', err);
|
||||||
reject(err);
|
reject(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -633,7 +648,7 @@ function getSongThumbnailData(
|
|||||||
thumbUrls,
|
thumbUrls,
|
||||||
(err, rows: SongThumbnailAvatarRow[]) => {
|
(err, rows: SongThumbnailAvatarRow[]) => {
|
||||||
if (err != null) {
|
if (err != null) {
|
||||||
log.error('Error loading avatars', err);
|
logger.error('Error loading avatars', err);
|
||||||
reject(err);
|
reject(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
159
src/lib/server/playlist/oauthPlaylistAdder.ts
Normal file
159
src/lib/server/playlist/oauthPlaylistAdder.ts
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
import { Logger } from '$lib/log';
|
||||||
|
import type { OauthResponse } from '$lib/mastodon/response';
|
||||||
|
import fs from 'fs/promises';
|
||||||
|
|
||||||
|
export abstract class OauthPlaylistAdder {
|
||||||
|
/// How many minutes before expiry the token will be refreshed
|
||||||
|
protected refresh_time: number = 15;
|
||||||
|
protected logger: Logger = new Logger('OauthPlaylistAdder');
|
||||||
|
|
||||||
|
protected constructor(
|
||||||
|
protected apiBase: string,
|
||||||
|
protected token_file_name: string
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async authCodeExists(): Promise<boolean> {
|
||||||
|
try {
|
||||||
|
const fileHandle = await fs.open(this.token_file_name);
|
||||||
|
await fileHandle.close();
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
this.logger.info('No auth token yet, authorizing...');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected constructAuthUrlInternal(
|
||||||
|
endpointUrl: string,
|
||||||
|
clientId: string,
|
||||||
|
scope: string,
|
||||||
|
redirectUri: URL,
|
||||||
|
additionalParameters: Map<string, string> = new Map()
|
||||||
|
): URL {
|
||||||
|
const authUrl = new URL(endpointUrl);
|
||||||
|
authUrl.searchParams.append('client_id', clientId);
|
||||||
|
authUrl.searchParams.append('redirect_uri', redirectUri.toString());
|
||||||
|
authUrl.searchParams.append('response_type', 'code');
|
||||||
|
authUrl.searchParams.append('scope', scope);
|
||||||
|
for (let p of additionalParameters.entries()) {
|
||||||
|
authUrl.searchParams.append(p[0], p[1]);
|
||||||
|
}
|
||||||
|
return authUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async receivedAuthCodeInternal(
|
||||||
|
tokenUrl: URL,
|
||||||
|
clientId: string,
|
||||||
|
code: string,
|
||||||
|
url: URL,
|
||||||
|
client_secret?: string,
|
||||||
|
customHeader?: HeadersInit
|
||||||
|
) {
|
||||||
|
this.logger.debug('received code');
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.append('client_id', clientId);
|
||||||
|
params.append('code', code);
|
||||||
|
params.append('grant_type', 'authorization_code');
|
||||||
|
params.append('redirect_uri', `${url.origin}${url.pathname}`);
|
||||||
|
if (client_secret) {
|
||||||
|
params.append('client_secret', client_secret);
|
||||||
|
}
|
||||||
|
this.logger.debug('sending token req', params);
|
||||||
|
const resp: OauthResponse = await fetch(tokenUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
body: params,
|
||||||
|
headers: customHeader
|
||||||
|
}).then((r) => r.json());
|
||||||
|
this.logger.debug('received access token', resp);
|
||||||
|
let expiration = new Date();
|
||||||
|
expiration.setTime(expiration.getTime() + resp.expires_in * 1000);
|
||||||
|
expiration.setSeconds(expiration.getSeconds() + resp.expires_in);
|
||||||
|
resp.expires = expiration;
|
||||||
|
await fs.writeFile(this.token_file_name, JSON.stringify(resp));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async auth(): Promise<OauthResponse | null> {
|
||||||
|
try {
|
||||||
|
const token_file = await fs.readFile(this.token_file_name, { encoding: 'utf8' });
|
||||||
|
let token = JSON.parse(token_file);
|
||||||
|
if (token.expires) {
|
||||||
|
if (typeof token.expires === typeof '') {
|
||||||
|
token.expires = new Date(token.expires);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return token;
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('Could not read access token', e);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async shouldRefreshToken(): Promise<{ token: OauthResponse; refresh: boolean } | null> {
|
||||||
|
const token = await this.auth();
|
||||||
|
if (token == null || !token?.expires) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let refreshAt = new Date();
|
||||||
|
refreshAt.setTime(refreshAt.getTime() - this.refresh_time * 60 * 1000);
|
||||||
|
this.logger.info('token expiry', token.expires, 'vs refresh @', refreshAt);
|
||||||
|
if (token.expires.getTime() > refreshAt.getTime()) {
|
||||||
|
return {
|
||||||
|
token: token,
|
||||||
|
refresh: false
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.info(
|
||||||
|
'Token expires',
|
||||||
|
token.expires,
|
||||||
|
token.expires.getTime(),
|
||||||
|
`which is after the refresh time`,
|
||||||
|
refreshAt,
|
||||||
|
refreshAt.getTime()
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
token: token,
|
||||||
|
refresh: true
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async requestRefreshToken(
|
||||||
|
tokenUrl: URL,
|
||||||
|
clientId: string,
|
||||||
|
refresh_token: string,
|
||||||
|
redirect_uri?: string,
|
||||||
|
client_secret?: string,
|
||||||
|
customHeader?: HeadersInit
|
||||||
|
) {
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.append('client_id', clientId);
|
||||||
|
params.append('grant_type', 'refresh_token');
|
||||||
|
params.append('refresh_token', refresh_token);
|
||||||
|
if (client_secret) {
|
||||||
|
params.append('client_secret', client_secret);
|
||||||
|
}
|
||||||
|
if (redirect_uri) {
|
||||||
|
params.append('redirect_uri', redirect_uri);
|
||||||
|
}
|
||||||
|
this.logger.debug('sending token req', params);
|
||||||
|
const resp: OauthResponse = await fetch(tokenUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
body: params,
|
||||||
|
headers: customHeader
|
||||||
|
}).then((r) => r.json());
|
||||||
|
this.logger.verbose('received access token', resp);
|
||||||
|
if (resp.error) {
|
||||||
|
this.logger.error('token resp error', resp);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
if (!resp.refresh_token) {
|
||||||
|
resp.refresh_token = refresh_token;
|
||||||
|
}
|
||||||
|
let expiration = new Date();
|
||||||
|
expiration.setTime(expiration.getTime() + resp.expires_in * 1000);
|
||||||
|
expiration.setSeconds(expiration.getSeconds() + resp.expires_in);
|
||||||
|
resp.expires = expiration;
|
||||||
|
await fs.writeFile(this.token_file_name, JSON.stringify(resp));
|
||||||
|
return resp;
|
||||||
|
}
|
||||||
|
}
|
5
src/lib/server/playlist/playlistAdder.ts
Normal file
5
src/lib/server/playlist/playlistAdder.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import type { SongInfo } from '$lib/odesliResponse';
|
||||||
|
|
||||||
|
export interface PlaylistAdder {
|
||||||
|
addToPlaylist(song: SongInfo): Promise<void>;
|
||||||
|
}
|
124
src/lib/server/playlist/spotifyPlaylistAdder.ts
Normal file
124
src/lib/server/playlist/spotifyPlaylistAdder.ts
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
import { SPOTIFY_CLIENT_ID, SPOTIFY_CLIENT_SECRET, SPOTIFY_PLAYLIST_ID } from '$env/static/private';
|
||||||
|
import { Logger } from '$lib/log';
|
||||||
|
import type { OauthResponse } from '$lib/mastodon/response';
|
||||||
|
import type { SongInfo } from '$lib/odesliResponse';
|
||||||
|
import { OauthPlaylistAdder } from './oauthPlaylistAdder';
|
||||||
|
import type { PlaylistAdder } from './playlistAdder';
|
||||||
|
|
||||||
|
export class SpotifyPlaylistAdder extends OauthPlaylistAdder implements PlaylistAdder {
|
||||||
|
public constructor() {
|
||||||
|
super('https://api.spotify.com/v1', 'spotify_auth_token');
|
||||||
|
this.logger = new Logger('SpotifyPlaylistAdder');
|
||||||
|
}
|
||||||
|
|
||||||
|
public constructAuthUrl(redirectUri: URL): URL {
|
||||||
|
const endpoint = 'https://accounts.spotify.com/authorize';
|
||||||
|
return this.constructAuthUrlInternal(
|
||||||
|
endpoint,
|
||||||
|
SPOTIFY_CLIENT_ID,
|
||||||
|
'playlist-modify-private playlist-modify-public',
|
||||||
|
redirectUri
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async receivedAuthCode(code: string, url: URL) {
|
||||||
|
this.logger.debug('received code');
|
||||||
|
const authHeader =
|
||||||
|
'Basic ' + Buffer.from(SPOTIFY_CLIENT_ID + ':' + SPOTIFY_CLIENT_SECRET).toString('base64');
|
||||||
|
|
||||||
|
const tokenUrl = new URL('https://accounts.spotify.com/api/token');
|
||||||
|
await this.receivedAuthCodeInternal(tokenUrl, SPOTIFY_CLIENT_ID, code, url, undefined, {
|
||||||
|
Authorization: authHeader
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async refreshToken(force: boolean = false): Promise<OauthResponse | null> {
|
||||||
|
const tokenInfo = await this.shouldRefreshToken();
|
||||||
|
if (tokenInfo == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let token = tokenInfo.token;
|
||||||
|
if (!tokenInfo.refresh && !force) {
|
||||||
|
return token;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!token.refresh_token) {
|
||||||
|
this.logger.error('Need to refresh access token, but no refresh token provided');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const authHeader =
|
||||||
|
'Basic ' + Buffer.from(SPOTIFY_CLIENT_ID + ':' + SPOTIFY_CLIENT_SECRET).toString('base64');
|
||||||
|
const tokenUrl = new URL('https://accounts.spotify.com/api/token');
|
||||||
|
return await this.requestRefreshToken(
|
||||||
|
tokenUrl,
|
||||||
|
SPOTIFY_CLIENT_ID,
|
||||||
|
token.refresh_token,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
{ Authorization: authHeader }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async addToPlaylistRetry(song: SongInfo, remaning: number = 3) {
|
||||||
|
if (remaning < 0) {
|
||||||
|
this.logger.error('max retries reached, song will not be added to spotify playlist');
|
||||||
|
}
|
||||||
|
this.logger.debug('addToSpotifyPlaylist', remaning);
|
||||||
|
const token = await this.refreshToken();
|
||||||
|
if (token == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SPOTIFY_PLAYLIST_ID || SPOTIFY_PLAYLIST_ID === 'CHANGE_ME') {
|
||||||
|
this.logger.debug('no spotify playlist ID configured');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!song.spotifyUri) {
|
||||||
|
this.logger.info('Skip adding song to spotify playlist, no Uri', song);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO. Spotify's check for "is this song already in the playlist" is... ugh
|
||||||
|
/*
|
||||||
|
const playlistItemsUrl = new URL(`${this.apiBase}/playlists/${SPOTIFY_PLAYLIST_ID}/tracks`);
|
||||||
|
playlistItemsUrl.searchParams.append('videoId', youtubeId);
|
||||||
|
playlistItemsUrl.searchParams.append('playlistId', SPOTIFY_PLAYLIST_ID);
|
||||||
|
playlistItemsUrl.searchParams.append('part', 'id');*/
|
||||||
|
/*const existingPlaylistItem = await fetch(this.apiBase + '/playlistItems', {
|
||||||
|
headers: { Authorization: `${token.token_type} ${token.access_token}` }
|
||||||
|
}).then((r) => r.json());
|
||||||
|
logger.debug('existingPlaylistItem', existingPlaylistItem);
|
||||||
|
if (existingPlaylistItem.pageInfo && existingPlaylistItem.pageInfo.totalResults > 0) {
|
||||||
|
logger.info('Item already in playlist');
|
||||||
|
return;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
//const searchParams = new URLSearchParams([['part', 'snippet']]);
|
||||||
|
const options: RequestInit = {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { Authorization: `${token.token_type} ${token.access_token}` },
|
||||||
|
body: JSON.stringify({
|
||||||
|
uris: [song.spotifyUri]
|
||||||
|
})
|
||||||
|
};
|
||||||
|
const apiUrl = new URL(`${this.apiBase}/playlists/${SPOTIFY_PLAYLIST_ID}/tracks`);
|
||||||
|
const resp = await fetch(apiUrl, options);
|
||||||
|
const respObj = await resp.json();
|
||||||
|
if (respObj.error) {
|
||||||
|
this.logger.debug('Add to playlist failed', song.spotifyUri, respObj.error);
|
||||||
|
if (respObj.error.status === 401) {
|
||||||
|
const token = await this.refreshToken(true);
|
||||||
|
if (token == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.addToPlaylistRetry(song, remaning--);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.logger.info('Added to playlist', song.spotifyUri, song.title);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public async addToPlaylist(song: SongInfo) {
|
||||||
|
await this.addToPlaylistRetry(song);
|
||||||
|
}
|
||||||
|
}
|
131
src/lib/server/playlist/ytPlaylistAdder.ts
Normal file
131
src/lib/server/playlist/ytPlaylistAdder.ts
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
import {
|
||||||
|
BASE_URL,
|
||||||
|
YOUTUBE_CLIENT_ID,
|
||||||
|
YOUTUBE_CLIENT_SECRET,
|
||||||
|
YOUTUBE_PLAYLIST_ID
|
||||||
|
} from '$env/static/private';
|
||||||
|
import { Logger } from '$lib/log';
|
||||||
|
import type { OauthResponse } from '$lib/mastodon/response';
|
||||||
|
import type { SongInfo } from '$lib/odesliResponse';
|
||||||
|
import { OauthPlaylistAdder } from './oauthPlaylistAdder';
|
||||||
|
import type { PlaylistAdder } from './playlistAdder';
|
||||||
|
|
||||||
|
export class YoutubePlaylistAdder extends OauthPlaylistAdder implements PlaylistAdder {
|
||||||
|
public constructor() {
|
||||||
|
super('https://www.googleapis.com/youtube/v3', 'yt_auth_token');
|
||||||
|
this.logger = new Logger('YoutubePlaylistAdder');
|
||||||
|
}
|
||||||
|
|
||||||
|
public constructAuthUrl(redirectUri: URL): URL {
|
||||||
|
let additionalParameters = new Map([
|
||||||
|
['access_type', 'offline'],
|
||||||
|
['include_granted_scopes', 'false']
|
||||||
|
]);
|
||||||
|
const endpoint = 'https://accounts.google.com/o/oauth2/v2/auth';
|
||||||
|
return this.constructAuthUrlInternal(
|
||||||
|
endpoint,
|
||||||
|
YOUTUBE_CLIENT_ID,
|
||||||
|
'https://www.googleapis.com/auth/youtube',
|
||||||
|
redirectUri,
|
||||||
|
additionalParameters
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async receivedAuthCode(code: string, url: URL) {
|
||||||
|
this.logger.debug('received code');
|
||||||
|
const tokenUrl = new URL('https://oauth2.googleapis.com/token');
|
||||||
|
await this.receivedAuthCodeInternal(
|
||||||
|
tokenUrl,
|
||||||
|
YOUTUBE_CLIENT_ID,
|
||||||
|
code,
|
||||||
|
url,
|
||||||
|
YOUTUBE_CLIENT_SECRET
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async refreshToken(): Promise<OauthResponse | null> {
|
||||||
|
const tokenInfo = await this.shouldRefreshToken();
|
||||||
|
if (tokenInfo == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let token = tokenInfo.token;
|
||||||
|
if (!tokenInfo.refresh) {
|
||||||
|
return token;
|
||||||
|
}
|
||||||
|
if (!token.refresh_token) {
|
||||||
|
this.logger.error('Need to refresh access token, but no refresh token provided');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tokenUrl = new URL('https://oauth2.googleapis.com/token');
|
||||||
|
return await this.requestRefreshToken(
|
||||||
|
tokenUrl,
|
||||||
|
YOUTUBE_CLIENT_ID,
|
||||||
|
token.refresh_token,
|
||||||
|
`${BASE_URL}/ytauth`,
|
||||||
|
YOUTUBE_CLIENT_SECRET
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async addToPlaylist(song: SongInfo) {
|
||||||
|
this.logger.debug('addToYoutubePlaylist');
|
||||||
|
const token = await this.refreshToken();
|
||||||
|
if (token == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!YOUTUBE_PLAYLIST_ID || YOUTUBE_PLAYLIST_ID === 'CHANGE_ME') {
|
||||||
|
this.logger.debug('no playlist ID configured');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!song.youtubeUrl) {
|
||||||
|
this.logger.info('Skip adding song to YT playlist, no youtube Url', song);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const songUrl = new URL(song.youtubeUrl);
|
||||||
|
const youtubeId = songUrl.searchParams.get('v');
|
||||||
|
if (!youtubeId) {
|
||||||
|
this.logger.debug(
|
||||||
|
'Skip adding song to YT playlist, could not extract YT id from URL',
|
||||||
|
song.youtubeUrl
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.logger.debug('Found YT id from URL', song.youtubeUrl, youtubeId);
|
||||||
|
|
||||||
|
const playlistItemsUrl = new URL(this.apiBase + '/playlistItems');
|
||||||
|
playlistItemsUrl.searchParams.append('videoId', youtubeId);
|
||||||
|
playlistItemsUrl.searchParams.append('playlistId', YOUTUBE_PLAYLIST_ID);
|
||||||
|
playlistItemsUrl.searchParams.append('part', 'id');
|
||||||
|
const existingPlaylistItem = await fetch(playlistItemsUrl, {
|
||||||
|
headers: { Authorization: `${token.token_type} ${token.access_token}` }
|
||||||
|
}).then((r) => r.json());
|
||||||
|
if (existingPlaylistItem.pageInfo && existingPlaylistItem.pageInfo.totalResults > 0) {
|
||||||
|
this.logger.info('Item already in playlist', existingPlaylistItem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const addItemUrl = new URL(this.apiBase + '/playlistItems');
|
||||||
|
addItemUrl.searchParams.append('part', 'snippet');
|
||||||
|
const options: RequestInit = {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { Authorization: `${token.token_type} ${token.access_token}` },
|
||||||
|
body: JSON.stringify({
|
||||||
|
snippet: {
|
||||||
|
playlistId: YOUTUBE_PLAYLIST_ID,
|
||||||
|
resourceId: {
|
||||||
|
videoId: youtubeId,
|
||||||
|
kind: 'youtube#video'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
};
|
||||||
|
const resp = await fetch(addItemUrl, options);
|
||||||
|
const respObj = await resp.json();
|
||||||
|
this.logger.info('Added to playlist', youtubeId, song.title);
|
||||||
|
if (respObj.error) {
|
||||||
|
this.logger.debug('Add to playlist failed', respObj.error.errors);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,10 +1,12 @@
|
|||||||
import { BASE_URL, WEBSUB_HUB } from '$env/static/private';
|
import { BASE_URL, WEBSUB_HUB } from '$env/static/private';
|
||||||
import { PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME } from '$env/static/public';
|
import { PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME } from '$env/static/public';
|
||||||
import type { Post } from '$lib//mastodon/response';
|
import type { Post } from '$lib//mastodon/response';
|
||||||
import { log } from '$lib/log';
|
import { Logger } from '$lib/log';
|
||||||
import { Feed } from 'feed';
|
import { Feed } from 'feed';
|
||||||
import fs from 'fs/promises';
|
import fs from 'fs/promises';
|
||||||
|
|
||||||
|
const logger = new Logger('RSS');
|
||||||
|
|
||||||
export function createFeed(posts: Post[]): Feed {
|
export function createFeed(posts: Post[]): Feed {
|
||||||
const baseUrl = BASE_URL.endsWith('/') ? BASE_URL : BASE_URL + '/';
|
const baseUrl = BASE_URL.endsWith('/') ? BASE_URL : BASE_URL + '/';
|
||||||
const hub = WEBSUB_HUB ? WEBSUB_HUB : undefined;
|
const hub = WEBSUB_HUB ? WEBSUB_HUB : undefined;
|
||||||
@ -60,6 +62,6 @@ export async function saveAtomFeed(feed: Feed) {
|
|||||||
body: params
|
body: params
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log.error('Failed to update WebSub hub', e);
|
logger.error('Failed to update WebSub hub', e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
import {
|
import {
|
||||||
HASHTAG_FILTER,
|
HASHTAG_FILTER,
|
||||||
|
MASTODON_ACCESS_TOKEN,
|
||||||
MASTODON_INSTANCE,
|
MASTODON_INSTANCE,
|
||||||
ODESLI_API_KEY,
|
ODESLI_API_KEY,
|
||||||
YOUTUBE_API_KEY
|
YOUTUBE_API_KEY
|
||||||
} from '$env/static/private';
|
} from '$env/static/private';
|
||||||
import { log } from '$lib/log';
|
import { Logger } from '$lib/log';
|
||||||
import type {
|
import type {
|
||||||
Account,
|
Account,
|
||||||
AccountAvatar,
|
AccountAvatar,
|
||||||
@ -24,12 +25,17 @@ import {
|
|||||||
savePost,
|
savePost,
|
||||||
saveSongThumbnail
|
saveSongThumbnail
|
||||||
} from '$lib/server/db';
|
} from '$lib/server/db';
|
||||||
|
import { SpotifyPlaylistAdder } from '$lib/server/playlist/spotifyPlaylistAdder';
|
||||||
|
import { YoutubePlaylistAdder } from '$lib/server/playlist/ytPlaylistAdder';
|
||||||
import { createFeed, saveAtomFeed } from '$lib/server/rss';
|
import { createFeed, saveAtomFeed } from '$lib/server/rss';
|
||||||
import { sleep } from '$lib/sleep';
|
import { sleep } from '$lib/sleep';
|
||||||
import crypto from 'crypto';
|
import crypto from 'crypto';
|
||||||
import fs from 'fs/promises';
|
import fs from 'fs/promises';
|
||||||
|
import { console } from 'inspector/promises';
|
||||||
import sharp from 'sharp';
|
import sharp from 'sharp';
|
||||||
|
import { URL, URLSearchParams } from 'url';
|
||||||
import { WebSocket } from 'ws';
|
import { WebSocket } from 'ws';
|
||||||
|
import type { PlaylistAdder } from './playlist/playlistAdder';
|
||||||
|
|
||||||
const URL_REGEX = new RegExp(/href="(?<postUrl>[^>]+?)" target="_blank"/gm);
|
const URL_REGEX = new RegExp(/href="(?<postUrl>[^>]+?)" target="_blank"/gm);
|
||||||
const INVIDIOUS_REGEX = new RegExp(/invidious.*?watch.*?v=(?<videoId>[a-zA-Z_0-9-]+)/gm);
|
const INVIDIOUS_REGEX = new RegExp(/invidious.*?watch.*?v=(?<videoId>[a-zA-Z_0-9-]+)/gm);
|
||||||
@ -39,10 +45,14 @@ const YOUTUBE_REGEX = new RegExp(
|
|||||||
|
|
||||||
export class TimelineReader {
|
export class TimelineReader {
|
||||||
private static _instance: TimelineReader;
|
private static _instance: TimelineReader;
|
||||||
|
private lastPosts: string[] = [];
|
||||||
|
private playlistAdders: PlaylistAdder[];
|
||||||
|
private logger: Logger;
|
||||||
|
|
||||||
private static async isMusicVideo(videoId: string) {
|
private async isMusicVideo(videoId: string) {
|
||||||
if (!YOUTUBE_API_KEY || YOUTUBE_API_KEY === 'CHANGE_ME') {
|
if (!YOUTUBE_API_KEY || YOUTUBE_API_KEY === 'CHANGE_ME') {
|
||||||
// Assume that it *is* a music link when no YT API key is provided
|
// Assume that it *is* a music link when no YT API key is provided
|
||||||
|
this.logger.debug('YT API not configured');
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
const searchParams = new URLSearchParams([
|
const searchParams = new URLSearchParams([
|
||||||
@ -54,13 +64,13 @@ export class TimelineReader {
|
|||||||
const resp = await fetch(youtubeVideoUrl);
|
const resp = await fetch(youtubeVideoUrl);
|
||||||
const respObj = await resp.json();
|
const respObj = await resp.json();
|
||||||
if (!respObj.items.length) {
|
if (!respObj.items.length) {
|
||||||
console.warn('Could not find video with id', videoId);
|
this.logger.warn('Could not find video with id', videoId);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const item = respObj.items[0];
|
const item = respObj.items[0];
|
||||||
if (!item.snippet) {
|
if (!item.snippet) {
|
||||||
console.warn('Could not load snippet for video', videoId, item);
|
this.logger.warn('Could not load snippet for video', videoId, item);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (item.snippet.tags?.includes('music')) {
|
if (item.snippet.tags?.includes('music')) {
|
||||||
@ -78,15 +88,19 @@ export class TimelineReader {
|
|||||||
const categoryTitle: string = await fetch(youtubeCategoryUrl)
|
const categoryTitle: string = await fetch(youtubeCategoryUrl)
|
||||||
.then((r) => r.json())
|
.then((r) => r.json())
|
||||||
.then((r) => r.items[0]?.snippet?.title);
|
.then((r) => r.items[0]?.snippet?.title);
|
||||||
|
this.logger.debug('YT category', categoryTitle);
|
||||||
return categoryTitle === 'Music';
|
return categoryTitle === 'Music';
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async getSongInfoInPost(post: Post): Promise<SongInfo[]> {
|
public async getSongInfoInPost(post: Post): Promise<SongInfo[]> {
|
||||||
const urlMatches = post.content.matchAll(URL_REGEX);
|
const urlMatches = post.content.matchAll(URL_REGEX);
|
||||||
const songs: SongInfo[] = [];
|
const songs: SongInfo[] = [];
|
||||||
for (const match of urlMatches) {
|
for (const match of urlMatches) {
|
||||||
if (match === undefined || match.groups === undefined) {
|
if (match === undefined || match.groups === undefined) {
|
||||||
log.warn('Match listed in allMatches, but either it or its groups are undefined', match);
|
this.logger.warn(
|
||||||
|
'Match listed in allMatches, but either it or its groups are undefined',
|
||||||
|
match
|
||||||
|
);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
const urlMatch = match.groups.postUrl.toString();
|
const urlMatch = match.groups.postUrl.toString();
|
||||||
@ -94,14 +108,14 @@ export class TimelineReader {
|
|||||||
try {
|
try {
|
||||||
url = new URL(urlMatch);
|
url = new URL(urlMatch);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log.error('URL found via Regex does not seem to be a valud url', urlMatch, e);
|
this.logger.error('URL found via Regex does not seem to be a valud url', urlMatch, e);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check *all* found url and let odesli determine if it is music or not
|
// Check *all* found url and let odesli determine if it is music or not
|
||||||
log.debug(`Checking ${url} if it contains song data`);
|
this.logger.debug(`Checking ${url} if it contains song data`);
|
||||||
const info = await TimelineReader.getSongInfo(url);
|
const info = await this.getSongInfo(url);
|
||||||
log.debug(`Found song info for ${url}?`, info);
|
//this.logger.debug(`Found song info for ${url}?`, info);
|
||||||
if (info) {
|
if (info) {
|
||||||
songs.push(info);
|
songs.push(info);
|
||||||
}
|
}
|
||||||
@ -109,9 +123,9 @@ export class TimelineReader {
|
|||||||
return songs;
|
return songs;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async getSongInfo(url: URL, remainingTries = 6): Promise<SongInfo | null> {
|
private async getSongInfo(url: URL, remainingTries = 6): Promise<SongInfo | null> {
|
||||||
if (remainingTries === 0) {
|
if (remainingTries === 0) {
|
||||||
log.error('No tries remaining. Lookup failed!');
|
this.logger.error('No tries remaining. Lookup failed!');
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if (url.hostname === 'songwhip.com') {
|
if (url.hostname === 'songwhip.com') {
|
||||||
@ -143,6 +157,7 @@ export class TimelineReader {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
const info = odesliInfo.entitiesByUniqueId[odesliInfo.entityUniqueId];
|
const info = odesliInfo.entitiesByUniqueId[odesliInfo.entityUniqueId];
|
||||||
|
//this.logger.debug('odesli response', info);
|
||||||
const platform: Platform = 'youtube';
|
const platform: Platform = 'youtube';
|
||||||
if (info.platforms.includes(platform)) {
|
if (info.platforms.includes(platform)) {
|
||||||
const youtubeId =
|
const youtubeId =
|
||||||
@ -150,33 +165,50 @@ export class TimelineReader {
|
|||||||
YOUTUBE_REGEX.exec(url.href)?.groups?.videoId ??
|
YOUTUBE_REGEX.exec(url.href)?.groups?.videoId ??
|
||||||
new URL(odesliInfo.pageUrl).pathname.split('/y/').pop();
|
new URL(odesliInfo.pageUrl).pathname.split('/y/').pop();
|
||||||
if (youtubeId === undefined) {
|
if (youtubeId === undefined) {
|
||||||
log.warn('Looks like a youtube video, but could not extract a video id', url, odesliInfo);
|
this.logger.warn(
|
||||||
|
'Looks like a youtube video, but could not extract a video id',
|
||||||
|
url,
|
||||||
|
odesliInfo
|
||||||
|
);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
const isMusic = await TimelineReader.isMusicVideo(youtubeId);
|
const isMusic = await this.isMusicVideo(youtubeId);
|
||||||
if (!isMusic) {
|
if (!isMusic) {
|
||||||
log.debug('Probably not a music video', url);
|
this.logger.debug('Probably not a music video', youtubeId, url);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return {
|
const spotify: Platform = 'spotify';
|
||||||
|
const songInfo = {
|
||||||
...info,
|
...info,
|
||||||
pageUrl: odesliInfo.pageUrl,
|
pageUrl: odesliInfo.pageUrl,
|
||||||
youtubeUrl: odesliInfo.linksByPlatform[platform]?.url,
|
youtubeUrl: odesliInfo.linksByPlatform[platform]?.url,
|
||||||
|
spotifyUrl: odesliInfo.linksByPlatform[spotify]?.url,
|
||||||
|
spotifyUri: odesliInfo.linksByPlatform[spotify]?.nativeAppUriDesktop,
|
||||||
postedUrl: url.toString()
|
postedUrl: url.toString()
|
||||||
} as SongInfo;
|
} as SongInfo;
|
||||||
|
if (songInfo.youtubeUrl && !songInfo.spotifyUrl) {
|
||||||
|
this.logger.warn('SongInfo with YT, but no spotify URL', odesliInfo);
|
||||||
|
}
|
||||||
|
return songInfo;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (e instanceof Error && e.cause === 429) {
|
if (e instanceof Error && e.cause === 429) {
|
||||||
log.warn('song.link rate limit reached. Trying again in 10 seconds');
|
this.logger.warn('song.link rate limit reached. Trying again in 10 seconds');
|
||||||
await sleep(10_000);
|
await sleep(10_000);
|
||||||
return await this.getSongInfo(url, remainingTries - 1);
|
return await this.getSongInfo(url, remainingTries - 1);
|
||||||
}
|
}
|
||||||
log.error(`Failed to load ${url} info from song.link`, e);
|
this.logger.error(`Failed to load ${url} info from song.link`, e);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async resizeAvatar(
|
private async addToPlaylist(song: SongInfo) {
|
||||||
|
for (let adder of this.playlistAdders) {
|
||||||
|
await adder.addToPlaylist(song);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async resizeAvatar(
|
||||||
baseName: string,
|
baseName: string,
|
||||||
size: number,
|
size: number,
|
||||||
suffix: string,
|
suffix: string,
|
||||||
@ -189,15 +221,15 @@ export class TimelineReader {
|
|||||||
.then(() => true)
|
.then(() => true)
|
||||||
.catch(() => false);
|
.catch(() => false);
|
||||||
if (exists) {
|
if (exists) {
|
||||||
log.debug('File already exists', fileName);
|
this.logger.debug('File already exists', fileName);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
log.debug('Saving avatar', fileName);
|
this.logger.debug('Saving avatar', fileName);
|
||||||
await sharpAvatar.resize(size).toFile(fileName);
|
await sharpAvatar.resize(size).toFile(fileName);
|
||||||
return fileName;
|
return fileName;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static resizeAvatarPromiseMaker(
|
private resizeAvatarPromiseMaker(
|
||||||
avatarFilenameBase: string,
|
avatarFilenameBase: string,
|
||||||
baseSize: number,
|
baseSize: number,
|
||||||
maxPixelDensity: number,
|
maxPixelDensity: number,
|
||||||
@ -210,20 +242,14 @@ export class TimelineReader {
|
|||||||
for (let i = 1; i <= maxPixelDensity; i++) {
|
for (let i = 1; i <= maxPixelDensity; i++) {
|
||||||
promises.push(
|
promises.push(
|
||||||
...formats.map((f) =>
|
...formats.map((f) =>
|
||||||
TimelineReader.resizeAvatar(
|
this.resizeAvatar(avatarFilenameBase, baseSize * i, `${i}x.${f}`, 'avatars', sharpAvatar)
|
||||||
avatarFilenameBase,
|
|
||||||
baseSize * i,
|
|
||||||
`${i}x.${f}`,
|
|
||||||
'avatars',
|
|
||||||
sharpAvatar
|
|
||||||
)
|
|
||||||
.then(
|
.then(
|
||||||
(fn) =>
|
(fn) =>
|
||||||
({
|
({
|
||||||
accountUrl: accountUrl,
|
accountUrl: accountUrl,
|
||||||
file: fn,
|
file: fn,
|
||||||
sizeDescriptor: `${i}x`
|
sizeDescriptor: `${i}x`
|
||||||
} as AccountAvatar)
|
}) as AccountAvatar
|
||||||
)
|
)
|
||||||
.then(saveAvatar)
|
.then(saveAvatar)
|
||||||
)
|
)
|
||||||
@ -232,7 +258,7 @@ export class TimelineReader {
|
|||||||
return promises;
|
return promises;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static resizeThumbnailPromiseMaker(
|
private resizeThumbnailPromiseMaker(
|
||||||
filenameBase: string,
|
filenameBase: string,
|
||||||
baseSize: number,
|
baseSize: number,
|
||||||
maxPixelDensity: number,
|
maxPixelDensity: number,
|
||||||
@ -246,13 +272,7 @@ export class TimelineReader {
|
|||||||
for (let i = 1; i <= maxPixelDensity; i++) {
|
for (let i = 1; i <= maxPixelDensity; i++) {
|
||||||
promises.push(
|
promises.push(
|
||||||
...formats.map((f) =>
|
...formats.map((f) =>
|
||||||
TimelineReader.resizeAvatar(
|
this.resizeAvatar(filenameBase, baseSize * i, `${i}x.${f}`, 'thumbnails', sharpAvatar)
|
||||||
filenameBase,
|
|
||||||
baseSize * i,
|
|
||||||
`${i}x.${f}`,
|
|
||||||
'thumbnails',
|
|
||||||
sharpAvatar
|
|
||||||
)
|
|
||||||
.then(
|
.then(
|
||||||
(fn) =>
|
(fn) =>
|
||||||
({
|
({
|
||||||
@ -260,7 +280,7 @@ export class TimelineReader {
|
|||||||
file: fn,
|
file: fn,
|
||||||
sizeDescriptor: `${i}x`,
|
sizeDescriptor: `${i}x`,
|
||||||
kind: kind
|
kind: kind
|
||||||
} as SongThumbnailImage)
|
}) as SongThumbnailImage
|
||||||
)
|
)
|
||||||
.then(saveSongThumbnail)
|
.then(saveSongThumbnail)
|
||||||
)
|
)
|
||||||
@ -269,7 +289,7 @@ export class TimelineReader {
|
|||||||
return promises;
|
return promises;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async saveAvatar(account: Account) {
|
private async saveAvatar(account: Account) {
|
||||||
try {
|
try {
|
||||||
const existingAvatars = await getAvatars(account.url, 1);
|
const existingAvatars = await getAvatars(account.url, 1);
|
||||||
const existingAvatarBase = existingAvatars.shift()?.file.split('/').pop()?.split('_').shift();
|
const existingAvatarBase = existingAvatars.shift()?.file.split('/').pop()?.split('_').shift();
|
||||||
@ -282,7 +302,7 @@ export class TimelineReader {
|
|||||||
const avatarsToDelete = (await fs.readdir('avatars'))
|
const avatarsToDelete = (await fs.readdir('avatars'))
|
||||||
.filter((x) => x.startsWith(existingAvatarBase + '_'))
|
.filter((x) => x.startsWith(existingAvatarBase + '_'))
|
||||||
.map((x) => {
|
.map((x) => {
|
||||||
log.debug('Removing existing avatar file', x);
|
this.logger.debug('Removing existing avatar file', x);
|
||||||
return x;
|
return x;
|
||||||
})
|
})
|
||||||
.map((x) => fs.unlink('avatars/' + x));
|
.map((x) => fs.unlink('avatars/' + x));
|
||||||
@ -291,7 +311,7 @@ export class TimelineReader {
|
|||||||
const avatarResponse = await fetch(account.avatar);
|
const avatarResponse = await fetch(account.avatar);
|
||||||
const avatar = await avatarResponse.arrayBuffer();
|
const avatar = await avatarResponse.arrayBuffer();
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
TimelineReader.resizeAvatarPromiseMaker(
|
this.resizeAvatarPromiseMaker(
|
||||||
avatarFilenameBase,
|
avatarFilenameBase,
|
||||||
50,
|
50,
|
||||||
3,
|
3,
|
||||||
@ -305,7 +325,7 @@ export class TimelineReader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async saveSongThumbnails(songs: SongInfo[]) {
|
private async saveSongThumbnails(songs: SongInfo[]) {
|
||||||
for (const song of songs) {
|
for (const song of songs) {
|
||||||
if (!song.thumbnailUrl) {
|
if (!song.thumbnailUrl) {
|
||||||
continue;
|
continue;
|
||||||
@ -319,7 +339,7 @@ export class TimelineReader {
|
|||||||
const imageResponse = await fetch(song.thumbnailUrl);
|
const imageResponse = await fetch(song.thumbnailUrl);
|
||||||
const avatar = await imageResponse.arrayBuffer();
|
const avatar = await imageResponse.arrayBuffer();
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
TimelineReader.resizeThumbnailPromiseMaker(
|
this.resizeThumbnailPromiseMaker(
|
||||||
fileBaseName + '_large',
|
fileBaseName + '_large',
|
||||||
200,
|
200,
|
||||||
3,
|
3,
|
||||||
@ -330,7 +350,7 @@ export class TimelineReader {
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
TimelineReader.resizeThumbnailPromiseMaker(
|
this.resizeThumbnailPromiseMaker(
|
||||||
fileBaseName + '_small',
|
fileBaseName + '_small',
|
||||||
60,
|
60,
|
||||||
3,
|
3,
|
||||||
@ -351,63 +371,124 @@ export class TimelineReader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private startWebsocket() {
|
private async checkAndSavePost(post: Post) {
|
||||||
const socket = new WebSocket(`wss://${MASTODON_INSTANCE}/api/v1/streaming`);
|
|
||||||
socket.onopen = () => {
|
|
||||||
log.log('Connected to WS');
|
|
||||||
socket.send('{ "type": "subscribe", "stream": "public:local"}');
|
|
||||||
};
|
|
||||||
socket.onmessage = async (event) => {
|
|
||||||
try {
|
|
||||||
const data: TimelineEvent = JSON.parse(event.data.toString());
|
|
||||||
if (data.event !== 'update') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const post: Post = JSON.parse(data.payload);
|
|
||||||
|
|
||||||
const hashttags: string[] = HASHTAG_FILTER.split(',');
|
const hashttags: string[] = HASHTAG_FILTER.split(',');
|
||||||
const found_tags: Tag[] = post.tags.filter((t: Tag) => hashttags.includes(t.name));
|
const found_tags: Tag[] = post.tags.filter((t: Tag) => hashttags.includes(t.name));
|
||||||
|
|
||||||
const songs = await TimelineReader.getSongInfoInPost(post);
|
const songs = await this.getSongInfoInPost(post);
|
||||||
|
|
||||||
// If we don't have any tags or non-youtube urls, check youtube
|
// If we don't have any tags or non-youtube urls, check youtube
|
||||||
// YT is handled separately, because it requires an API call and therefore is slower
|
// YT is handled separately, because it requires an API call and therefore is slower
|
||||||
if (songs.length === 0 && found_tags.length === 0) {
|
if (songs.length === 0 && found_tags.length === 0) {
|
||||||
log.log('Ignoring post', post.url);
|
this.logger.log('Ignoring post', post.url);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
await savePost(post, songs);
|
await savePost(post, songs);
|
||||||
|
|
||||||
await TimelineReader.saveAvatar(post.account);
|
await this.saveAvatar(post.account);
|
||||||
await TimelineReader.saveSongThumbnails(songs);
|
await this.saveSongThumbnails(songs);
|
||||||
|
|
||||||
log.debug('Saved post', post.url);
|
this.logger.debug('Saved post', post.url, 'songs', songs);
|
||||||
|
|
||||||
const posts = await getPosts(null, null, 100);
|
const posts = await getPosts(null, null, 100);
|
||||||
await saveAtomFeed(createFeed(posts));
|
await saveAtomFeed(createFeed(posts));
|
||||||
|
|
||||||
|
for (let song of songs) {
|
||||||
|
this.logger.debug('Adding to playlist', song);
|
||||||
|
await this.addToPlaylist(song);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private startWebsocket() {
|
||||||
|
const socketLogger = new Logger('Websocket');
|
||||||
|
const socket = new WebSocket(
|
||||||
|
`wss://${MASTODON_INSTANCE}/api/v1/streaming?type=subscribe&stream=public:local&access_token=${MASTODON_ACCESS_TOKEN}`
|
||||||
|
);
|
||||||
|
socket.onopen = () => {
|
||||||
|
socketLogger.log('Connected to WS');
|
||||||
|
};
|
||||||
|
socket.onmessage = async (event) => {
|
||||||
|
try {
|
||||||
|
const data: TimelineEvent = JSON.parse(event.data.toString());
|
||||||
|
socketLogger.debug('ES event', data.event);
|
||||||
|
if (data.event !== 'update') {
|
||||||
|
socketLogger.log('Ignoring ES event', data.event);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const post: Post = JSON.parse(data.payload);
|
||||||
|
|
||||||
|
// Sometimes onmessage is called twice for the same post.
|
||||||
|
// This looks to be an issue with automatic reloading in the dev environment,
|
||||||
|
// but hard to tell
|
||||||
|
if (this.lastPosts.includes(post.id)) {
|
||||||
|
socketLogger.log('Skipping post, already handled', post.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.lastPosts.push(post.id);
|
||||||
|
while (this.lastPosts.length > 10) {
|
||||||
|
this.lastPosts.shift();
|
||||||
|
}
|
||||||
|
await this.checkAndSavePost(post);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log.error('error message', event, event.data, e);
|
socketLogger.error('error message', event, event.data, e);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
socket.onclose = (event) => {
|
socket.onclose = (event) => {
|
||||||
log.warn(
|
socketLogger.warn(
|
||||||
`Websocket connection to ${MASTODON_INSTANCE} closed. Code: ${event.code}, reason: '${event.reason}'`
|
`Websocket connection to ${MASTODON_INSTANCE} closed. Code: ${event.code}, reason: '${event.reason}'`,
|
||||||
|
event
|
||||||
);
|
);
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
log.info(`Attempting to reconenct to WS`);
|
socketLogger.info(`Attempting to reconenct to WS`);
|
||||||
this.startWebsocket();
|
this.startWebsocket();
|
||||||
}, 10000);
|
}, 10000);
|
||||||
};
|
};
|
||||||
socket.onerror = (event) => {
|
socket.onerror = (event) => {
|
||||||
log.error(
|
socketLogger.error(
|
||||||
`Websocket connection to ${MASTODON_INSTANCE} failed. ${event.type}: ${event.error}, message: '${event.message}'`
|
`Websocket connection to ${MASTODON_INSTANCE} failed. ${event.type}: ${event.error}, message: '${event.message}'`
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async loadPostsSinceLastRun() {
|
||||||
|
const now = new Date().toISOString();
|
||||||
|
let latestPost = await getPosts(null, now, 1);
|
||||||
|
if (latestPost.length > 0) {
|
||||||
|
this.logger.log('Last post in DB since', now, latestPost[0].created_at);
|
||||||
|
} else {
|
||||||
|
this.logger.log('No posts in DB since');
|
||||||
|
}
|
||||||
|
let u = new URL(`https://${MASTODON_INSTANCE}/api/v1/timelines/public?local=true&limit=40`);
|
||||||
|
if (latestPost.length > 0) {
|
||||||
|
u.searchParams.append('since_id', latestPost[0].id);
|
||||||
|
}
|
||||||
|
for (let tag of HASHTAG_FILTER.split(',')) {
|
||||||
|
u.searchParams.append('q', '#' + tag);
|
||||||
|
}
|
||||||
|
const headers = {
|
||||||
|
Authorization: `Bearer ${MASTODON_ACCESS_TOKEN}`
|
||||||
|
};
|
||||||
|
const latestPosts: Post[] = await fetch(u, { headers }).then((r) => r.json());
|
||||||
|
this.logger.info('searched posts', latestPosts.length);
|
||||||
|
for (const post of latestPosts) {
|
||||||
|
await this.checkAndSavePost(post);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private constructor() {
|
private constructor() {
|
||||||
|
this.logger = new Logger('Timeline');
|
||||||
|
this.logger.log('Constructing timeline object');
|
||||||
|
this.playlistAdders = [new YoutubePlaylistAdder(), new SpotifyPlaylistAdder()];
|
||||||
this.startWebsocket();
|
this.startWebsocket();
|
||||||
|
|
||||||
|
this.loadPostsSinceLastRun()
|
||||||
|
.then((_) => {
|
||||||
|
this.logger.info('loaded posts since last run');
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
this.logger.error('cannot fetch latest posts', e);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public static init() {
|
public static init() {
|
||||||
|
@ -1,6 +1,11 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import FooterComponent from '$lib/components/FooterComponent.svelte';
|
import FooterComponent from '$lib/components/FooterComponent.svelte';
|
||||||
import { SvelteToast } from '@zerodevx/svelte-toast';
|
import { SvelteToast } from '@zerodevx/svelte-toast';
|
||||||
|
interface Props {
|
||||||
|
children?: import('svelte').Snippet;
|
||||||
|
}
|
||||||
|
|
||||||
|
let { children }: Props = $props();
|
||||||
|
|
||||||
const options = {
|
const options = {
|
||||||
pausable: true,
|
pausable: true,
|
||||||
@ -8,7 +13,7 @@
|
|||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<slot />
|
{@render children?.()}
|
||||||
<SvelteToast {options} />
|
<SvelteToast {options} />
|
||||||
<div class="footer">
|
<div class="footer">
|
||||||
<FooterComponent />
|
<FooterComponent />
|
||||||
|
@ -12,7 +12,12 @@
|
|||||||
import { cubicInOut } from 'svelte/easing';
|
import { cubicInOut } from 'svelte/easing';
|
||||||
import { errorToast } from '$lib/errorToast';
|
import { errorToast } from '$lib/errorToast';
|
||||||
|
|
||||||
export let data: PageData;
|
interface Props {
|
||||||
|
data: PageData;
|
||||||
|
}
|
||||||
|
|
||||||
|
let { data = $bindable() }: Props = $props();
|
||||||
|
let posts: Post[] = $state(data.posts);
|
||||||
|
|
||||||
interface FetchOptions {
|
interface FetchOptions {
|
||||||
since?: string;
|
since?: string;
|
||||||
@ -26,8 +31,8 @@
|
|||||||
|
|
||||||
const refreshInterval = parseInt(PUBLIC_REFRESH_INTERVAL);
|
const refreshInterval = parseInt(PUBLIC_REFRESH_INTERVAL);
|
||||||
let interval: ReturnType<typeof setTimeout> | null = null;
|
let interval: ReturnType<typeof setTimeout> | null = null;
|
||||||
let moreOlderPostsAvailable = true;
|
let moreOlderPostsAvailable = $state(true);
|
||||||
let loadingOlderPosts = false;
|
let loadingOlderPosts = $state(false);
|
||||||
|
|
||||||
// Needed, so that edgeFly() can do its thing:
|
// Needed, so that edgeFly() can do its thing:
|
||||||
// To determine whether a newly loaded post is older than the existing ones, is required to know what the oldest
|
// To determine whether a newly loaded post is older than the existing ones, is required to know what the oldest
|
||||||
@ -40,11 +45,11 @@
|
|||||||
*/
|
*/
|
||||||
function edgeFly(node: Element, opts: EdgeFlyParams) {
|
function edgeFly(node: Element, opts: EdgeFlyParams) {
|
||||||
const createdAt = new Date(opts.created_at).getTime();
|
const createdAt = new Date(opts.created_at).getTime();
|
||||||
const diffNewest = Math.abs(new Date(data.posts[0].created_at).getTime() - createdAt);
|
const diffNewest = Math.abs(new Date(posts[0].created_at).getTime() - createdAt);
|
||||||
const oldest =
|
const oldest =
|
||||||
oldestBeforeLastFetch !== null
|
oldestBeforeLastFetch !== null
|
||||||
? oldestBeforeLastFetch
|
? oldestBeforeLastFetch
|
||||||
: new Date(data.posts[data.posts.length - 1].created_at).getTime();
|
: new Date(posts[posts.length - 1].created_at).getTime();
|
||||||
const diffOldest = Math.abs(oldest - createdAt);
|
const diffOldest = Math.abs(oldest - createdAt);
|
||||||
const fromTop = diffNewest <= diffOldest;
|
const fromTop = diffNewest <= diffOldest;
|
||||||
|
|
||||||
@ -79,15 +84,15 @@
|
|||||||
|
|
||||||
function refresh() {
|
function refresh() {
|
||||||
let filter: FetchOptions = {};
|
let filter: FetchOptions = {};
|
||||||
if (data.posts.length > 0) {
|
if (posts.length > 0) {
|
||||||
filter = { since: data.posts[0].created_at };
|
filter = { since: posts[0].created_at };
|
||||||
}
|
}
|
||||||
fetchPosts(filter)
|
fetchPosts(filter)
|
||||||
.then((resp) => {
|
.then((resp) => {
|
||||||
if (resp.length > 0) {
|
if (resp.length > 0) {
|
||||||
// Prepend new posts, filter dupes
|
// Prepend new posts, filter dupes
|
||||||
// There shouldn't be any duplicates, but better be safe than sorry
|
// There shouldn't be any duplicates, but better be safe than sorry
|
||||||
data.posts = filterDuplicates(resp.concat(data.posts));
|
posts = filterDuplicates(resp.concat(posts));
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((e: Error) => {
|
.catch((e: Error) => {
|
||||||
@ -96,8 +101,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(async () => {
|
||||||
if (data.posts.length > 0) {
|
posts = data.posts;
|
||||||
oldestBeforeLastFetch = new Date(data.posts[data.posts.length - 1].created_at).getTime();
|
if (posts.length > 0) {
|
||||||
|
oldestBeforeLastFetch = new Date(posts[posts.length - 1].created_at).getTime();
|
||||||
}
|
}
|
||||||
interval = setInterval(refresh, refreshInterval);
|
interval = setInterval(refresh, refreshInterval);
|
||||||
|
|
||||||
@ -121,8 +127,8 @@
|
|||||||
function loadOlderPosts() {
|
function loadOlderPosts() {
|
||||||
loadingOlderPosts = true;
|
loadingOlderPosts = true;
|
||||||
const filter: FetchOptions = { count: 20 };
|
const filter: FetchOptions = { count: 20 };
|
||||||
if (data.posts.length > 0) {
|
if (posts.length > 0) {
|
||||||
const before = data.posts[data.posts.length - 1].created_at;
|
const before = posts[posts.length - 1].created_at;
|
||||||
filter.before = before;
|
filter.before = before;
|
||||||
oldestBeforeLastFetch = new Date(before).getTime();
|
oldestBeforeLastFetch = new Date(before).getTime();
|
||||||
}
|
}
|
||||||
@ -132,7 +138,7 @@
|
|||||||
if (resp.length > 0) {
|
if (resp.length > 0) {
|
||||||
// Append old posts, filter dupes
|
// Append old posts, filter dupes
|
||||||
// There shouldn't be any duplicates, but better be safe than sorry
|
// There shouldn't be any duplicates, but better be safe than sorry
|
||||||
data.posts = filterDuplicates(data.posts.concat(resp));
|
posts = filterDuplicates(posts.concat(resp));
|
||||||
// If we got less than we expected, there are no older posts available
|
// If we got less than we expected, there are no older posts available
|
||||||
moreOlderPostsAvailable = resp.length >= (filter.count ?? 20);
|
moreOlderPostsAvailable = resp.length >= (filter.count ?? 20);
|
||||||
} else {
|
} else {
|
||||||
@ -152,15 +158,15 @@
|
|||||||
</svelte:head>
|
</svelte:head>
|
||||||
<h2>{PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME} music list</h2>
|
<h2>{PUBLIC_MASTODON_INSTANCE_DISPLAY_NAME} music list</h2>
|
||||||
<div class="wrapper">
|
<div class="wrapper">
|
||||||
<div />
|
<div></div>
|
||||||
<div class="posts">
|
<div class="posts">
|
||||||
{#if data.posts.length === 0}
|
{#if posts.length === 0}
|
||||||
Sorry, no posts recommending music have been found yet
|
Sorry, no posts recommending music have been found yet
|
||||||
{/if}
|
{/if}
|
||||||
{#each data.posts as post (post.url)}
|
{#each posts as post (post.url)}
|
||||||
<div
|
<div
|
||||||
class="post"
|
class="post"
|
||||||
transition:edgeFly={{
|
transition:edgeFly|global={{
|
||||||
y: 10,
|
y: 10,
|
||||||
created_at: post.created_at,
|
created_at: post.created_at,
|
||||||
duration: 300,
|
duration: 300,
|
||||||
@ -171,12 +177,12 @@
|
|||||||
</div>
|
</div>
|
||||||
{/each}
|
{/each}
|
||||||
<LoadMoreComponent
|
<LoadMoreComponent
|
||||||
on:loadOlderPosts={loadOlderPosts}
|
{loadOlderPosts}
|
||||||
moreAvailable={moreOlderPostsAvailable}
|
moreAvailable={moreOlderPostsAvailable}
|
||||||
isLoading={loadingOlderPosts}
|
isLoading={loadingOlderPosts}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div />
|
<div></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
30
src/routes/spotifyAuth/+page.server.ts
Normal file
30
src/routes/spotifyAuth/+page.server.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import { Logger } from '$lib/log';
|
||||||
|
import { SpotifyPlaylistAdder } from '$lib/server/playlist/spotifyPlaylistAdder';
|
||||||
|
import { redirect } from '@sveltejs/kit';
|
||||||
|
import type { PageServerLoad } from './$types';
|
||||||
|
|
||||||
|
const logger = new Logger('SpotifyAuth');
|
||||||
|
|
||||||
|
export const load: PageServerLoad = async ({ url }) => {
|
||||||
|
const adder = new SpotifyPlaylistAdder();
|
||||||
|
let redirectUri = url;
|
||||||
|
if (url.hostname === 'localhost') {
|
||||||
|
redirectUri.hostname = '127.0.0.1';
|
||||||
|
}
|
||||||
|
logger.debug(url.searchParams, url.hostname);
|
||||||
|
if (url.searchParams.has('code')) {
|
||||||
|
await adder.receivedAuthCode(url.searchParams.get('code') || '', url);
|
||||||
|
redirect(307, '/');
|
||||||
|
} else if (url.searchParams.has('error')) {
|
||||||
|
logger.error('received error', url.searchParams.get('error'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (await adder.authCodeExists()) {
|
||||||
|
redirect(307, '/');
|
||||||
|
}
|
||||||
|
|
||||||
|
const authUrl = adder.constructAuthUrl(url);
|
||||||
|
logger.debug('+page.server.ts', authUrl.toString());
|
||||||
|
redirect(307, authUrl);
|
||||||
|
};
|
1
src/routes/spotifyAuth/+page.svelte
Normal file
1
src/routes/spotifyAuth/+page.svelte
Normal file
@ -0,0 +1 @@
|
|||||||
|
<h1>Something went wrong</h1>
|
26
src/routes/ytauth/+page.server.ts
Normal file
26
src/routes/ytauth/+page.server.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import { Logger } from '$lib/log';
|
||||||
|
import { YoutubePlaylistAdder } from '$lib/server/playlist/ytPlaylistAdder';
|
||||||
|
import { redirect } from '@sveltejs/kit';
|
||||||
|
import type { PageServerLoad } from './$types';
|
||||||
|
|
||||||
|
const logger = new Logger('YT Auth');
|
||||||
|
|
||||||
|
export const load: PageServerLoad = async ({ url }) => {
|
||||||
|
const adder = new YoutubePlaylistAdder();
|
||||||
|
if (url.searchParams.has('code')) {
|
||||||
|
logger.debug(url.searchParams);
|
||||||
|
await adder.receivedAuthCode(url.searchParams.get('code') || '', url);
|
||||||
|
redirect(307, '/');
|
||||||
|
} else if (url.searchParams.has('error')) {
|
||||||
|
logger.error('received error', url.searchParams.get('error'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (await adder.authCodeExists()) {
|
||||||
|
redirect(307, '/');
|
||||||
|
}
|
||||||
|
|
||||||
|
const authUrl = adder.constructAuthUrl(url);
|
||||||
|
logger.debug('+page.server.ts', authUrl.toString());
|
||||||
|
redirect(307, authUrl);
|
||||||
|
};
|
1
src/routes/ytauth/+page.svelte
Normal file
1
src/routes/ytauth/+page.svelte
Normal file
@ -0,0 +1 @@
|
|||||||
|
<h1>Something went wrong</h1>
|
@ -1,5 +1,5 @@
|
|||||||
import adapter from '@sveltejs/adapter-node';
|
import adapter from '@sveltejs/adapter-node';
|
||||||
import { vitePreprocess } from '@sveltejs/kit/vite';
|
import { vitePreprocess } from '@sveltejs/vite-plugin-svelte';
|
||||||
|
|
||||||
/** @type {import('@sveltejs/kit').Config} */
|
/** @type {import('@sveltejs/kit').Config} */
|
||||||
const config = {
|
const config = {
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
"strict": true
|
"strict": true
|
||||||
|
//"lib": ["ESNext.Array"]
|
||||||
}
|
}
|
||||||
// Path aliases are handled by https://kit.svelte.dev/docs/configuration#alias
|
// Path aliases are handled by https://kit.svelte.dev/docs/configuration#alias
|
||||||
//
|
//
|
||||||
|
Reference in New Issue
Block a user