gadl/library.ts

116 lines
3.2 KiB
TypeScript
Raw Normal View History

2023-11-20 01:37:55 +09:00
import fs from "node:fs/promises";
import { createWriteStream } from "node:fs";
import stream from "node:stream/promises";
import { Zip, ZipPassThrough } from "fflate";
2023-11-19 02:25:32 +09:00
import { Database } from "./database";
export type Book = {
id: number;
platform: "dmm-books" | "google-play-books";
readerUrl: string;
2023-11-20 22:56:09 +09:00
title: string;
authors: Array<string>;
2023-11-19 02:25:32 +09:00
};
export function createLibrary(db: Database) {
return {
2023-11-20 22:56:09 +09:00
async add(readerUrlOrBook: string | Book) {
2023-11-19 02:25:32 +09:00
const platform = "dmm-books";
2023-11-20 22:56:09 +09:00
if (typeof readerUrlOrBook === "string") {
await db.run(
`insert into books(platform_id, reader_url) values((select id from platforms where name = ?), ?)`,
platform,
readerUrlOrBook,
);
return;
}
2023-11-19 02:25:32 +09:00
await db.run(
2023-11-20 22:56:09 +09:00
`\
insert into books(
platform_id,
reader_url,
title,
authors)
values((select id from platforms where name = ?), ?, ?, ?)
on conflict(reader_url)
do update set title = excluded.title, authors = excluded.authors
`,
2023-11-19 02:25:32 +09:00
platform,
2023-11-20 22:56:09 +09:00
readerUrlOrBook.readerUrl,
readerUrlOrBook.title,
JSON.stringify(readerUrlOrBook.authors),
2023-11-19 02:25:32 +09:00
);
2023-11-19 20:51:29 +09:00
},
async delete(id: number) {
await db.run(`delete from books where id = ?`, id);
2023-11-19 02:25:32 +09:00
},
2023-11-19 22:04:19 +09:00
async get(id: number): Promise<Book | undefined> {
2023-11-20 22:56:09 +09:00
const row = await db.get(
`select books.id, platforms.name as platform, books.reader_url as readerUrl, books.title, books.authors from books left join platforms on books.platform_id = platforms.id where books.id = ?`,
2023-11-19 22:04:19 +09:00
id,
);
2023-11-20 22:56:09 +09:00
const book: Book | undefined = row && {
...row,
authors: JSON.parse(row.authors),
};
2023-11-19 22:04:19 +09:00
return book;
},
2023-11-19 02:25:32 +09:00
async getBooks(): Promise<Array<Book>> {
2023-11-20 22:56:09 +09:00
const rows = await db.all(
`select books.id, platforms.name as platform, books.reader_url as readerUrl, books.title, books.authors from books left join platforms on books.platform_id = platforms.id`,
2023-11-19 02:25:32 +09:00
);
2023-11-20 22:56:09 +09:00
const books: Array<Book> = rows.map((row) => ({
...row,
authors: JSON.parse(row.authors),
}));
2023-11-19 02:25:32 +09:00
return books;
},
2023-11-20 01:37:55 +09:00
async archive(dir: string) {
const bookDirs = await fs.readdir(dir, { withFileTypes: true });
for (const bookDir of bookDirs) {
if (!bookDir.isDirectory()) {
continue;
}
const path = `${bookDir.path}/${bookDir.name}`;
2023-11-23 01:31:27 +09:00
const book = await this.get(bookDir.name);
const title = book
? `${book.authors.join("、")}${book.title}`.replace(/[/]/g, "%2F")
: bookDir.name;
const out = createWriteStream(`${dir}/${title}.cbz`);
2023-11-20 01:37:55 +09:00
const zip = new Zip(function cb(err, data, final) {
if (err) {
out.destroy(err);
return;
}
out[final ? "end" : "write"](data);
});
const files = await fs.readdir(path);
for (const file of files) {
const data = new ZipPassThrough(file);
zip.add(data);
const buffer = await fs.readFile(`${path}/${file}`);
data.push(buffer, true);
}
zip.end();
await stream.finished(out);
await fs.rm(path, { recursive: true });
}
},
2023-11-19 02:25:32 +09:00
};
}