feat: initial commit

This commit is contained in:
oxypomme
2025-12-30 19:52:07 +01:00
commit 6514dc165b
10 changed files with 372 additions and 0 deletions

4
.dockerignore Normal file
View File

@@ -0,0 +1,4 @@
.zed/
.prettierc.yml
.oxlintrc.json
.env

1
.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
.env

143
.oxlintrc.json Normal file
View File

@@ -0,0 +1,143 @@
{
"$schema": "./node_modules/oxlint/configuration_schema.json",
"plugins": ["oxc", "unicorn", "typescript", "import", "node", "promise"],
"categories": {
"correctness": "error",
"suspicious": "error",
"pedantic": "warn",
"perf": "warn"
},
"rules": {
// It's a TypeScript project, it should use ECMA imports
"import/no-amd": "error",
"import/no-commonjs": "error",
// Restrict dangerous practices
"eslint/eqeqeq": ["error", "always", { "null": "ignore" }],
"default-case": "error",
"no-bitwise": "error",
"no-console": "off",
"no-empty": "error",
"no-empty-function": "error",
"no-iterator": "error",
"no-plusplus": "error",
"no-proto": "error",
"no-regex-spaces": "error",
"no-restricted-globals": "error",
"no-unused-expressions": "error",
"no-var": "error",
"no-void": "error",
"no-new-require": "error",
"no-const-enum": "error",
"import/no-cycle": "error",
"import/no-dynamic-require": "error",
"promise/catch-or-return": "error",
"promise/spec-only": "error",
"typescript/no-dynamic-delete": "error",
"typescript/no-empty-object-type": "error",
"typescript/no-explicit-any": "error",
"unicorn/no-abusive-eslint-disable": "error",
"unicorn/no-length-as-slice-end": "error",
"unicorn/prefer-number-properties": "error",
"typescript/no-import-type-side-effects": "error",
"unicorn/prefer-event-target": "off",
"unicorn/prefer-add-event-listener": "off",
// Enforce browser rules
"no-alert": "error",
"no-script-url": "error",
"unicorn/no-document-cookie": "error",
// Enforce styling
"arrow-body-style": ["error", "as-needed"],
"curly": "error",
"default-case-last": "error",
"default-param-last": "error",
"func-name": ["error", "as-needed"],
"func-style": ["error", "declaration", { "allowArrowFunctions": true }],
"grouped-accessor-pair": ["error", "getBeforeSet"],
"id-length": ["error", { "exceptions": ["_", "t", "z"] }],
"max-params": "warn",
"new-cap": ["error", { "newIsCap": true, "capIsNew": true }],
"no-duplicate-import": "error",
"no-extra-label": "error",
"no-label-var": "error",
"no-labels": "error",
"no-lone-blocks": "error",
"no-magic-number": [
"warn",
{
"ignore": [0, 1, 2],
"ignoreDefaultValues": true,
"ignoreClassFieldInitialValues": true,
"ignoreEnums": true,
"ignoreNumericLiteralTypes": true
}
],
"no-multi-assign": "error",
"no-nested-ternary": "error",
"no-new-func": "error",
"no-return-assign": "error",
"no-template-curly-in-string": "error",
"operator-assignment": ["error", "always"],
"prefer-exponentiation-operator": "error",
"prefer-object-spread": "error",
"prefer-promise-reject-errors": "error",
"prefer-rest-params": "error",
"prefer-spread": "error",
"import/first": ["error", "absolute-first"],
"import/no-anonymous-default-export": "error",
"import/no-duplicates": "error",
"import/no-mutable-exports": "warn",
"import/no-named-default": "error",
"import/no-namespace": "warn",
"import/no-default-export": "warn", // TODO: move to error once migrated
"promise/avoid-new": "warn", // Maybe error ?
"promise/no-nesting": "error",
"promise/no-return-wrap": "error",
"promise/param-names": "error",
"promise/prefer-await-to-callbacks": "warn", // Maybe error ?
"promise/prefer-await-to-then": "error", // Maybe error ?
"promise/prefer-catch": "error",
"typescript/adjacent-overload-signatures": "error",
"typescript/array-type": ["error", { "default": "array" }],
"typescript/consistent-generic-constructors": "error",
"typescript/consistent-indexed-object-style": ["error", "record"],
"typescript/consistent-type-definitions": ["error", "type"],
"typescript/consistent-type-imports": "error",
"typescript/no-empty-interface": "error",
"typescript/no-inferrable-types": "warn", // Maybe error ?
"typescript/prefer-for-of": "error",
"typescript/prefer-function-type": "error",
"typescript/no-non-null-asserted-nullish-coalescing": "error",
"unicorn/error-message": "error",
"unicorn/no-array-method-this-argument": "error",
"unicorn/no-await-expression-member": "error",
"unicorn/no-console-spaces": "error",
"unicorn/no-zero-fractions": "error",
"unicorn/number-literal-case": "error",
"unicorn/prefer-array-flat-map": "error",
"unicorn/prefer-logical-operator-over-ternary": "error",
"unicorn/no-array-for-each": "error",
"unicorn/no-array-reduce": "warn",
"unicorn/no-magic-array-flat-depth": "error",
"unicorn/no-process-exit": "error",
"unicorn/prefer-modern-math-apis": "error",
"unicorn/prefer-node-protocol": "error",
"typescript/explicit-function-return-type": "warn"
},
"overrides": [
{
"files": ["src/composables/**/*.ts", "src/utils/**/*.ts", "*.config.ts"],
"rules": {
"no-default-export": "off"
}
},
{
"files": ["src/stores/**/*.ts", "src/composables/**/*.ts"],
"rules": {
"max-lines-per-function": "off"
}
}
]
}

4
.prettierrc.yml Normal file
View File

@@ -0,0 +1,4 @@
trailingComma: es5
tabWidth: 2
semi: true
singleQuote: true

23
.zed/settings.json Normal file
View File

@@ -0,0 +1,23 @@
{
"lsp": {
"deno": {
"settings": {
"deno": {
"enable": true
}
}
}
},
"languages": {
"TypeScript": {
"language_servers": [
"deno",
"!typescript-language-server",
"!vtsls",
"!eslint",
"..."
],
"formatter": "prettier"
}
}
}

13
Dockerfile Normal file
View File

@@ -0,0 +1,13 @@
FROM denoland/deno:2.6.3-alpine
# Create working directory
WORKDIR /app
# Copy source
COPY . .
# Install dependencies
RUN deno install
# Prepare flags to run script
ENTRYPOINT ["deno", "run", "-P"]

76
bin/reorder-all-series.ts Normal file
View File

@@ -0,0 +1,76 @@
import {
fetchLibraryMangas,
fetchMangaChapters,
updateMetadata,
} from '~/lib/komga.ts';
const getChapterSort = (chapter: any) => {
const volumeNumber = /vol(?:ume)?.?([0-9]+)/i.exec(chapter.name)?.[1] || '1';
const chapterNumber =
/ch(?:apter)?.?([0-9.]+)/i.exec(chapter.name)?.[1] || '';
const chapterFloat = Number.parseFloat(chapterNumber);
if (!chapterNumber || Number.isNaN(chapterFloat)) {
throw new Error(
`Ch. ${chapterNumber} cannot be parsed from ${chapter.name}`
);
}
return {
title: `Ch. ${chapterNumber} - Vol. ${volumeNumber}`,
numberSort: chapterFloat,
};
};
const sanitiseMetadata = (chapter: any, chaptersSeen: Set<number>) => {
const data = getChapterSort(chapter);
const isDuplicate = chaptersSeen.has(data.numberSort);
if (isDuplicate) {
console.log(`⚠️ Ch. ${data.numberSort} already seen, marking as duplicate`);
}
chaptersSeen.add(data.numberSort);
return {
...data,
numberSortLock: true,
tags: isDuplicate
? [...chapter.metadata.tags, 'oxy.DUPLICATE']
: chapter.metadata.tags.filter((tag: string) => tag !== 'oxy.DUPLICATE'),
};
};
// Main
console.log(
`== Sorting library ${Deno.env.get('LIBRARY_ID')} (${new Date().toISOString()}) ==`
);
for (const manga of await fetchLibraryMangas(Deno.env.get('LIBRARY_ID'))) {
console.log(`Sorting ${manga.name} (${manga.booksCount} chapters)...`);
console.group();
const chapters = await fetchMangaChapters(manga.id);
const chaptersSeen = new Set<number>();
const metadata = new Map(
chapters.map((chap) => [chap.id, sanitiseMetadata(chap, chaptersSeen)])
);
const updates = [...metadata].toSorted(
([, a], [, b]) => a.numberSort - b.numberSort
);
let updateCount = 0;
for (const [id, metadata] of updates) {
await updateMetadata(id, {
...metadata,
number: updateCount,
});
updateCount += 1;
}
console.log(`✔ Updated ${updateCount} chapters !`);
console.groupEnd();
}

19
deno.json Normal file
View File

@@ -0,0 +1,19 @@
{
"$schema": "https://raw.githubusercontent.com/denoland/deno/refs/heads/main/cli/schemas/config-file.v1.json",
"tasks": {
"format": "deno x prettier --write .",
"lint": "deno x oxlint --fix"
},
"imports": {
"~": "./",
"ofetch": "npm:ofetch@^1.5.1"
},
"permissions": {
"default": {
"env": true,
"net": true
}
}
}

30
deno.lock generated Normal file
View File

@@ -0,0 +1,30 @@
{
"version": "5",
"specifiers": {
"npm:ofetch@^1.5.1": "1.5.1"
},
"npm": {
"destr@2.0.5": {
"integrity": "sha512-ugFTXCtDZunbzasqBxrK93Ik/DRYsO6S/fedkWEMKqt04xZ4csmnmwGDBAb07QWNaGMAmnTIemsYZCksjATwsA=="
},
"node-fetch-native@1.6.7": {
"integrity": "sha512-g9yhqoedzIUm0nTnTqAQvueMPVOuIY16bqgAJJC8XOOubYFNwz6IER9qs0Gq2Xd0+CecCKFjtdDTMA4u4xG06Q=="
},
"ofetch@1.5.1": {
"integrity": "sha512-2W4oUZlVaqAPAil6FUg/difl6YhqhUR7x2eZY4bQCko22UXg3hptq9KLQdqFClV+Wu85UX7hNtdGTngi/1BxcA==",
"dependencies": [
"destr",
"node-fetch-native",
"ufo"
]
},
"ufo@1.6.1": {
"integrity": "sha512-9a4/uxlTWJ4+a5i0ooc1rU7C7YOw3wT+UGqdeNNHWnOF9qcMBgLRS+4IYUqbczewFx4mLEig6gawh7X6mFlEkA=="
}
},
"workspace": {
"dependencies": [
"npm:ofetch@^1.5.1"
]
}
}

59
lib/komga.ts Normal file
View File

@@ -0,0 +1,59 @@
import { ofetch } from 'ofetch';
export const komga = ofetch.create({
baseURL: new URL('/api', Deno.env.get('BASE_URL')!).href,
headers: {
'X-API-Key': Deno.env.get('API_KEY')!,
},
});
export const fetchLibraryMangas = async (
libraryId: string
): Promise<Record<string, any>[]> => {
const { content } = await komga('/v1/series/list', {
method: 'POST',
query: {
unpaged: true,
},
body: {
condition: {
libraryId: {
operator: 'is',
value: libraryId,
},
},
},
});
return content;
};
export const fetchMangaChapters = async (
seriesId: string
): Promise<Record<string, any>[]> => {
const { content } = await komga('/v1/books/list', {
method: 'POST',
query: {
unpaged: true,
},
body: {
condition: {
seriesId: {
operator: 'is',
value: seriesId,
},
},
},
});
return content;
};
export const updateMetadata = (
id: string,
body: Record<string, unknown>
): Promise<void> =>
komga(`/v1/books/${id}/metadata`, {
method: 'PATCH',
body,
});