Example Project
Description
We created an example project that uses Dandy to generate a book using most of the examples in this tutorial.
For more information and the full context of the provided example, please check out the Example Project on GitHub to view and understand the code structure in more detail.
Code Snippets
Main
example/main.py
import logging
import os
from book.intelligence.workflow import BookWorkflow
from dandy.cache import SqliteCache
from dandy.core.exceptions import DandyException
from dandy.recorder import Recorder
if __name__ == '__main__':
try:
# if os.getenv('DEBUG'):
# SqliteCache.clear('example')
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s - %(levelname)s - %(message)s',
datefmt='%Y-%m-%d %H:%M:%S',
)
Recorder.start_recording('book_generation')
logging.info('Creating a Book')
dandy_book = BookWorkflow.process(
user_input="""It's 2035 and the dandy intelligence python library has changed the world for ever. In an effort
to build awareness the people behind the library have created a working version of their lovable mascot the
dandy robot. He has now been turned on and given the directive to let the world know about the dandy library
and given full agency to do it ... what could go wrong!""",
)
dandy_book.to_markdown_file()
logging.info('Complete')
except DandyException:
raise DandyException
finally:
Recorder.stop_recording('book_generation')
Recorder.to_html_file('book_generation')
Intel
example/book/intelligence/intel.py
from typing import Union
from dandy.intel import BaseIntel
from example.book.enums import BookTheme
from example.book.intelligence.chapter.intel import ChaptersIntel
from example.book.intelligence.character.intel import CharactersIntel
from example.book.intelligence.plot.intel import PlotIntel
from example.book.intelligence.world.intel import WorldIntel
class BookStartIntel(BaseIntel):
title: str
overview: str
class BookIntel(BaseIntel):
user_input: str
theme: BookTheme
start: Union[BookStartIntel, None] = None
characters: Union[CharactersIntel, None] = None
plot: Union[PlotIntel, None] = None
world: Union[WorldIntel, None] = None
chapters: Union[ChaptersIntel, None] = None
Prompt
example/book/intelligence/prompts.py
from __future__ import annotations
from typing_extensions import TYPE_CHECKING
from dandy.llm import Prompt
from example.book.intelligence.character.prompts import characters_intel_prompt
from example.book.intelligence.plot.prompts import plot_intel_prompt
if TYPE_CHECKING:
from example.book.intelligence.intel import BookIntel
def book_intel_prompt(book_intel: BookIntel) -> Prompt:
prompt = Prompt()
if book_intel.start:
prompt.title(f'Title: {book_intel.start.title}')
prompt.heading(f'Overview: {book_intel.start.overview}')
prompt.line_break()
if book_intel.world:
prompt.heading('World:')
prompt.sub_heading(f'Name: {book_intel.world.name}')
prompt.text(label='Description', text=book_intel.world.description)
prompt.line_break()
prompt.heading('Locations:')
for location_intel in book_intel.world.locations:
prompt.line_break()
prompt.sub_heading(f'Name: {location_intel.name}')
prompt.text(label='Description', text=location_intel.description)
prompt.line_break()
if book_intel.characters:
prompt.prompt(characters_intel_prompt(book_intel.characters))
prompt.line_break()
if book_intel.plot:
prompt.prompt(plot_intel_prompt(book_intel.plot))
return prompt
LLMBot
example/book/intelligence/bots/book_start_llm_bot.py
from dandy.cache import cache_to_sqlite
from dandy.llm import BaseLlmBot, Prompt
from example.book.intelligence.intel import BookStartIntel
class BookStartLlmBot(BaseLlmBot):
instructions_prompt = (
Prompt()
.text('You are a book starting bot. You will be given an idea by the user.')
.text('you will generate a book title and overview.')
)
@classmethod
@cache_to_sqlite('example')
def process(
cls,
user_input: str,
) -> BookStartIntel:
return cls.process_prompt_to_intel(
prompt=Prompt(user_input),
intel_class=BookStartIntel
)
Workflow
example/book/intelligence/workflow.py
from __future__ import annotations
import logging
from typing_extensions import TYPE_CHECKING
from dandy.workflow import BaseWorkflow
from example.book.intelligence.bots import BookStartLlmBot
from example.book.intelligence.chapter.workflow import ChaptersWorkflow
from example.book.intelligence.character.workflow import CharactersWorkflow
from example.book.intelligence.intel import BookIntel
from example.book.intelligence.maps import BookThemeLlmMap
from example.book.intelligence.plot.workflow import PlotWorkflow
from example.book.intelligence.world.bot import WorldLlmBot
from example.book.models import Book, Chapter
if TYPE_CHECKING:
pass
class BookWorkflow(BaseWorkflow):
@classmethod
def process(
cls,
user_input: str,
) -> Book:
book_intel = BookIntel(
user_input=user_input,
theme=BookThemeLlmMap.process(user_input)[0]
)
logging.info('Working on book title and overview')
book_intel.start = BookStartLlmBot.process(user_input)
logging.info('Creating "those" characters')
book_intel.characters = CharactersWorkflow.process(book_intel)
logging.info('Forging a world')
book_intel.world = WorldLlmBot.process(book_intel)
logging.info('Setting up the plot')
book_intel.plot = PlotWorkflow.process(book_intel)
logging.info('Writing the chapters for all to read')
book_intel.chapters = ChaptersWorkflow.process(
book_intel=book_intel,
chapter_count=4
)
new_book = Book(
title=book_intel.start.title,
author='Dandy McAuthor',
theme=book_intel.theme,
overview=book_intel.start.overview,
chapters=[
Chapter(title=chapter_intel.title, content=chapter_intel.content)
for chapter_intel in book_intel.chapters
]
)
print(new_book.model_dump_json(indent=4))
print(book_intel.model_dump_json(indent=4))
return new_book