automuse/book.ts

237 lines
7.6 KiB
TypeScript
Raw Normal View History

import * as fs from "node:fs/promises";
import { OpenAIApi } from "openai";
import PlotGenerator, { Plot } from "@xeserv/plottoriffic";
export const authorBio = `Yasomi Midori is a science fiction author who explores the themes of identity, memory, and technology in her novels. Her debut novel “The Memory Thief” was a critically acclaimed bestseller that captivated readers with its thrilling plot and complex characters. Yasomi also contributes to the Xe Iaso blog as the character Mimi, a hacker and activist who exposes the secrets of the powerful corporations that control the world. Yasomi was born and raised in Tokyo, Japan, where she developed a passion for reading and writing at an early age. She studied computer science and literature at the University of Tokyo, and worked as a software engineer before becoming a full-time writer. She lives in Kyoto with her husband and two cats.`;
export interface ChapterListItem {
title: string;
summary: string;
}
export interface Chapter extends ChapterListItem {
sceneDescriptions: string[];
}
export interface Character {
name: string;
symbol: string;
role: string;
desc: string;
}
export interface Summary {
title: string;
chapterList: ChapterListItem[];
plotSummary: string;
characters: Character[];
}
export const createPlot = async (dirName: string): Promise<Plot> => {
const pg = new PlotGenerator({ flipGenders: false });
const plot = pg.generate();
await fs.writeFile(`${dirName}/plotto.json`, JSON.stringify(plot));
return plot;
};
export const createAndParseSummary = async (
dirName: string,
openai: OpenAIApi,
plot: Plot
): Promise<Summary> => {
console.log("generating plot summary");
const promptBase = `Write me the following about the following plot summary for a novel:
- A two to five word title for the novel starting with "Title: " and followed by two newlines. For example: "Fresh Beginnings" or "Jared's Adventure through Crime".
- A detailed plot summary for the story starting with "Plot Summary: " and followed by two newlines. The plot summary should be on the same line as the prefix. Adapt the story to be about peer to peer networks somehow.
- The string "Chapter Summaries" followed by two newlines.
- A markdown list of detailed chapter summaries in at least 3 sentences and titles for each of the 10 chapters that a novel based on the plot summary would have. Surround each chapter title in quotes and put a dash after the name like this:
- Chapter name - Chapter summary goes here. More words in the summary go here.
- Second chapter name - Second chapter summary goes here.`;
const summary = await openai.createChatCompletion({
model: "gpt-3.5-turbo",
messages: [
{
role: "user",
content: promptBase + "\n\n" + plot.plot,
},
],
});
if (!!summary.data.usage) {
const usage = summary.data.usage;
console.log(
`${usage.total_tokens} tokens (${usage.prompt_tokens} prompt, ${usage?.completion_tokens} completion)`
);
}
const summaryText = summary.data.choices[0].message?.content;
await fs.writeFile(`${dirName}/summary.txt`, summaryText as string);
const titleRegex = /^Title: (.+)$/gm;
const plotSummaryRegex = /^Plot Summary: (.+)$/gm;
const chapterSummaryRegex = /^- (.+) - (.+)$/gm;
let title = summaryText?.split("\n", 2)[0].split(titleRegex)[1] as string;
if (title[0] === '"') {
title = title.slice(1, -1);
}
console.log(`title: ${title}`);
const chapterList: ChapterListItem[] = summaryText
?.split("\n\n")
.slice(-1)[0]
.split("\n")
.map((line) => {
return line.split(chapterSummaryRegex);
})
.map((ch) => {
ch.shift();
ch.pop();
return {
title: ch[0].slice(1, -1) as string,
summary: ch[1] as string,
} as ChapterListItem;
}) as ChapterListItem[];
chapterList.forEach((ch) => console.log(`chapter ${ch.title}: ${ch.summary}`));
const plotSummary = summaryText?.split("\n\n", 3)[1].split(plotSummaryRegex)[1] as string;
console.log(`${plotSummary}`);
const bookInfo: Summary = {
title,
chapterList,
plotSummary,
characters: [],
};
for (const ch of plot.cast) {
console.log(ch);
const prompt = `Given the following plot summary and character information, write a two-sentence summary of that character and create plausible character details and an appearance for them. Don't include anything but the summary. Adapt the character's story to be about peer to peer networks somehow.
Plot summary: ${plotSummary}
Character name: ${ch.name} (${ch.symbol})
Character role: ${ch.description}`;
console.log(`getting information for ${ch.name} (${ch.symbol} - ${ch.description})`);
const chInfo = await openai.createChatCompletion({
model: "gpt-3.5-turbo",
messages: [
{
role: "user",
content: prompt,
},
],
});
const chInfoText = chInfo.data.choices[0].message?.content;
console.log(`${ch.name}: ${chInfoText}`);
bookInfo.characters.push({
name: ch.name,
symbol: ch.symbol,
role: ch.description,
desc: chInfoText as string,
});
}
await fs.writeFile(`${dirName}/summary.json`, JSON.stringify(bookInfo));
return bookInfo;
};
export const createChapterScenes = async (
dirName: string,
openai: OpenAIApi,
summary: Summary,
ch: ChapterListItem
): Promise<Chapter> => {
console.log(`creating chapter scene information for chapter ${ch.title}`);
const prompt =
`Given the following plot summary, character information, and chapter information, write descriptions of scenes that would happen in that chapter. End each description with two newlines. Write at least 4 scenes. DO NOT only write one scene. Use detail and be creative. DO NOT include the chapter title in your output. ONLY output the scenes separated by newlines like this.
What happens first.
What happens after that.
Plot summary: ${summary.plotSummary}
Character information:
` +
summary.characters.map((char) => `- ${char.name}: ${char.role}`).join("\n") +
`
Chapter title: ${ch.title}
Chapter summary: ${ch.summary}`;
const chInfo = await openai.createChatCompletion({
model: "gpt-3.5-turbo",
messages: [
{
role: "user",
content: prompt,
},
],
});
const chInfoText = chInfo.data.choices[0].message?.content as string;
return {
title: ch.title,
summary: ch.summary,
sceneDescriptions: chInfoText.split("\n\n"),
};
};
// https://pandoc.org/epub.html
export const writeChapterScene = async (
dirName: string,
openai: OpenAIApi,
summary: Summary,
ch: Chapter,
chNum: number,
sceneNum: number,
scene: string
): Promise<string> => {
const prompt =
`Given the following information, write the scene of the novel. Be detailed about the setting and character descriptions. End each paragraph with two newlines. Write many sentences. ONLY return the text of the novel.
` +
summary.characters.map((char) => `- ${char.name}: ${char.role}`).join("\n") +
`
Chapter title: ${ch.title}
Chapter summary: ${ch.summary}
Scene summary: ${scene} ${
chNum == 1 && sceneNum == 1
? "\nWrite details about what the character in the scene and their environment looks like."
: ""
}`;
const sceneInfo = await openai.createChatCompletion({
model: "gpt-3.5-turbo",
messages: [
{
role: "user",
content: prompt,
},
],
});
const fname = `ch-${chNum}-sc-${sceneNum}.md`;
await fs.writeFile(
`${dirName}/src/${fname}`,
sceneInfo.data.choices[0].message?.content as string
);
console.log(`wrote chapter ${chNum} scene ${sceneNum}`);
return fname;
};