Skip to content

Commit 53adbf0

Browse files
committed
refactor: follow up on example structure
1 parent 7cb6f6d commit 53adbf0

28 files changed

+70
-85
lines changed

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,7 @@ docker compose exec app bin/console app:blog:embed -vv
7979
Now you should be able to run the test command and get some results:
8080

8181
```shell
82-
docker compose exec app bin/console app:chroma:test
82+
docker compose exec app bin/console app:blog:query
8383
```
8484

8585
**Don't forget to set up the project in your favorite IDE or editor.**
@@ -89,4 +89,4 @@ docker compose exec app bin/console app:chroma:test
8989
* The chatbot application is a simple and small Symfony 7.2 application.
9090
* The UI is coupled to a [Twig LiveComponent](https://symfony.com/bundles/ux-live-component/current/index.html), that integrates different `Chat` implementations on top of the user's session.
9191
* You can reset the chat context by hitting the `Reset` button in the top right corner.
92-
* You find three different usage scenarios in the upper navbar.
92+
* You find three different usage scenarios in the upper navbar.

assets/icons/entypo/chat.svg

-1
This file was deleted.

assets/icons/mdi/symfony.svg

+1
Loading

assets/icons/mingcute/ai-fill.svg

-1
This file was deleted.

assets/styles/app.css

+7-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
body {
22
min-height: 100vh;
3-
&.rag, .rag .card-img-top {
3+
&.blog, .blog .card-img-top {
44
background: rgb(220,139,110);
55
background: linear-gradient(0deg, rgba(220,139,110,1) 0%, rgba(244,233,115,1) 100%);
66
}
@@ -53,7 +53,7 @@ body {
5353
border-radius: 10px 10px 0 10px;
5454
color: #292929;
5555

56-
.rag & {
56+
.blog & {
5757
background: #f4e973;
5858
}
5959

@@ -70,7 +70,7 @@ body {
7070
.bot-message {
7171
border-radius: 10px 10px 10px 0;
7272

73-
.rag & {
73+
.blog & {
7474
background: #dc8b6e;
7575

7676
a {
@@ -103,12 +103,12 @@ body {
103103
height: 50px;
104104
border: 2px solid white;
105105

106-
.rag &.bot {
106+
.blog &.bot {
107107
outline: 1px solid #ffdacc;
108108
background: #ffdacc;
109109
}
110110

111-
.rag &.user {
111+
.blog &.user {
112112
outline: 1px solid #fffad1;
113113
background: #fffad1;
114114
}
@@ -141,7 +141,7 @@ body {
141141

142142
#welcome {
143143
h4 {
144-
.rag & {
144+
.blog & {
145145
color: #f97b62;
146146
}
147147

@@ -152,7 +152,7 @@ body {
152152
}
153153

154154
#chat-reset, #chat-submit {
155-
.rag &:hover {
155+
.blog &:hover {
156156
background: #f97b62;
157157
border-color: #f97b62;
158158
}

config/packages/llm_chain.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ llm_chain:
1616
openai:
1717
api_key: '%env(OPENAI_API_KEY)%'
1818
chain:
19-
rag:
19+
blog:
2020
# platform: 'llm_chain.platform.anthropic'
2121
model:
2222
name: 'GPT'

config/routes.yaml

+5-5
Original file line numberDiff line numberDiff line change
@@ -4,20 +4,20 @@ index:
44
defaults:
55
template: 'index.html.twig'
66

7-
rag:
8-
path: '/rag'
7+
blog:
8+
path: '/blog'
99
controller: 'Symfony\Bundle\FrameworkBundle\Controller\TemplateController'
1010
defaults:
11-
template: 'chat/rag.html.twig'
11+
template: 'blog/chat.html.twig'
1212

1313
youtube:
1414
path: '/youtube'
1515
controller: 'Symfony\Bundle\FrameworkBundle\Controller\TemplateController'
1616
defaults:
17-
template: 'chat/youtube.html.twig'
17+
template: 'youtube/chat.html.twig'
1818

1919
wikipedia:
2020
path: '/wikipedia'
2121
controller: 'Symfony\Bundle\FrameworkBundle\Controller\TemplateController'
2222
defaults:
23-
template: 'chat/wikipedia.html.twig'
23+
template: 'wikipedia/chat.html.twig'

demo.png

11.8 KB
Loading

src/Blog/Chat/Blog.php src/Blog/Chat.php

+4-4
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
declare(strict_types=1);
44

5-
namespace App\Blog\Chat;
5+
namespace App\Blog;
66

77
use PhpLlm\LlmChain\ChainInterface;
88
use PhpLlm\LlmChain\Model\Message\Message;
@@ -11,13 +11,13 @@
1111
use Symfony\Component\DependencyInjection\Attribute\Autowire;
1212
use Symfony\Component\HttpFoundation\RequestStack;
1313

14-
final class Blog
14+
final class Chat
1515
{
16-
private const SESSION_KEY = 'rag-chat';
16+
private const SESSION_KEY = 'blog-chat';
1717

1818
public function __construct(
1919
private readonly RequestStack $requestStack,
20-
#[Autowire(service: 'llm_chain.chain.rag')]
20+
#[Autowire(service: 'llm_chain.chain.blog')]
2121
private readonly ChainInterface $chain,
2222
) {
2323
}

src/Blog/Command/BlogEmbedCommand.php src/Blog/Command/EmbedCommand.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212
use Symfony\Component\Console\Style\SymfonyStyle;
1313

1414
#[AsCommand('app:blog:embed', description: 'Create embeddings for Symfony blog and push to ChromaDB.')]
15-
final class BlogEmbedCommand extends Command
15+
final class EmbedCommand extends Command
1616
{
1717
public function __construct(
1818
private readonly Embedder $embedder,

src/Command/ChromaTestCommand.php src/Blog/Command/QueryCommand.php

+12-22
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
declare(strict_types=1);
44

5-
namespace App\Command;
5+
namespace App\Blog\Command;
66

77
use Codewithkyrian\ChromaDB\Client;
88
use PhpLlm\LlmChain\Bridge\OpenAI\Embeddings;
@@ -15,8 +15,8 @@
1515
use Symfony\Component\Console\Output\OutputInterface;
1616
use Symfony\Component\Console\Style\SymfonyStyle;
1717

18-
#[AsCommand('app:chroma:test', description: 'Testing Chroma DB connection.')]
19-
final class ChromaTestCommand extends Command
18+
#[AsCommand('app:blog:query', description: 'Test command for querying the blog collection in Chroma DB.')]
19+
final class QueryCommand extends Command
2020
{
2121
public function __construct(
2222
private readonly Client $chromaClient,
@@ -31,23 +31,19 @@ protected function execute(InputInterface $input, OutputInterface $output): int
3131
$io->title('Testing Chroma DB Connection');
3232

3333
$io->comment('Connecting to Chroma DB ...');
34-
35-
// Check current ChromaDB version
36-
$version = $this->chromaClient->version();
37-
38-
// Get WSC Collection
3934
$collection = $this->chromaClient->getOrCreateCollection('symfony_blog');
40-
4135
$io->table(['Key', 'Value'], [
42-
['ChromaDB Version', $version],
36+
['ChromaDB Version', $this->chromaClient->version()],
4337
['Collection Name', $collection->name],
4438
['Collection ID', $collection->id],
4539
['Total Documents', $collection->count()],
4640
]);
4741

48-
$io->comment('Searching for content about "New Symfony Features" ...');
42+
$search = $io->ask('What do you want to know about?', 'New Symfony Features');
43+
$io->comment(sprintf('Converting "%s" to vector & searching in Chroma DB ...', $search));
44+
$io->comment('Results are limited to 4 most similar documents.');
4945

50-
$platformResponse = $this->platform->request(new Embeddings(), 'New Symfony Features');
46+
$platformResponse = $this->platform->request(new Embeddings(), $search);
5147
assert($platformResponse instanceof AsyncResponse);
5248
$platformResponse = $platformResponse->unwrap();
5349
assert($platformResponse instanceof VectorResponse);
@@ -62,16 +58,10 @@ protected function execute(InputInterface $input, OutputInterface $output): int
6258
return Command::FAILURE;
6359
}
6460

65-
$io->table(['ID', 'Title'], [
66-
/* @phpstan-ignore-next-line */
67-
[$queryResponse->ids[0][0], $queryResponse->metadatas[0][0]['title']],
68-
/* @phpstan-ignore-next-line */
69-
[$queryResponse->ids[0][1], $queryResponse->metadatas[0][1]['title']],
70-
/* @phpstan-ignore-next-line */
71-
[$queryResponse->ids[0][2], $queryResponse->metadatas[0][2]['title']],
72-
/* @phpstan-ignore-next-line */
73-
[$queryResponse->ids[0][3], $queryResponse->metadatas[0][3]['title']],
74-
]);
61+
foreach ($queryResponse->ids[0] as $i => $id) {
62+
$io->section($queryResponse->metadatas[0][$i]['title']);
63+
$io->block($queryResponse->metadatas[0][$i]['description']);
64+
}
7565

7666
$io->success('Chroma DB Connection & Similarity Search Test Successful!');
7767

src/Blog/Embedder.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
final readonly class Embedder
1212
{
1313
public function __construct(
14-
private Loader $loader,
14+
private FeedLoader $loader,
1515
private LlmChainEmbedder $embedder,
1616
) {
1717
}

src/Blog/Loader.php src/Blog/FeedLoader.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
use Symfony\Component\Uid\Uuid;
99
use Symfony\Contracts\HttpClient\HttpClientInterface;
1010

11-
class Loader
11+
class FeedLoader
1212
{
1313
public function __construct(
1414
private HttpClientInterface $httpClient,

src/Blog/Twig/BlogComponent.php src/Blog/TwigComponent.php

+4-5
Original file line numberDiff line numberDiff line change
@@ -2,22 +2,21 @@
22

33
declare(strict_types=1);
44

5-
namespace App\Blog\Twig;
5+
namespace App\Blog;
66

7-
use App\Blog\Chat\Blog;
87
use PhpLlm\LlmChain\Model\Message\MessageBag;
98
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
109
use Symfony\UX\LiveComponent\Attribute\LiveAction;
1110
use Symfony\UX\LiveComponent\Attribute\LiveArg;
1211
use Symfony\UX\LiveComponent\DefaultActionTrait;
1312

14-
#[AsLiveComponent('rag')]
15-
final class BlogComponent
13+
#[AsLiveComponent('blog', template: 'blog/component.html.twig')]
14+
final class TwigComponent
1615
{
1716
use DefaultActionTrait;
1817

1918
public function __construct(
20-
private readonly Blog $chat,
19+
private readonly Chat $chat,
2120
) {
2221
}
2322

src/Wikipedia/Chat/Wikipedia.php src/Wikipedia/Chat.php

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
declare(strict_types=1);
44

5-
namespace App\Wikipedia\Chat;
5+
namespace App\Wikipedia;
66

77
use PhpLlm\LlmChain\ChainInterface;
88
use PhpLlm\LlmChain\Model\Message\Message;
@@ -11,7 +11,7 @@
1111
use Symfony\Component\DependencyInjection\Attribute\Autowire;
1212
use Symfony\Component\HttpFoundation\RequestStack;
1313

14-
final class Wikipedia
14+
final class Chat
1515
{
1616
private const SESSION_KEY = 'wikipedia-chat';
1717

src/Wikipedia/Twig/WikipediaComponent.php src/Wikipedia/TwigComponent.php

+4-5
Original file line numberDiff line numberDiff line change
@@ -2,22 +2,21 @@
22

33
declare(strict_types=1);
44

5-
namespace App\Wikipedia\Twig;
5+
namespace App\Wikipedia;
66

7-
use App\Wikipedia\Chat\Wikipedia;
87
use PhpLlm\LlmChain\Model\Message\MessageBag;
98
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
109
use Symfony\UX\LiveComponent\Attribute\LiveAction;
1110
use Symfony\UX\LiveComponent\Attribute\LiveArg;
1211
use Symfony\UX\LiveComponent\DefaultActionTrait;
1312

14-
#[AsLiveComponent('wikipedia')]
15-
final class WikipediaComponent
13+
#[AsLiveComponent('wikipedia', template: 'wikipedia/component.html.twig')]
14+
final class TwigComponent
1615
{
1716
use DefaultActionTrait;
1817

1918
public function __construct(
20-
private readonly Wikipedia $wikipedia,
19+
private readonly Chat $wikipedia,
2120
) {
2221
}
2322

src/YouTube/Chat/YouTube.php src/YouTube/Chat.php

+2-3
Original file line numberDiff line numberDiff line change
@@ -2,17 +2,16 @@
22

33
declare(strict_types=1);
44

5-
namespace App\YouTube\Chat;
5+
namespace App\YouTube;
66

7-
use App\YouTube\TranscriptFetcher;
87
use PhpLlm\LlmChain\ChainInterface;
98
use PhpLlm\LlmChain\Model\Message\Message;
109
use PhpLlm\LlmChain\Model\Message\MessageBag;
1110
use PhpLlm\LlmChain\Model\Response\TextResponse;
1211
use Symfony\Component\DependencyInjection\Attribute\Autowire;
1312
use Symfony\Component\HttpFoundation\RequestStack;
1413

15-
final class YouTube
14+
final class Chat
1615
{
1716
private const SESSION_KEY = 'youtube-chat';
1817

src/YouTube/Twig/YouTubeComponent.php src/YouTube/TwigComponent.php

+4-5
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,8 @@
22

33
declare(strict_types=1);
44

5-
namespace App\YouTube\Twig;
5+
namespace App\YouTube;
66

7-
use App\YouTube\Chat\YouTube;
87
use PhpLlm\LlmChain\Model\Message\MessageBag;
98
use Psr\Log\LoggerInterface;
109
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
@@ -14,13 +13,13 @@
1413

1514
use function Symfony\Component\String\u;
1615

17-
#[AsLiveComponent('youtube')]
18-
final class YouTubeComponent
16+
#[AsLiveComponent('youtube', template: 'youtube/component.html.twig')]
17+
final class TwigComponent
1918
{
2019
use DefaultActionTrait;
2120

2221
public function __construct(
23-
private readonly YouTube $youTube,
22+
private readonly Chat $youTube,
2423
private readonly LoggerInterface $logger,
2524
) {
2625
}
File renamed without changes.

templates/base.html.twig

+2-2
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,13 @@
1717
<nav class="navbar bg-white shadow-lg navbar-expand">
1818
<div class="container">
1919
<a class="navbar-brand ms-2 p-0" href="{{ path('index') }}">
20-
{{ ux_icon('mingcute:ai-fill', { height: '40px', width: '40px' }) }}
20+
{{ ux_icon('fluent:bot-24-filled', { height: '40px', width: '40px' }) }}
2121
<strong>LLM Chain</strong> Demo
2222
</a>
2323
<div class="collapse navbar-collapse">
2424
<ul class="navbar-nav ms-auto me-2 mb-0">
2525
<li class="nav-item">
26-
<a class="nav-link" href="{{ path('rag') }}">{{ ux_icon('fluent:bot-24-filled', { height: '20px', width: '20px' }) }} RAG Chat Bot</a>
26+
<a class="nav-link" href="{{ path('blog') }}">{{ ux_icon('mdi:symfony', { height: '20px', width: '20px' }) }} Symfony Blog Bot</a>
2727
</li>
2828
<li class="nav-item">
2929
<a class="nav-link" href="{{ path('youtube') }}">{{ ux_icon('bi:youtube', { height: '20px', width: '20px' }) }} YouTube Transcript Bot</a>
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
11
{% extends 'base.html.twig' %}
22

3-
{% block body_class 'chat rag' %}
3+
{% block body_class 'chat blog' %}
44

55
{% block content %}
66
<div>
7-
<twig:rag />
7+
<twig:blog />
88
</div>
99
{% endblock %}

0 commit comments

Comments
 (0)