├── public
├── favicon.ico
├── robots.txt
├── .htaccess
└── index.php
├── database
├── .gitignore
├── seeders
│ └── DatabaseSeeder.php
├── migrations
│ ├── 2014_10_12_100000_create_password_reset_tokens_table.php
│ ├── 2023_03_29_181836_chatboxes.php
│ ├── 2014_10_12_000000_create_users_table.php
│ ├── 2019_08_19_000000_create_failed_jobs_table.php
│ ├── 2023_03_24_095651_create_jobs_table.php
│ └── 2019_12_14_000001_create_personal_access_tokens_table.php
└── factories
│ └── UserFactory.php
├── bootstrap
├── cache
│ └── .gitignore
└── app.php
├── storage
├── logs
│ └── .gitignore
├── app
│ ├── public
│ │ └── .gitignore
│ └── .gitignore
└── framework
│ ├── testing
│ └── .gitignore
│ ├── views
│ └── .gitignore
│ ├── cache
│ ├── data
│ │ └── .gitignore
│ └── .gitignore
│ ├── sessions
│ └── .gitignore
│ └── .gitignore
├── resources
├── css
│ └── app.css
├── js
│ ├── app.js
│ └── bootstrap.js
└── views
│ ├── components
│ ├── input-label.blade.php
│ ├── auth-session-status.blade.php
│ ├── dropdown-link.blade.php
│ ├── input-error.blade.php
│ ├── text-input.blade.php
│ ├── danger-button.blade.php
│ ├── secondary-button.blade.php
│ ├── primary-button.blade.php
│ ├── nav-link.blade.php
│ ├── responsive-nav-link.blade.php
│ ├── dropdown.blade.php
│ ├── application-logo.blade.php
│ └── modal.blade.php
│ ├── layouts
│ ├── mail.blade.php
│ ├── guest.blade.php
│ ├── app.blade.php
│ └── navigation.blade.php
│ ├── emails
│ └── chatbox.blade.php
│ ├── wordpress
│ └── index.blade.php
│ ├── auth
│ ├── confirm-password.blade.php
│ ├── forgot-password.blade.php
│ ├── verify-email.blade.php
│ ├── reset-password.blade.php
│ ├── login.blade.php
│ └── register.blade.php
│ ├── profile
│ ├── edit.blade.php
│ └── partials
│ │ ├── update-password-form.blade.php
│ │ ├── delete-user-form.blade.php
│ │ └── update-profile-information-form.blade.php
│ ├── chatbox
│ └── index.blade.php
│ ├── livewire
│ └── transcribe-box
│ │ └── transcribe-box.blade.php
│ └── dashboard.blade.php
├── postcss.config.js
├── tests
├── TestCase.php
├── Unit
│ └── ExampleTest.php
├── Feature
│ ├── ExampleTest.php
│ ├── Auth
│ │ ├── RegistrationTest.php
│ │ ├── AuthenticationTest.php
│ │ ├── PasswordConfirmationTest.php
│ │ ├── PasswordUpdateTest.php
│ │ ├── EmailVerificationTest.php
│ │ └── PasswordResetTest.php
│ └── ProfileTest.php
└── CreatesApplication.php
├── .gitattributes
├── app
├── Http
│ ├── Controllers
│ │ ├── WordpressController.php
│ │ ├── Controller.php
│ │ ├── ChatBoxController.php
│ │ ├── Auth
│ │ │ ├── EmailVerificationPromptController.php
│ │ │ ├── EmailVerificationNotificationController.php
│ │ │ ├── PasswordController.php
│ │ │ ├── VerifyEmailController.php
│ │ │ ├── ConfirmablePasswordController.php
│ │ │ ├── AuthenticatedSessionController.php
│ │ │ ├── PasswordResetLinkController.php
│ │ │ ├── RegisteredUserController.php
│ │ │ └── NewPasswordController.php
│ │ └── ProfileController.php
│ ├── Middleware
│ │ ├── EncryptCookies.php
│ │ ├── VerifyCsrfToken.php
│ │ ├── PreventRequestsDuringMaintenance.php
│ │ ├── TrimStrings.php
│ │ ├── TrustHosts.php
│ │ ├── Authenticate.php
│ │ ├── ValidateSignature.php
│ │ ├── TrustProxies.php
│ │ └── RedirectIfAuthenticated.php
│ ├── Requests
│ │ ├── ProfileUpdateRequest.php
│ │ └── Auth
│ │ │ └── LoginRequest.php
│ └── Kernel.php
├── Models
│ ├── ChatBox.php
│ ├── User.php
│ └── Wordpress.php
├── View
│ └── Components
│ │ ├── AppLayout.php
│ │ └── GuestLayout.php
├── Providers
│ ├── BroadcastServiceProvider.php
│ ├── AppServiceProvider.php
│ ├── AuthServiceProvider.php
│ ├── EventServiceProvider.php
│ └── RouteServiceProvider.php
├── Console
│ ├── Kernel.php
│ └── Commands
│ │ └── Chat.php
├── Jobs
│ └── SendEmailJob.php
├── Exceptions
│ └── Handler.php
├── Mail
│ └── SendChatToEmail.php
├── Livewire
│ ├── TranscribeBox.php
│ ├── WordpressSeo.php
│ ├── WordpressCreatePost.php
│ └── ChatBox.php
└── Services
│ └── openAIService.php
├── .gitignore
├── .editorconfig
├── vite.config.js
├── package.json
├── config
├── openai.php
├── cors.php
├── services.php
├── view.php
├── hashing.php
├── broadcasting.php
├── sanctum.php
├── filesystems.php
├── queue.php
├── markdown.php
├── cache.php
├── mail.php
├── logging.php
├── auth.php
└── database.php
├── routes
├── channels.php
├── api.php
├── console.php
├── web.php
└── auth.php
├── tailwind.config.js
├── LICENSE
├── phpunit.xml
├── .env.example
├── artisan
├── composer.json
├── README.md
└── CODE_OF_CONDUCT.md
/public/favicon.ico:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/database/.gitignore:
--------------------------------------------------------------------------------
1 | *.sqlite*
2 |
--------------------------------------------------------------------------------
/bootstrap/cache/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/logs/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/public/robots.txt:
--------------------------------------------------------------------------------
1 | User-agent: *
2 | Disallow:
3 |
--------------------------------------------------------------------------------
/storage/app/public/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/app/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !public/
3 | !.gitignore
4 |
--------------------------------------------------------------------------------
/storage/framework/testing/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/framework/views/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/framework/cache/data/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/framework/sessions/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 |
--------------------------------------------------------------------------------
/storage/framework/cache/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !data/
3 | !.gitignore
4 |
--------------------------------------------------------------------------------
/resources/css/app.css:
--------------------------------------------------------------------------------
1 | @tailwind base;
2 | @tailwind components;
3 | @tailwind utilities;
4 |
--------------------------------------------------------------------------------
/postcss.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | plugins: {
3 | tailwindcss: {},
4 | autoprefixer: {},
5 | },
6 | };
7 |
--------------------------------------------------------------------------------
/resources/js/app.js:
--------------------------------------------------------------------------------
1 | import "./bootstrap";
2 |
3 | import "flowbite";
4 |
5 | // import Alpine from "alpinejs";
6 |
7 | // window.Alpine = Alpine;
8 |
9 | // Alpine.start();
10 |
--------------------------------------------------------------------------------
/storage/framework/.gitignore:
--------------------------------------------------------------------------------
1 | compiled.php
2 | config.php
3 | down
4 | events.scanned.php
5 | maintenance.php
6 | routes.php
7 | routes.scanned.php
8 | schedule-*
9 | services.json
10 |
--------------------------------------------------------------------------------
/resources/views/components/input-label.blade.php:
--------------------------------------------------------------------------------
1 | @props(['value'])
2 |
3 | merge(['class' => 'block font-medium text-sm text-gray-700 dark:text-gray-300']) }}>
4 | {{ $value ?? $slot }}
5 |
6 |
--------------------------------------------------------------------------------
/tests/TestCase.php:
--------------------------------------------------------------------------------
1 | merge(['class' => 'font-medium text-sm text-green-600 dark:text-green-400']) }}>
5 | {{ $status }}
6 |
7 | @endif
8 |
--------------------------------------------------------------------------------
/app/Http/Controllers/WordpressController.php:
--------------------------------------------------------------------------------
1 | merge(['class' => 'block w-full px-4 py-2 text-left text-sm leading-5 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-800 focus:outline-none focus:bg-gray-100 dark:focus:bg-gray-800 transition duration-150 ease-in-out']) }}>{{ $slot }}
2 |
--------------------------------------------------------------------------------
/app/Models/ChatBox.php:
--------------------------------------------------------------------------------
1 | merge(['class' => 'text-sm text-red-600 dark:text-red-400 space-y-1']) }}>
5 | @foreach ((array) $messages as $message)
6 |
{{ $message }}
7 | @endforeach
8 |
9 | @endif
10 |
--------------------------------------------------------------------------------
/tests/Unit/ExampleTest.php:
--------------------------------------------------------------------------------
1 | assertTrue(true);
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | root = true
2 |
3 | [*]
4 | charset = utf-8
5 | end_of_line = lf
6 | indent_size = 4
7 | indent_style = space
8 | insert_final_newline = true
9 | trim_trailing_whitespace = true
10 |
11 | [*.md]
12 | trim_trailing_whitespace = false
13 |
14 | [*.{yml,yaml}]
15 | indent_size = 2
16 |
17 | [docker-compose.yml]
18 | indent_size = 4
19 |
--------------------------------------------------------------------------------
/resources/views/components/text-input.blade.php:
--------------------------------------------------------------------------------
1 | @props(['disabled' => false])
2 |
3 | merge(['class' => 'border-gray-300 dark:border-gray-700 dark:bg-gray-900 dark:text-gray-300 focus:border-indigo-500 dark:focus:border-indigo-600 focus:ring-indigo-500 dark:focus:ring-indigo-600 rounded-md shadow-sm']) !!}>
4 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Controller.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | protected $except = [
15 | //
16 | ];
17 | }
18 |
--------------------------------------------------------------------------------
/app/Http/Middleware/VerifyCsrfToken.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | protected $except = [
15 | //
16 | ];
17 | }
18 |
--------------------------------------------------------------------------------
/resources/views/components/danger-button.blade.php:
--------------------------------------------------------------------------------
1 | merge(['type' => 'submit', 'class' => 'inline-flex items-center px-4 py-2 bg-red-600 border border-transparent rounded-md font-semibold text-xs text-white uppercase tracking-widest hover:bg-red-500 active:bg-red-700 focus:outline-none focus:ring-2 focus:ring-red-500 focus:ring-offset-2 dark:focus:ring-offset-gray-800 transition ease-in-out duration-150']) }}>
2 | {{ $slot }}
3 |
4 |
--------------------------------------------------------------------------------
/tests/Feature/ExampleTest.php:
--------------------------------------------------------------------------------
1 | get('/');
16 |
17 | $response->assertStatus(200);
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/app/Providers/BroadcastServiceProvider.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | protected $except = [
15 | //
16 | ];
17 | }
18 |
--------------------------------------------------------------------------------
/app/Http/Middleware/TrimStrings.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | protected $except = [
15 | 'current_password',
16 | 'password',
17 | 'password_confirmation',
18 | ];
19 | }
20 |
--------------------------------------------------------------------------------
/tests/CreatesApplication.php:
--------------------------------------------------------------------------------
1 | make(Kernel::class)->bootstrap();
18 |
19 | return $app;
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/app/Http/Middleware/TrustHosts.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | public function hosts(): array
15 | {
16 | return [
17 | $this->allSubdomainsOfApplicationUrl(),
18 | ];
19 | }
20 | }
21 |
--------------------------------------------------------------------------------
/app/Providers/AppServiceProvider.php:
--------------------------------------------------------------------------------
1 | expectsJson() ? null : route('login');
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/app/Http/Controllers/ChatBoxController.php:
--------------------------------------------------------------------------------
1 | $chatbox,
13 | ]);
14 | }
15 |
16 | public function destroy(ChatBox $chatbox)
17 | {
18 | $chatbox->delete();
19 |
20 | return redirect()->route('dashboard');
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/resources/views/components/secondary-button.blade.php:
--------------------------------------------------------------------------------
1 | merge(['type' => 'button', 'class' => 'inline-flex items-center px-4 py-2 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-500 rounded-md font-semibold text-xs text-gray-700 dark:text-gray-300 uppercase tracking-widest shadow-sm hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:ring-offset-2 dark:focus:ring-offset-gray-800 disabled:opacity-25 transition ease-in-out duration-150']) }}>
2 | {{ $slot }}
3 |
4 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "private": true,
3 | "scripts": {
4 | "dev": "vite",
5 | "build": "vite build"
6 | },
7 | "devDependencies": {
8 | "@tailwindcss/forms": "^0.5.2",
9 | "alpinejs": "^3.13.2",
10 | "autoprefixer": "^10.4.14",
11 | "axios": "^1.1.2",
12 | "flowbite": "^1.6.4",
13 | "laravel-vite-plugin": "^0.7.2",
14 | "postcss": "^8.4.21",
15 | "tailwindcss": "^3.2.7",
16 | "vite": "^4.0.0"
17 | },
18 | "dependencies": {
19 | "shiki": "^0.14.1"
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/resources/views/components/primary-button.blade.php:
--------------------------------------------------------------------------------
1 | merge(['type' => 'submit', 'class' => 'inline-flex items-center px-4 py-2 bg-gray-800 dark:bg-gray-200 border border-transparent rounded-md font-semibold text-xs text-white dark:text-gray-800 uppercase tracking-widest hover:bg-gray-700 dark:hover:bg-white focus:bg-gray-700 dark:focus:bg-white active:bg-gray-900 dark:active:bg-gray-300 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:ring-offset-2 dark:focus:ring-offset-gray-800 transition ease-in-out duration-150']) }}>
2 | {{ $slot }}
3 |
4 |
--------------------------------------------------------------------------------
/app/Http/Middleware/ValidateSignature.php:
--------------------------------------------------------------------------------
1 |
13 | */
14 | protected $except = [
15 | // 'fbclid',
16 | // 'utm_campaign',
17 | // 'utm_content',
18 | // 'utm_medium',
19 | // 'utm_source',
20 | // 'utm_term',
21 | ];
22 | }
23 |
--------------------------------------------------------------------------------
/database/seeders/DatabaseSeeder.php:
--------------------------------------------------------------------------------
1 | create();
16 |
17 | // \App\Models\User::factory()->create([
18 | // 'name' => 'Test User',
19 | // 'email' => 'test@example.com',
20 | // ]);
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/config/openai.php:
--------------------------------------------------------------------------------
1 | env('OPENAI_API_KEY'),
16 | 'organization' => env('OPENAI_ORGANIZATION'),
17 |
18 | ];
19 |
--------------------------------------------------------------------------------
/routes/channels.php:
--------------------------------------------------------------------------------
1 | id === (int) $id;
18 | });
19 |
--------------------------------------------------------------------------------
/routes/api.php:
--------------------------------------------------------------------------------
1 | get('/user', function (Request $request) {
18 | return $request->user();
19 | });
20 |
--------------------------------------------------------------------------------
/routes/console.php:
--------------------------------------------------------------------------------
1 | comment(Inspiring::quote());
19 | })->purpose('Display an inspiring quote');
20 |
--------------------------------------------------------------------------------
/app/Http/Requests/ProfileUpdateRequest.php:
--------------------------------------------------------------------------------
1 |
15 | */
16 | public function rules(): array
17 | {
18 | return [
19 | 'name' => ['string', 'max:255'],
20 | 'email' => ['email', 'max:255', Rule::unique(User::class)->ignore($this->user()->id)],
21 | ];
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/public/.htaccess:
--------------------------------------------------------------------------------
1 |
2 |
3 | Options -MultiViews -Indexes
4 |
5 |
6 | RewriteEngine On
7 |
8 | # Handle Authorization Header
9 | RewriteCond %{HTTP:Authorization} .
10 | RewriteRule .* - [E=HTTP_AUTHORIZATION:%{HTTP:Authorization}]
11 |
12 | # Redirect Trailing Slashes If Not A Folder...
13 | RewriteCond %{REQUEST_FILENAME} !-d
14 | RewriteCond %{REQUEST_URI} (.+)/$
15 | RewriteRule ^ %1 [L,R=301]
16 |
17 | # Send Requests To Front Controller...
18 | RewriteCond %{REQUEST_FILENAME} !-d
19 | RewriteCond %{REQUEST_FILENAME} !-f
20 | RewriteRule ^ index.php [L]
21 |
22 |
--------------------------------------------------------------------------------
/tailwind.config.js:
--------------------------------------------------------------------------------
1 | const defaultTheme = require("tailwindcss/defaultTheme");
2 |
3 | /** @type {import('tailwindcss').Config} */
4 | module.exports = {
5 | content: [
6 | "./vendor/laravel/framework/src/Illuminate/Pagination/resources/views/*.blade.php",
7 | "./storage/framework/views/*.php",
8 | "./resources/views/**/*.blade.php",
9 | "./node_modules/flowbite/**/*.js",
10 | ],
11 |
12 | theme: {
13 | extend: {
14 | fontFamily: {
15 | sans: ["Figtree", ...defaultTheme.fontFamily.sans],
16 | },
17 | },
18 | },
19 |
20 | plugins: [require("@tailwindcss/forms"), require('flowbite/plugin')],
21 | };
22 |
--------------------------------------------------------------------------------
/app/Providers/AuthServiceProvider.php:
--------------------------------------------------------------------------------
1 |
14 | */
15 | protected $policies = [
16 | // 'App\Models\Model' => 'App\Policies\ModelPolicy',
17 | ];
18 |
19 | /**
20 | * Register any authentication / authorization services.
21 | */
22 | public function boot(): void
23 | {
24 | //
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/EmailVerificationPromptController.php:
--------------------------------------------------------------------------------
1 | user()->hasVerifiedEmail()
19 | ? redirect()->intended(RouteServiceProvider::HOME)
20 | : view('auth.verify-email');
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/app/Console/Kernel.php:
--------------------------------------------------------------------------------
1 | command('queue:work --stop-when-empty')
16 | ->everyMinute()
17 | ->withoutOverlapping();
18 | }
19 |
20 | /**
21 | * Register the commands for the application.
22 | */
23 | protected function commands(): void
24 | {
25 | $this->load(__DIR__.'/Commands');
26 |
27 | require base_path('routes/console.php');
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/app/Http/Middleware/TrustProxies.php:
--------------------------------------------------------------------------------
1 | |string|null
14 | */
15 | protected $proxies;
16 |
17 | /**
18 | * The headers that should be used to detect proxies.
19 | *
20 | * @var int
21 | */
22 | protected $headers =
23 | Request::HEADER_X_FORWARDED_FOR |
24 | Request::HEADER_X_FORWARDED_HOST |
25 | Request::HEADER_X_FORWARDED_PORT |
26 | Request::HEADER_X_FORWARDED_PROTO |
27 | Request::HEADER_X_FORWARDED_AWS_ELB;
28 | }
29 |
--------------------------------------------------------------------------------
/resources/views/layouts/mail.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | @yield('title')
7 |
8 |
11 |
12 |
13 |
14 |
17 |
18 | @yield('content')
19 |
20 |
23 |
24 |
25 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/EmailVerificationNotificationController.php:
--------------------------------------------------------------------------------
1 | user()->hasVerifiedEmail()) {
18 | return redirect()->intended(RouteServiceProvider::HOME);
19 | }
20 |
21 | $request->user()->sendEmailVerificationNotification();
22 |
23 | return back()->with('status', 'verification-link-sent');
24 | }
25 | }
26 |
--------------------------------------------------------------------------------
/database/migrations/2014_10_12_100000_create_password_reset_tokens_table.php:
--------------------------------------------------------------------------------
1 | string('email')->primary();
16 | $table->string('token');
17 | $table->timestamp('created_at')->nullable();
18 | });
19 | }
20 |
21 | /**
22 | * Reverse the migrations.
23 | */
24 | public function down(): void
25 | {
26 | Schema::dropIfExists('password_reset_tokens');
27 | }
28 | };
29 |
--------------------------------------------------------------------------------
/resources/views/components/nav-link.blade.php:
--------------------------------------------------------------------------------
1 | @props(['active'])
2 |
3 | @php
4 | $classes = ($active ?? false)
5 | ? 'inline-flex items-center px-1 pt-1 border-b-2 border-indigo-400 dark:border-indigo-600 text-sm font-medium leading-5 text-gray-900 dark:text-gray-100 focus:outline-none focus:border-indigo-700 transition duration-150 ease-in-out'
6 | : 'inline-flex items-center px-1 pt-1 border-b-2 border-transparent text-sm font-medium leading-5 text-gray-500 dark:text-gray-400 hover:text-gray-700 dark:hover:text-gray-300 hover:border-gray-300 dark:hover:border-gray-700 focus:outline-none focus:text-gray-700 dark:focus:text-gray-300 focus:border-gray-300 dark:focus:border-gray-700 transition duration-150 ease-in-out';
7 | @endphp
8 |
9 | merge(['class' => $classes]) }}>
10 | {{ $slot }}
11 |
12 |
--------------------------------------------------------------------------------
/database/migrations/2023_03_29_181836_chatboxes.php:
--------------------------------------------------------------------------------
1 | id();
16 | $table->foreignId('user_id')->references('id')->on('users');
17 | $table->longText('messages');
18 | $table->string('total_tokens');
19 | $table->timestamps();
20 | });
21 | }
22 |
23 | /**
24 | * Reverse the migrations.
25 | */
26 | public function down(): void
27 | {
28 | Schema::dropIfExists('chatboxes');
29 | }
30 | };
31 |
--------------------------------------------------------------------------------
/app/Http/Middleware/RedirectIfAuthenticated.php:
--------------------------------------------------------------------------------
1 | check()) {
24 | return redirect(RouteServiceProvider::HOME);
25 | }
26 | }
27 |
28 | return $next($request);
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/resources/views/emails/chatbox.blade.php:
--------------------------------------------------------------------------------
1 | @extends('layouts.mail')
2 |
3 | @section('title', 'This is your Chatbox Conversation')
4 |
5 | @section('content')
6 | @foreach ($email['messages'] as $message)
7 |
8 |
9 | @if ($message['role'] === 'assistant')
10 | Your Assistant
11 | @else
12 | You
13 | @endif
14 |
15 |
16 |
17 | {!! \Illuminate\Mail\Markdown::parse($message['content']) !!}
18 |
19 |
20 |
21 | @endforeach
22 | @endsection
23 |
24 | @section('footer')
25 | Thanks,
26 | {{ config('app.name') }}
27 | @endsection
--------------------------------------------------------------------------------
/database/migrations/2014_10_12_000000_create_users_table.php:
--------------------------------------------------------------------------------
1 | id();
16 | $table->string('name');
17 | $table->string('email')->unique();
18 | $table->timestamp('email_verified_at')->nullable();
19 | $table->string('password');
20 | $table->rememberToken();
21 | $table->timestamps();
22 | });
23 | }
24 |
25 | /**
26 | * Reverse the migrations.
27 | */
28 | public function down(): void
29 | {
30 | Schema::dropIfExists('users');
31 | }
32 | };
33 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/RegistrationTest.php:
--------------------------------------------------------------------------------
1 | get('/register');
16 |
17 | $response->assertStatus(200);
18 | }
19 |
20 | public function test_new_users_can_register(): void
21 | {
22 | $response = $this->post('/register', [
23 | 'name' => 'Test User',
24 | 'email' => 'test@example.com',
25 | 'password' => 'password',
26 | 'password_confirmation' => 'password',
27 | ]);
28 |
29 | $this->assertAuthenticated();
30 | $response->assertRedirect(RouteServiceProvider::HOME);
31 | }
32 | }
33 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/PasswordController.php:
--------------------------------------------------------------------------------
1 | validateWithBag('updatePassword', [
19 | 'current_password' => ['required', 'current_password'],
20 | 'password' => ['required', Password::defaults(), 'confirmed'],
21 | ]);
22 |
23 | $request->user()->update([
24 | 'password' => Hash::make($validated['password']),
25 | ]);
26 |
27 | return back()->with('status', 'password-updated');
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/database/migrations/2019_08_19_000000_create_failed_jobs_table.php:
--------------------------------------------------------------------------------
1 | id();
16 | $table->string('uuid')->unique();
17 | $table->text('connection');
18 | $table->text('queue');
19 | $table->longText('payload');
20 | $table->longText('exception');
21 | $table->timestamp('failed_at')->useCurrent();
22 | });
23 | }
24 |
25 | /**
26 | * Reverse the migrations.
27 | */
28 | public function down(): void
29 | {
30 | Schema::dropIfExists('failed_jobs');
31 | }
32 | };
33 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/VerifyEmailController.php:
--------------------------------------------------------------------------------
1 | user()->hasVerifiedEmail()) {
19 | return redirect()->intended(RouteServiceProvider::HOME.'?verified=1');
20 | }
21 |
22 | if ($request->user()->markEmailAsVerified()) {
23 | event(new Verified($request->user()));
24 | }
25 |
26 | return redirect()->intended(RouteServiceProvider::HOME.'?verified=1');
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/config/cors.php:
--------------------------------------------------------------------------------
1 | ['api/*', 'sanctum/csrf-cookie'],
19 |
20 | 'allowed_methods' => ['*'],
21 |
22 | 'allowed_origins' => ['*'],
23 |
24 | 'allowed_origins_patterns' => [],
25 |
26 | 'allowed_headers' => ['*'],
27 |
28 | 'exposed_headers' => [],
29 |
30 | 'max_age' => 0,
31 |
32 | 'supports_credentials' => false,
33 |
34 | ];
35 |
--------------------------------------------------------------------------------
/database/migrations/2023_03_24_095651_create_jobs_table.php:
--------------------------------------------------------------------------------
1 | bigIncrements('id');
16 | $table->string('queue')->index();
17 | $table->longText('payload');
18 | $table->unsignedTinyInteger('attempts');
19 | $table->unsignedInteger('reserved_at')->nullable();
20 | $table->unsignedInteger('available_at');
21 | $table->unsignedInteger('created_at');
22 | });
23 | }
24 |
25 | /**
26 | * Reverse the migrations.
27 | */
28 | public function down(): void
29 | {
30 | Schema::dropIfExists('jobs');
31 | }
32 | };
33 |
--------------------------------------------------------------------------------
/resources/views/wordpress/index.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | {{ __('Wordpress') }}
5 |
6 |
7 |
23 |
24 |
--------------------------------------------------------------------------------
/app/Jobs/SendEmailJob.php:
--------------------------------------------------------------------------------
1 | details = $details;
26 | }
27 |
28 | /**
29 | * Execute the job.
30 | */
31 | public function handle(): void
32 | {
33 | Log::info('Sending email to '.$this->details['email']);
34 | $email = new SendChatToEmail($this->details);
35 | Mail::to($this->details['email'])->send($email);
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/resources/views/auth/confirm-password.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 | {{ __('This is a secure area of the application. Please confirm your password before continuing.') }}
4 |
5 |
6 |
27 |
28 |
--------------------------------------------------------------------------------
/database/migrations/2019_12_14_000001_create_personal_access_tokens_table.php:
--------------------------------------------------------------------------------
1 | id();
16 | $table->morphs('tokenable');
17 | $table->string('name');
18 | $table->string('token', 64)->unique();
19 | $table->text('abilities')->nullable();
20 | $table->timestamp('last_used_at')->nullable();
21 | $table->timestamp('expires_at')->nullable();
22 | $table->timestamps();
23 | });
24 | }
25 |
26 | /**
27 | * Reverse the migrations.
28 | */
29 | public function down(): void
30 | {
31 | Schema::dropIfExists('personal_access_tokens');
32 | }
33 | };
34 |
--------------------------------------------------------------------------------
/app/Providers/EventServiceProvider.php:
--------------------------------------------------------------------------------
1 | >
16 | */
17 | protected $listen = [
18 | Registered::class => [
19 | SendEmailVerificationNotification::class,
20 | ],
21 | ];
22 |
23 | /**
24 | * Register any events for your application.
25 | */
26 | public function boot(): void
27 | {
28 | //
29 | }
30 |
31 | /**
32 | * Determine if events and listeners should be automatically discovered.
33 | */
34 | public function shouldDiscoverEvents(): bool
35 | {
36 | return false;
37 | }
38 | }
39 |
--------------------------------------------------------------------------------
/resources/views/components/responsive-nav-link.blade.php:
--------------------------------------------------------------------------------
1 | @props(['active'])
2 |
3 | @php
4 | $classes = ($active ?? false)
5 | ? 'block w-full pl-3 pr-4 py-2 border-l-4 border-indigo-400 dark:border-indigo-600 text-left text-base font-medium text-indigo-700 dark:text-indigo-300 bg-indigo-50 dark:bg-indigo-900/50 focus:outline-none focus:text-indigo-800 dark:focus:text-indigo-200 focus:bg-indigo-100 dark:focus:bg-indigo-900 focus:border-indigo-700 dark:focus:border-indigo-300 transition duration-150 ease-in-out'
6 | : 'block w-full pl-3 pr-4 py-2 border-l-4 border-transparent text-left text-base font-medium text-gray-600 dark:text-gray-400 hover:text-gray-800 dark:hover:text-gray-200 hover:bg-gray-50 dark:hover:bg-gray-700 hover:border-gray-300 dark:hover:border-gray-600 focus:outline-none focus:text-gray-800 dark:focus:text-gray-200 focus:bg-gray-50 dark:focus:bg-gray-700 focus:border-gray-300 dark:focus:border-gray-600 transition duration-150 ease-in-out';
7 | @endphp
8 |
9 | merge(['class' => $classes]) }}>
10 | {{ $slot }}
11 |
12 |
--------------------------------------------------------------------------------
/app/Models/User.php:
--------------------------------------------------------------------------------
1 |
19 | */
20 | protected $fillable = [
21 | 'name',
22 | 'email',
23 | 'password',
24 | ];
25 |
26 | /**
27 | * The attributes that should be hidden for serialization.
28 | *
29 | * @var array
30 | */
31 | protected $hidden = [
32 | 'password',
33 | 'remember_token',
34 | ];
35 |
36 | /**
37 | * The attributes that should be cast.
38 | *
39 | * @var array
40 | */
41 | protected $casts = [
42 | 'email_verified_at' => 'datetime',
43 | ];
44 | }
45 |
--------------------------------------------------------------------------------
/config/services.php:
--------------------------------------------------------------------------------
1 | [
18 | 'domain' => env('MAILGUN_DOMAIN'),
19 | 'secret' => env('MAILGUN_SECRET'),
20 | 'endpoint' => env('MAILGUN_ENDPOINT', 'api.mailgun.net'),
21 | 'scheme' => 'https',
22 | ],
23 |
24 | 'postmark' => [
25 | 'token' => env('POSTMARK_TOKEN'),
26 | ],
27 |
28 | 'ses' => [
29 | 'key' => env('AWS_ACCESS_KEY_ID'),
30 | 'secret' => env('AWS_SECRET_ACCESS_KEY'),
31 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'),
32 | ],
33 |
34 | ];
35 |
--------------------------------------------------------------------------------
/resources/views/auth/forgot-password.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 | {{ __('Forgot your password? No problem. Just let us know your email address and we will email you a password reset link that will allow you to choose a new one.') }}
4 |
5 |
6 |
7 |
8 |
9 |
25 |
26 |
--------------------------------------------------------------------------------
/database/factories/UserFactory.php:
--------------------------------------------------------------------------------
1 |
10 | */
11 | class UserFactory extends Factory
12 | {
13 | /**
14 | * Define the model's default state.
15 | *
16 | * @return array
17 | */
18 | public function definition(): array
19 | {
20 | return [
21 | 'name' => fake()->name(),
22 | 'email' => fake()->unique()->safeEmail(),
23 | 'email_verified_at' => now(),
24 | 'password' => '$2y$10$92IXUNpkjO0rOQ5byMi.Ye4oKoEa3Ro9llC/.og/at2.uheWG/igi', // password
25 | 'remember_token' => Str::random(10),
26 | ];
27 | }
28 |
29 | /**
30 | * Indicate that the model's email address should be unverified.
31 | */
32 | public function unverified(): static
33 | {
34 | return $this->state(fn (array $attributes) => [
35 | 'email_verified_at' => null,
36 | ]);
37 | }
38 | }
39 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2023 Theodoros Kafantaris
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/resources/views/profile/edit.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | {{ __('Profile') }}
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 | @include('profile.partials.update-profile-information-form')
13 |
14 |
15 |
16 |
17 |
18 | @include('profile.partials.update-password-form')
19 |
20 |
21 |
22 |
23 |
24 | @include('profile.partials.delete-user-form')
25 |
26 |
27 |
28 |
29 |
30 |
--------------------------------------------------------------------------------
/config/view.php:
--------------------------------------------------------------------------------
1 | [
17 | resource_path('views'),
18 | ],
19 |
20 | /*
21 | |--------------------------------------------------------------------------
22 | | Compiled View Path
23 | |--------------------------------------------------------------------------
24 | |
25 | | This option determines where all the compiled Blade templates will be
26 | | stored for your application. Typically, this is within the storage
27 | | directory. However, as usual, you are free to change this value.
28 | |
29 | */
30 |
31 | 'compiled' => env(
32 | 'VIEW_COMPILED_PATH',
33 | realpath(storage_path('framework/views'))
34 | ),
35 |
36 | ];
37 |
--------------------------------------------------------------------------------
/app/Exceptions/Handler.php:
--------------------------------------------------------------------------------
1 | , \Psr\Log\LogLevel::*>
14 | */
15 | protected $levels = [
16 | //
17 | ];
18 |
19 | /**
20 | * A list of the exception types that are not reported.
21 | *
22 | * @var array>
23 | */
24 | protected $dontReport = [
25 | //
26 | ];
27 |
28 | /**
29 | * A list of the inputs that are never flashed to the session on validation exceptions.
30 | *
31 | * @var array
32 | */
33 | protected $dontFlash = [
34 | 'current_password',
35 | 'password',
36 | 'password_confirmation',
37 | ];
38 |
39 | /**
40 | * Register the exception handling callbacks for the application.
41 | */
42 | public function register(): void
43 | {
44 | $this->reportable(function (Throwable $e) {
45 | //
46 | });
47 | }
48 | }
49 |
--------------------------------------------------------------------------------
/phpunit.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
9 | ./tests/Unit
10 |
11 |
12 | ./tests/Feature
13 |
14 |
15 |
16 |
17 | ./app
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/ConfirmablePasswordController.php:
--------------------------------------------------------------------------------
1 | validate([
29 | 'email' => $request->user()->email,
30 | 'password' => $request->password,
31 | ])) {
32 | throw ValidationException::withMessages([
33 | 'password' => __('auth.password'),
34 | ]);
35 | }
36 |
37 | $request->session()->put('auth.password_confirmed_at', time());
38 |
39 | return redirect()->intended(RouteServiceProvider::HOME);
40 | }
41 | }
42 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/AuthenticationTest.php:
--------------------------------------------------------------------------------
1 | get('/login');
17 |
18 | $response->assertStatus(200);
19 | }
20 |
21 | public function test_users_can_authenticate_using_the_login_screen(): void
22 | {
23 | $user = User::factory()->create();
24 |
25 | $response = $this->post('/login', [
26 | 'email' => $user->email,
27 | 'password' => 'password',
28 | ]);
29 |
30 | $this->assertAuthenticated();
31 | $response->assertRedirect(RouteServiceProvider::HOME);
32 | }
33 |
34 | public function test_users_can_not_authenticate_with_invalid_password(): void
35 | {
36 | $user = User::factory()->create();
37 |
38 | $this->post('/login', [
39 | 'email' => $user->email,
40 | 'password' => 'wrong-password',
41 | ]);
42 |
43 | $this->assertGuest();
44 | }
45 | }
46 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/PasswordConfirmationTest.php:
--------------------------------------------------------------------------------
1 | create();
16 |
17 | $response = $this->actingAs($user)->get('/confirm-password');
18 |
19 | $response->assertStatus(200);
20 | }
21 |
22 | public function test_password_can_be_confirmed(): void
23 | {
24 | $user = User::factory()->create();
25 |
26 | $response = $this->actingAs($user)->post('/confirm-password', [
27 | 'password' => 'password',
28 | ]);
29 |
30 | $response->assertRedirect();
31 | $response->assertSessionHasNoErrors();
32 | }
33 |
34 | public function test_password_is_not_confirmed_with_invalid_password(): void
35 | {
36 | $user = User::factory()->create();
37 |
38 | $response = $this->actingAs($user)->post('/confirm-password', [
39 | 'password' => 'wrong-password',
40 | ]);
41 |
42 | $response->assertSessionHasErrors();
43 | }
44 | }
45 |
--------------------------------------------------------------------------------
/resources/views/layouts/guest.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | {{ config('app.name', 'Laravel') }}
9 |
10 |
11 |
12 |
13 |
14 |
15 | @vite(['resources/css/app.css', 'resources/js/app.js'])
16 |
17 |
18 |
19 |
24 |
25 |
26 | {{ $slot }}
27 |
28 |
29 |
30 |
31 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/AuthenticatedSessionController.php:
--------------------------------------------------------------------------------
1 | authenticate();
29 |
30 | $request->session()->regenerate();
31 |
32 | return redirect()->intended(RouteServiceProvider::HOME);
33 | }
34 |
35 | /**
36 | * Destroy an authenticated session.
37 | */
38 | public function destroy(Request $request): RedirectResponse
39 | {
40 | Auth::guard('web')->logout();
41 |
42 | $request->session()->invalidate();
43 |
44 | $request->session()->regenerateToken();
45 |
46 | return redirect('/');
47 | }
48 | }
49 |
--------------------------------------------------------------------------------
/app/Mail/SendChatToEmail.php:
--------------------------------------------------------------------------------
1 | email = $email;
23 | }
24 |
25 | /**
26 | * Get the message envelope.
27 | */
28 | public function envelope(): Envelope
29 | {
30 | return new Envelope(
31 | subject: 'Mail from Laravel Livewire ChatGPT',
32 | );
33 | }
34 |
35 | /**
36 | * Get the message content definition.
37 | */
38 | public function content(): Content
39 | {
40 | return new Content(
41 | view: 'emails.chatbox',
42 | with: [
43 | 'email' => $this->email,
44 | ],
45 | );
46 | }
47 |
48 | /**
49 | * Get the attachments for the message.
50 | *
51 | * @return array
52 | */
53 | public function attachments(): array
54 | {
55 | return [];
56 | }
57 | }
58 |
--------------------------------------------------------------------------------
/resources/js/bootstrap.js:
--------------------------------------------------------------------------------
1 | /**
2 | * We'll load the axios HTTP library which allows us to easily issue requests
3 | * to our Laravel back-end. This library automatically handles sending the
4 | * CSRF token as a header based on the value of the "XSRF" token cookie.
5 | */
6 |
7 | import axios from 'axios';
8 | window.axios = axios;
9 |
10 | window.axios.defaults.headers.common['X-Requested-With'] = 'XMLHttpRequest';
11 |
12 | /**
13 | * Echo exposes an expressive API for subscribing to channels and listening
14 | * for events that are broadcast by Laravel. Echo and event broadcasting
15 | * allows your team to easily build robust real-time web applications.
16 | */
17 |
18 | // import Echo from 'laravel-echo';
19 |
20 | // import Pusher from 'pusher-js';
21 | // window.Pusher = Pusher;
22 |
23 | // window.Echo = new Echo({
24 | // broadcaster: 'pusher',
25 | // key: import.meta.env.VITE_PUSHER_APP_KEY,
26 | // cluster: import.meta.env.VITE_PUSHER_APP_CLUSTER ?? 'mt1',
27 | // wsHost: import.meta.env.VITE_PUSHER_HOST ? import.meta.env.VITE_PUSHER_HOST : `ws-${import.meta.env.VITE_PUSHER_APP_CLUSTER}.pusher.com`,
28 | // wsPort: import.meta.env.VITE_PUSHER_PORT ?? 80,
29 | // wssPort: import.meta.env.VITE_PUSHER_PORT ?? 443,
30 | // forceTLS: (import.meta.env.VITE_PUSHER_SCHEME ?? 'https') === 'https',
31 | // enabledTransports: ['ws', 'wss'],
32 | // });
33 |
--------------------------------------------------------------------------------
/.env.example:
--------------------------------------------------------------------------------
1 | APP_NAME="Chatwire for Laravel"
2 | APP_ENV=local
3 | APP_KEY=
4 | APP_DEBUG=true
5 | APP_URL=http://localhost
6 |
7 | LOG_CHANNEL=stack
8 | LOG_DEPRECATIONS_CHANNEL=null
9 | LOG_LEVEL=debug
10 |
11 | DB_CONNECTION=mysql
12 | DB_HOST=127.0.0.1
13 | DB_PORT=3306
14 | DB_DATABASE=laravel_livewire_chatgpt
15 | DB_USERNAME=root
16 | DB_PASSWORD=
17 |
18 | BROADCAST_DRIVER=log
19 | CACHE_DRIVER=file
20 | FILESYSTEM_DISK=local
21 | QUEUE_CONNECTION=sync
22 | SESSION_DRIVER=file
23 | SESSION_LIFETIME=120
24 |
25 | MEMCACHED_HOST=127.0.0.1
26 |
27 | REDIS_HOST=127.0.0.1
28 | REDIS_PASSWORD=null
29 | REDIS_PORT=6379
30 |
31 | MAIL_MAILER=smtp
32 | MAIL_HOST=mailpit
33 | MAIL_PORT=1025
34 | MAIL_USERNAME=null
35 | MAIL_PASSWORD=null
36 | MAIL_ENCRYPTION=null
37 | MAIL_FROM_ADDRESS="hello@example.com"
38 | MAIL_FROM_NAME="${APP_NAME}"
39 |
40 | AWS_ACCESS_KEY_ID=
41 | AWS_SECRET_ACCESS_KEY=
42 | AWS_DEFAULT_REGION=us-east-1
43 | AWS_BUCKET=
44 | AWS_USE_PATH_STYLE_ENDPOINT=false
45 |
46 | PUSHER_APP_ID=
47 | PUSHER_APP_KEY=
48 | PUSHER_APP_SECRET=
49 | PUSHER_HOST=
50 | PUSHER_PORT=443
51 | PUSHER_SCHEME=https
52 | PUSHER_APP_CLUSTER=mt1
53 |
54 | VITE_PUSHER_APP_KEY="${PUSHER_APP_KEY}"
55 | VITE_PUSHER_HOST="${PUSHER_HOST}"
56 | VITE_PUSHER_PORT="${PUSHER_PORT}"
57 | VITE_PUSHER_SCHEME="${PUSHER_SCHEME}"
58 | VITE_PUSHER_APP_CLUSTER="${PUSHER_APP_CLUSTER}"
59 |
60 | OPENAI_API_KEY= "YOUR_OPENAI_API_KEY"
--------------------------------------------------------------------------------
/resources/views/layouts/app.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | {{ config('app.name', 'Laravel') }}
10 |
11 |
12 |
13 |
14 |
15 |
16 | @vite(['resources/css/app.css', 'resources/js/app.js'])
17 | @livewireStyles
18 | @stack('styles')
19 |
20 |
21 |
22 |
23 | @include('layouts.navigation')
24 |
25 |
26 | @if (isset($header))
27 |
28 |
29 | {{ $header }}
30 |
31 |
32 | @endif
33 |
34 |
35 |
36 | {{ $slot }}
37 |
38 |
39 | @livewireScripts
40 |
41 |
42 | @stack('scripts')
43 |
44 |
45 |
--------------------------------------------------------------------------------
/app/Livewire/TranscribeBox.php:
--------------------------------------------------------------------------------
1 | 'en',
19 | 'German' => 'de',
20 | 'Greek' => 'el',
21 | 'Spanish' => 'es',
22 | 'French' => 'fr',
23 | 'Italian' => 'it',
24 | 'Portuguese' => 'pt',
25 | ];
26 |
27 | protected $openAIService;
28 |
29 | public $message = [];
30 |
31 | public function boot(openAIService $openAIService)
32 | {
33 | $this->openAIService = $openAIService;
34 | }
35 |
36 | public function transcribe()
37 | {
38 | $this->validate([
39 | 'file' => 'required|mimes:mp3,mp4,wav,mpeg,mpga,m4a,webm|max:25000',
40 | 'language' => 'required',
41 | ]);
42 | $filePath = $this->file->storeAs($this->file->getClientOriginalName());
43 | $response = $this->openAIService->transcribe($filePath, $this->availableLanguages[$this->language]);
44 | $this->message = [
45 | 'duration' => $response->duration,
46 | 'text' => $response->text,
47 | ];
48 | }
49 |
50 | public function render()
51 | {
52 | return view('livewire.transcribe-box.transcribe-box');
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/resources/views/auth/verify-email.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 | {{ __('Thanks for signing up! Before getting started, could you verify your email address by clicking on the link we just emailed to you? If you didn\'t receive the email, we will gladly send you another.') }}
4 |
5 |
6 | @if (session('status') == 'verification-link-sent')
7 |
8 | {{ __('A new verification link has been sent to the email address you provided during registration.') }}
9 |
10 | @endif
11 |
12 |
31 |
32 |
--------------------------------------------------------------------------------
/app/Providers/RouteServiceProvider.php:
--------------------------------------------------------------------------------
1 | configureRateLimiting();
28 |
29 | $this->routes(function () {
30 | Route::middleware('api')
31 | ->prefix('api')
32 | ->group(base_path('routes/api.php'));
33 |
34 | Route::middleware('web')
35 | ->group(base_path('routes/web.php'));
36 | });
37 | }
38 |
39 | /**
40 | * Configure the rate limiters for the application.
41 | */
42 | protected function configureRateLimiting(): void
43 | {
44 | RateLimiter::for('api', function (Request $request) {
45 | return Limit::perMinute(60)->by($request->user()?->id ?: $request->ip());
46 | });
47 | }
48 | }
49 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/PasswordResetLinkController.php:
--------------------------------------------------------------------------------
1 | validate([
29 | 'email' => ['required', 'email'],
30 | ]);
31 |
32 | // We will send the password reset link to this user. Once we have attempted
33 | // to send the link, we will examine the response then see the message we
34 | // need to show to the user. Finally, we'll send out a proper response.
35 | $status = Password::sendResetLink(
36 | $request->only('email')
37 | );
38 |
39 | return $status == Password::RESET_LINK_SENT
40 | ? back()->with('status', __($status))
41 | : back()->withInput($request->only('email'))
42 | ->withErrors(['email' => __($status)]);
43 | }
44 | }
45 |
--------------------------------------------------------------------------------
/resources/views/components/dropdown.blade.php:
--------------------------------------------------------------------------------
1 | @props(['align' => 'right', 'width' => '48', 'contentClasses' => 'py-1 bg-white dark:bg-gray-700'])
2 |
3 | @php
4 | switch ($align) {
5 | case 'left':
6 | $alignmentClasses = 'origin-top-left left-0';
7 | break;
8 | case 'top':
9 | $alignmentClasses = 'origin-top';
10 | break;
11 | case 'right':
12 | default:
13 | $alignmentClasses = 'origin-top-right right-0';
14 | break;
15 | }
16 |
17 | switch ($width) {
18 | case '48':
19 | $width = 'w-48';
20 | break;
21 | }
22 | @endphp
23 |
24 |
25 |
26 | {{ $trigger }}
27 |
28 |
29 |
39 |
40 | {{ $content }}
41 |
42 |
43 |
44 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/RegisteredUserController.php:
--------------------------------------------------------------------------------
1 | validate([
34 | 'name' => ['required', 'string', 'max:255'],
35 | 'email' => ['required', 'string', 'email', 'max:255', 'unique:'.User::class],
36 | 'password' => ['required', 'confirmed', Rules\Password::defaults()],
37 | ]);
38 |
39 | $user = User::create([
40 | 'name' => $request->name,
41 | 'email' => $request->email,
42 | 'password' => Hash::make($request->password),
43 | ]);
44 |
45 | event(new Registered($user));
46 |
47 | Auth::login($user);
48 |
49 | return redirect(RouteServiceProvider::HOME);
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/PasswordUpdateTest.php:
--------------------------------------------------------------------------------
1 | create();
17 |
18 | $response = $this
19 | ->actingAs($user)
20 | ->from('/profile')
21 | ->put('/password', [
22 | 'current_password' => 'password',
23 | 'password' => 'new-password',
24 | 'password_confirmation' => 'new-password',
25 | ]);
26 |
27 | $response
28 | ->assertSessionHasNoErrors()
29 | ->assertRedirect('/profile');
30 |
31 | $this->assertTrue(Hash::check('new-password', $user->refresh()->password));
32 | }
33 |
34 | public function test_correct_password_must_be_provided_to_update_password(): void
35 | {
36 | $user = User::factory()->create();
37 |
38 | $response = $this
39 | ->actingAs($user)
40 | ->from('/profile')
41 | ->put('/password', [
42 | 'current_password' => 'wrong-password',
43 | 'password' => 'new-password',
44 | 'password_confirmation' => 'new-password',
45 | ]);
46 |
47 | $response
48 | ->assertSessionHasErrorsIn('updatePassword', 'current_password')
49 | ->assertRedirect('/profile');
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/app/Http/Controllers/ProfileController.php:
--------------------------------------------------------------------------------
1 | $request->user(),
21 | ]);
22 | }
23 |
24 | /**
25 | * Update the user's profile information.
26 | */
27 | public function update(ProfileUpdateRequest $request): RedirectResponse
28 | {
29 | $request->user()->fill($request->validated());
30 |
31 | if ($request->user()->isDirty('email')) {
32 | $request->user()->email_verified_at = null;
33 | }
34 |
35 | $request->user()->save();
36 |
37 | return Redirect::route('profile.edit')->with('status', 'profile-updated');
38 | }
39 |
40 | /**
41 | * Delete the user's account.
42 | */
43 | public function destroy(Request $request): RedirectResponse
44 | {
45 | $request->validateWithBag('userDeletion', [
46 | 'password' => ['required', 'current-password'],
47 | ]);
48 |
49 | $user = $request->user();
50 |
51 | Auth::logout();
52 |
53 | $user->delete();
54 |
55 | $request->session()->invalidate();
56 | $request->session()->regenerateToken();
57 |
58 | return Redirect::to('/');
59 | }
60 | }
61 |
--------------------------------------------------------------------------------
/routes/web.php:
--------------------------------------------------------------------------------
1 | id())->paginate(10);
26 |
27 | return view('dashboard', [
28 | 'chatboxes' => $chatboxes,
29 | ]);
30 | })->middleware(['auth', 'verified'])->name('dashboard');
31 |
32 | Route::middleware('auth')->group(function () {
33 | Route::delete('/chatbox/{chatbox}', [ChatBoxController::class, 'destroy'])->name('chatbox.destroy');
34 | Route::get('/chatbox/{chatbox?}', [ChatBoxController::class, 'index'])->name('chatbox');
35 |
36 | Route::get('/wordpress', [WordpressController::class, 'index'])->name('wordpress');
37 |
38 | Route::get('/profile', [ProfileController::class, 'edit'])->name('profile.edit');
39 | Route::patch('/profile', [ProfileController::class, 'update'])->name('profile.update');
40 | Route::delete('/profile', [ProfileController::class, 'destroy'])->name('profile.destroy');
41 | });
42 |
43 | require __DIR__.'/auth.php';
44 |
--------------------------------------------------------------------------------
/config/hashing.php:
--------------------------------------------------------------------------------
1 | 'bcrypt',
19 |
20 | /*
21 | |--------------------------------------------------------------------------
22 | | Bcrypt Options
23 | |--------------------------------------------------------------------------
24 | |
25 | | Here you may specify the configuration options that should be used when
26 | | passwords are hashed using the Bcrypt algorithm. This will allow you
27 | | to control the amount of time it takes to hash the given password.
28 | |
29 | */
30 |
31 | 'bcrypt' => [
32 | 'rounds' => env('BCRYPT_ROUNDS', 10),
33 | ],
34 |
35 | /*
36 | |--------------------------------------------------------------------------
37 | | Argon Options
38 | |--------------------------------------------------------------------------
39 | |
40 | | Here you may specify the configuration options that should be used when
41 | | passwords are hashed using the Argon algorithm. These will allow you
42 | | to control the amount of time it takes to hash the given password.
43 | |
44 | */
45 |
46 | 'argon' => [
47 | 'memory' => 65536,
48 | 'threads' => 1,
49 | 'time' => 4,
50 | ],
51 |
52 | ];
53 |
--------------------------------------------------------------------------------
/resources/views/auth/reset-password.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
39 |
40 |
--------------------------------------------------------------------------------
/bootstrap/app.php:
--------------------------------------------------------------------------------
1 | singleton(
30 | Illuminate\Contracts\Http\Kernel::class,
31 | App\Http\Kernel::class
32 | );
33 |
34 | $app->singleton(
35 | Illuminate\Contracts\Console\Kernel::class,
36 | App\Console\Kernel::class
37 | );
38 |
39 | $app->singleton(
40 | Illuminate\Contracts\Debug\ExceptionHandler::class,
41 | App\Exceptions\Handler::class
42 | );
43 |
44 | /*
45 | |--------------------------------------------------------------------------
46 | | Return The Application
47 | |--------------------------------------------------------------------------
48 | |
49 | | This script returns the application instance. The instance is given to
50 | | the calling script so we can separate the building of the instances
51 | | from the actual running of the application and sending responses.
52 | |
53 | */
54 |
55 | return $app;
56 |
--------------------------------------------------------------------------------
/artisan:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env php
2 | make(Illuminate\Contracts\Console\Kernel::class);
34 |
35 | $status = $kernel->handle(
36 | $input = new Symfony\Component\Console\Input\ArgvInput,
37 | new Symfony\Component\Console\Output\ConsoleOutput
38 | );
39 |
40 | /*
41 | |--------------------------------------------------------------------------
42 | | Shutdown The Application
43 | |--------------------------------------------------------------------------
44 | |
45 | | Once Artisan has finished running, we will fire off the shutdown events
46 | | so that any final work may be done by the application before we shut
47 | | down the process. This is the last thing to happen to the request.
48 | |
49 | */
50 |
51 | $kernel->terminate($input, $status);
52 |
53 | exit($status);
54 |
--------------------------------------------------------------------------------
/public/index.php:
--------------------------------------------------------------------------------
1 | make(Kernel::class);
50 |
51 | $response = $kernel->handle(
52 | $request = Request::capture()
53 | )->send();
54 |
55 | $kernel->terminate($request, $response);
56 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/EmailVerificationTest.php:
--------------------------------------------------------------------------------
1 | create([
20 | 'email_verified_at' => null,
21 | ]);
22 |
23 | $response = $this->actingAs($user)->get('/verify-email');
24 |
25 | $response->assertStatus(200);
26 | }
27 |
28 | public function test_email_can_be_verified(): void
29 | {
30 | $user = User::factory()->create([
31 | 'email_verified_at' => null,
32 | ]);
33 |
34 | Event::fake();
35 |
36 | $verificationUrl = URL::temporarySignedRoute(
37 | 'verification.verify',
38 | now()->addMinutes(60),
39 | ['id' => $user->id, 'hash' => sha1($user->email)]
40 | );
41 |
42 | $response = $this->actingAs($user)->get($verificationUrl);
43 |
44 | Event::assertDispatched(Verified::class);
45 | $this->assertTrue($user->fresh()->hasVerifiedEmail());
46 | $response->assertRedirect(RouteServiceProvider::HOME.'?verified=1');
47 | }
48 |
49 | public function test_email_is_not_verified_with_invalid_hash(): void
50 | {
51 | $user = User::factory()->create([
52 | 'email_verified_at' => null,
53 | ]);
54 |
55 | $verificationUrl = URL::temporarySignedRoute(
56 | 'verification.verify',
57 | now()->addMinutes(60),
58 | ['id' => $user->id, 'hash' => sha1('wrong-email')]
59 | );
60 |
61 | $this->actingAs($user)->get($verificationUrl);
62 |
63 | $this->assertFalse($user->fresh()->hasVerifiedEmail());
64 | }
65 | }
66 |
--------------------------------------------------------------------------------
/app/Services/openAIService.php:
--------------------------------------------------------------------------------
1 | transcribe([
13 | 'model' => 'whisper-1',
14 | 'file' => fopen(storage_path('app/' . $filePath), 'r'),
15 | 'language' => $language,
16 | 'response_format' => 'verbose_json',
17 | ]);
18 |
19 | return $response;
20 | }
21 |
22 | public function ask($chatBoxModel, $chatBoxMaxTokens, $chatBoxTemperature, $transactions)
23 | {
24 | $response = OpenAI::chat()->create([
25 | 'model' => $chatBoxModel,
26 | 'messages' => $transactions,
27 | 'max_tokens' => (int) $chatBoxMaxTokens,
28 | 'temperature' => (float) $chatBoxTemperature,
29 | ]);
30 |
31 | return $response;
32 | }
33 |
34 | public function availableGPTModels()
35 | {
36 | return [
37 | 'gpt-3.5-turbo' => 'GPT-3.5 Turbo',
38 | 'gpt-3.5-turbo-16k' => 'GPT-3.5 Turbo 16k',
39 | 'gpt-4' => 'GPT-4',
40 | 'gpt-4-32k' => 'GPT-4 32k',
41 | 'gpt-4o' => 'GPT-4o',
42 | ];
43 | }
44 |
45 | public function availableGPTRoles()
46 | {
47 | $client = new Client();
48 | $response = $client->get('https://raw.githubusercontent.com/f/awesome-chatgpt-prompts/main/prompts.csv');
49 | $records = [];
50 | $headers = null;
51 | $csvString = $response->getBody();
52 | // Remove the first line and last line
53 | $csvString = substr($csvString, strpos($csvString, "\n") + 1);
54 | $csvString = substr($csvString, 0, strrpos($csvString, "\n"));
55 | $prompts = [];
56 |
57 | foreach (explode("\n", $csvString) as $line) {
58 | $values = str_getcsv($line);
59 |
60 | $promptName = trim($values[0], '"');
61 | $promptDescription = trim($values[1], '"');
62 |
63 | $prompts[$promptName] = $promptDescription;
64 | }
65 |
66 | return $prompts;
67 | }
68 | }
69 |
--------------------------------------------------------------------------------
/tests/Feature/Auth/PasswordResetTest.php:
--------------------------------------------------------------------------------
1 | get('/forgot-password');
18 |
19 | $response->assertStatus(200);
20 | }
21 |
22 | public function test_reset_password_link_can_be_requested(): void
23 | {
24 | Notification::fake();
25 |
26 | $user = User::factory()->create();
27 |
28 | $this->post('/forgot-password', ['email' => $user->email]);
29 |
30 | Notification::assertSentTo($user, ResetPassword::class);
31 | }
32 |
33 | public function test_reset_password_screen_can_be_rendered(): void
34 | {
35 | Notification::fake();
36 |
37 | $user = User::factory()->create();
38 |
39 | $this->post('/forgot-password', ['email' => $user->email]);
40 |
41 | Notification::assertSentTo($user, ResetPassword::class, function ($notification) {
42 | $response = $this->get('/reset-password/'.$notification->token);
43 |
44 | $response->assertStatus(200);
45 |
46 | return true;
47 | });
48 | }
49 |
50 | public function test_password_can_be_reset_with_valid_token(): void
51 | {
52 | Notification::fake();
53 |
54 | $user = User::factory()->create();
55 |
56 | $this->post('/forgot-password', ['email' => $user->email]);
57 |
58 | Notification::assertSentTo($user, ResetPassword::class, function ($notification) use ($user) {
59 | $response = $this->post('/reset-password', [
60 | 'token' => $notification->token,
61 | 'email' => $user->email,
62 | 'password' => 'password',
63 | 'password_confirmation' => 'password',
64 | ]);
65 |
66 | $response->assertSessionHasNoErrors();
67 |
68 | return true;
69 | });
70 | }
71 | }
72 |
--------------------------------------------------------------------------------
/resources/views/chatbox/index.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | {{ __('Chat Box') }}
5 |
6 |
7 |
8 |
9 |
10 |
11 |
13 |
14 | Chat
17 |
18 |
19 | Transcribe
23 |
24 |
25 |
26 |
27 |
29 |
30 |
31 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
--------------------------------------------------------------------------------
/resources/views/profile/partials/update-password-form.blade.php:
--------------------------------------------------------------------------------
1 |
49 |
--------------------------------------------------------------------------------
/resources/views/auth/login.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
47 |
48 |
--------------------------------------------------------------------------------
/config/broadcasting.php:
--------------------------------------------------------------------------------
1 | env('BROADCAST_DRIVER', 'null'),
19 |
20 | /*
21 | |--------------------------------------------------------------------------
22 | | Broadcast Connections
23 | |--------------------------------------------------------------------------
24 | |
25 | | Here you may define all of the broadcast connections that will be used
26 | | to broadcast events to other systems or over websockets. Samples of
27 | | each available type of connection are provided inside this array.
28 | |
29 | */
30 |
31 | 'connections' => [
32 |
33 | 'pusher' => [
34 | 'driver' => 'pusher',
35 | 'key' => env('PUSHER_APP_KEY'),
36 | 'secret' => env('PUSHER_APP_SECRET'),
37 | 'app_id' => env('PUSHER_APP_ID'),
38 | 'options' => [
39 | 'host' => env('PUSHER_HOST') ?: 'api-'.env('PUSHER_APP_CLUSTER', 'mt1').'.pusher.com',
40 | 'port' => env('PUSHER_PORT', 443),
41 | 'scheme' => env('PUSHER_SCHEME', 'https'),
42 | 'encrypted' => true,
43 | 'useTLS' => env('PUSHER_SCHEME', 'https') === 'https',
44 | ],
45 | 'client_options' => [
46 | // Guzzle client options: https://docs.guzzlephp.org/en/stable/request-options.html
47 | ],
48 | ],
49 |
50 | 'ably' => [
51 | 'driver' => 'ably',
52 | 'key' => env('ABLY_KEY'),
53 | ],
54 |
55 | 'redis' => [
56 | 'driver' => 'redis',
57 | 'connection' => 'default',
58 | ],
59 |
60 | 'log' => [
61 | 'driver' => 'log',
62 | ],
63 |
64 | 'null' => [
65 | 'driver' => 'null',
66 | ],
67 |
68 | ],
69 |
70 | ];
71 |
--------------------------------------------------------------------------------
/composer.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "laravel/laravel",
3 | "type": "project",
4 | "description": "The Laravel Framework.",
5 | "keywords": ["framework", "laravel"],
6 | "license": "MIT",
7 | "require": {
8 | "php": "^8.2",
9 | "guzzlehttp/guzzle": "^7.2",
10 | "jantinnerezo/livewire-alert": "^3.0",
11 | "laravel/framework": "^11.0",
12 | "laravel/prompts": "^0.1.13",
13 | "laravel/sanctum": "^4.0",
14 | "laravel/tinker": "^2.8",
15 | "league/commonmark": "^2.3",
16 | "livewire/livewire": "^3.0",
17 | "nesbot/carbon": "^2.72",
18 | "openai-php/laravel": "^0.10.0"
19 | },
20 | "require-dev": {
21 | "fakerphp/faker": "^1.9.1",
22 | "laravel/breeze": "^2.0",
23 | "laravel/pint": "^1.6",
24 | "laravel/sail": "^1.18",
25 | "mockery/mockery": "^1.4.4",
26 | "nunomaduro/collision": "^8.1",
27 | "phpunit/phpunit": "^10.0",
28 | "spatie/laravel-ignition": "^2.0"
29 | },
30 | "autoload": {
31 | "psr-4": {
32 | "App\\": "app/",
33 | "Database\\Factories\\": "database/factories/",
34 | "Database\\Seeders\\": "database/seeders/"
35 | }
36 | },
37 | "autoload-dev": {
38 | "psr-4": {
39 | "Tests\\": "tests/"
40 | }
41 | },
42 | "scripts": {
43 | "post-autoload-dump": [
44 | "Illuminate\\Foundation\\ComposerScripts::postAutoloadDump",
45 | "@php artisan package:discover --ansi"
46 | ],
47 | "post-update-cmd": [
48 | "@php artisan vendor:publish --tag=laravel-assets --ansi --force"
49 | ],
50 | "post-root-package-install": [
51 | "@php -r \"file_exists('.env') || copy('.env.example', '.env');\""
52 | ],
53 | "post-create-project-cmd": [
54 | "@php artisan key:generate --ansi"
55 | ]
56 | },
57 | "extra": {
58 | "laravel": {
59 | "dont-discover": []
60 | }
61 | },
62 | "config": {
63 | "optimize-autoloader": true,
64 | "preferred-install": "dist",
65 | "sort-packages": true,
66 | "allow-plugins": {
67 | "pestphp/pest-plugin": true,
68 | "php-http/discovery": true
69 | }
70 | },
71 | "minimum-stability": "stable",
72 | "prefer-stable": true
73 | }
74 |
--------------------------------------------------------------------------------
/app/Http/Controllers/Auth/NewPasswordController.php:
--------------------------------------------------------------------------------
1 | $request]);
23 | }
24 |
25 | /**
26 | * Handle an incoming new password request.
27 | *
28 | * @throws \Illuminate\Validation\ValidationException
29 | */
30 | public function store(Request $request): RedirectResponse
31 | {
32 | $request->validate([
33 | 'token' => ['required'],
34 | 'email' => ['required', 'email'],
35 | 'password' => ['required', 'confirmed', Rules\Password::defaults()],
36 | ]);
37 |
38 | // Here we will attempt to reset the user's password. If it is successful we
39 | // will update the password on an actual user model and persist it to the
40 | // database. Otherwise we will parse the error and return the response.
41 | $status = Password::reset(
42 | $request->only('email', 'password', 'password_confirmation', 'token'),
43 | function ($user) use ($request) {
44 | $user->forceFill([
45 | 'password' => Hash::make($request->password),
46 | 'remember_token' => Str::random(60),
47 | ])->save();
48 |
49 | event(new PasswordReset($user));
50 | }
51 | );
52 |
53 | // If the password was successfully reset, we will redirect the user back to
54 | // the application's home authenticated view. If there is an error we can
55 | // redirect them back to where they came from with their error message.
56 | return $status == Password::PASSWORD_RESET
57 | ? redirect()->route('login')->with('status', __($status))
58 | : back()->withInput($request->only('email'))
59 | ->withErrors(['email' => __($status)]);
60 | }
61 | }
62 |
--------------------------------------------------------------------------------
/resources/views/profile/partials/delete-user-form.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | {{ __('Delete Account') }}
5 |
6 |
7 |
8 | {{ __('Once your account is deleted, all of its resources and data will be permanently deleted. Before deleting your account, please download any data or information that you wish to retain.') }}
9 |
10 |
11 |
12 | {{ __('Delete Account') }}
16 |
17 |
18 |
54 |
55 |
56 |
--------------------------------------------------------------------------------
/resources/views/auth/register.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
52 |
53 |
--------------------------------------------------------------------------------
/config/sanctum.php:
--------------------------------------------------------------------------------
1 | explode(',', env('SANCTUM_STATEFUL_DOMAINS', sprintf(
19 | '%s%s',
20 | 'localhost,localhost:3000,127.0.0.1,127.0.0.1:8000,::1',
21 | Sanctum::currentApplicationUrlWithPort()
22 | ))),
23 |
24 | /*
25 | |--------------------------------------------------------------------------
26 | | Sanctum Guards
27 | |--------------------------------------------------------------------------
28 | |
29 | | This array contains the authentication guards that will be checked when
30 | | Sanctum is trying to authenticate a request. If none of these guards
31 | | are able to authenticate the request, Sanctum will use the bearer
32 | | token that's present on an incoming request for authentication.
33 | |
34 | */
35 |
36 | 'guard' => ['web'],
37 |
38 | /*
39 | |--------------------------------------------------------------------------
40 | | Expiration Minutes
41 | |--------------------------------------------------------------------------
42 | |
43 | | This value controls the number of minutes until an issued token will be
44 | | considered expired. If this value is null, personal access tokens do
45 | | not expire. This won't tweak the lifetime of first-party sessions.
46 | |
47 | */
48 |
49 | 'expiration' => null,
50 |
51 | /*
52 | |--------------------------------------------------------------------------
53 | | Sanctum Middleware
54 | |--------------------------------------------------------------------------
55 | |
56 | | When authenticating your first-party SPA with Sanctum you may need to
57 | | customize some of the middleware Sanctum uses while processing the
58 | | request. You may change the middleware listed below as required.
59 | |
60 | */
61 |
62 | 'middleware' => [
63 | 'verify_csrf_token' => App\Http\Middleware\VerifyCsrfToken::class,
64 | 'encrypt_cookies' => App\Http\Middleware\EncryptCookies::class,
65 | ],
66 |
67 | ];
68 |
--------------------------------------------------------------------------------
/routes/auth.php:
--------------------------------------------------------------------------------
1 | group(function () {
15 | Route::get('register', [RegisteredUserController::class, 'create'])
16 | ->name('register');
17 |
18 | Route::post('register', [RegisteredUserController::class, 'store']);
19 |
20 | Route::get('login', [AuthenticatedSessionController::class, 'create'])
21 | ->name('login');
22 |
23 | Route::post('login', [AuthenticatedSessionController::class, 'store']);
24 |
25 | Route::get('forgot-password', [PasswordResetLinkController::class, 'create'])
26 | ->name('password.request');
27 |
28 | Route::post('forgot-password', [PasswordResetLinkController::class, 'store'])
29 | ->name('password.email');
30 |
31 | Route::get('reset-password/{token}', [NewPasswordController::class, 'create'])
32 | ->name('password.reset');
33 |
34 | Route::post('reset-password', [NewPasswordController::class, 'store'])
35 | ->name('password.store');
36 | });
37 |
38 | Route::middleware('auth')->group(function () {
39 | Route::get('verify-email', EmailVerificationPromptController::class)
40 | ->name('verification.notice');
41 |
42 | Route::get('verify-email/{id}/{hash}', VerifyEmailController::class)
43 | ->middleware(['signed', 'throttle:6,1'])
44 | ->name('verification.verify');
45 |
46 | Route::post('email/verification-notification', [EmailVerificationNotificationController::class, 'store'])
47 | ->middleware('throttle:6,1')
48 | ->name('verification.send');
49 |
50 | Route::get('confirm-password', [ConfirmablePasswordController::class, 'show'])
51 | ->name('password.confirm');
52 |
53 | Route::post('confirm-password', [ConfirmablePasswordController::class, 'store']);
54 |
55 | Route::put('password', [PasswordController::class, 'update'])->name('password.update');
56 |
57 | Route::post('logout', [AuthenticatedSessionController::class, 'destroy'])
58 | ->name('logout');
59 | });
60 |
--------------------------------------------------------------------------------
/app/Models/Wordpress.php:
--------------------------------------------------------------------------------
1 | request('GET', $url . '/wp-json/wp/v2/posts');
20 |
21 | //Get the number of posts
22 | $totalPosts = $response->getHeader('X-WP-Total')[0];
23 |
24 | //Return the total number of posts and posts
25 | return $totalPosts;
26 | }
27 |
28 | public static function getPostsPerPage($url, $page = 1, $perPage = 10)
29 | {
30 | //Create a new guzzle client
31 | $client = new Client();
32 |
33 | //Get the wordpress posts using the wordpress api
34 | $response = $client->request('GET', $url . '/wp-json/wp/v2/posts', [
35 | 'query' => [
36 | 'page' => $page,
37 | 'per_page' => $perPage,
38 | ],
39 | ]);
40 |
41 | //Get the response body
42 | $posts = $response->getBody();
43 |
44 | //Decode the response body
45 | $posts = json_decode($posts);
46 |
47 | //Return the posts
48 | // dd($posts);
49 | return $posts;
50 | }
51 |
52 | public static function createPost($url, $title, $body, $status, $username, $password)
53 | {
54 | // dd($url, $title, $body, $status, $username, $password);
55 | //Create a new guzzle client
56 | $client = new Client();
57 |
58 | $payload = [
59 | 'title' => $title,
60 | 'content' => $body,
61 | 'status' => $status,
62 | // Add any other required fields
63 | ];
64 |
65 | //Try to create a new post using the wordpress api
66 | try {
67 | $response = $client->post($url . '/wp-json/wp/v2/posts', [
68 | // 'auth' => [$username, $password],
69 | 'headers' => [
70 | 'Content-Type' => 'application/json',
71 | 'Authorization' => 'Basic ' . base64_encode($username . ':' . $password),
72 | ],
73 | 'json' => $payload,
74 | ]);
75 |
76 | return true;
77 | } catch (\Exception $e) {
78 | return false;
79 | }
80 | }
81 | }
82 |
--------------------------------------------------------------------------------
/app/Http/Requests/Auth/LoginRequest.php:
--------------------------------------------------------------------------------
1 |
26 | */
27 | public function rules(): array
28 | {
29 | return [
30 | 'email' => ['required', 'string', 'email'],
31 | 'password' => ['required', 'string'],
32 | ];
33 | }
34 |
35 | /**
36 | * Attempt to authenticate the request's credentials.
37 | *
38 | * @throws \Illuminate\Validation\ValidationException
39 | */
40 | public function authenticate(): void
41 | {
42 | $this->ensureIsNotRateLimited();
43 |
44 | if (! Auth::attempt($this->only('email', 'password'), $this->boolean('remember'))) {
45 | RateLimiter::hit($this->throttleKey());
46 |
47 | throw ValidationException::withMessages([
48 | 'email' => trans('auth.failed'),
49 | ]);
50 | }
51 |
52 | RateLimiter::clear($this->throttleKey());
53 | }
54 |
55 | /**
56 | * Ensure the login request is not rate limited.
57 | *
58 | * @throws \Illuminate\Validation\ValidationException
59 | */
60 | public function ensureIsNotRateLimited(): void
61 | {
62 | if (! RateLimiter::tooManyAttempts($this->throttleKey(), 5)) {
63 | return;
64 | }
65 |
66 | event(new Lockout($this));
67 |
68 | $seconds = RateLimiter::availableIn($this->throttleKey());
69 |
70 | throw ValidationException::withMessages([
71 | 'email' => trans('auth.throttle', [
72 | 'seconds' => $seconds,
73 | 'minutes' => ceil($seconds / 60),
74 | ]),
75 | ]);
76 | }
77 |
78 | /**
79 | * Get the rate limiting throttle key for the request.
80 | */
81 | public function throttleKey(): string
82 | {
83 | return Str::transliterate(Str::lower($this->input('email')).'|'.$this->ip());
84 | }
85 | }
86 |
--------------------------------------------------------------------------------
/config/filesystems.php:
--------------------------------------------------------------------------------
1 | env('FILESYSTEM_DISK', 'local'),
17 |
18 | /*
19 | |--------------------------------------------------------------------------
20 | | Filesystem Disks
21 | |--------------------------------------------------------------------------
22 | |
23 | | Here you may configure as many filesystem "disks" as you wish, and you
24 | | may even configure multiple disks of the same driver. Defaults have
25 | | been set up for each driver as an example of the required values.
26 | |
27 | | Supported Drivers: "local", "ftp", "sftp", "s3"
28 | |
29 | */
30 |
31 | 'disks' => [
32 |
33 | 'local' => [
34 | 'driver' => 'local',
35 | 'root' => storage_path('app'),
36 | 'throw' => false,
37 | ],
38 |
39 | 'public' => [
40 | 'driver' => 'local',
41 | 'root' => storage_path('app/public'),
42 | 'url' => env('APP_URL').'/storage',
43 | 'visibility' => 'public',
44 | 'throw' => false,
45 | ],
46 |
47 | 's3' => [
48 | 'driver' => 's3',
49 | 'key' => env('AWS_ACCESS_KEY_ID'),
50 | 'secret' => env('AWS_SECRET_ACCESS_KEY'),
51 | 'region' => env('AWS_DEFAULT_REGION'),
52 | 'bucket' => env('AWS_BUCKET'),
53 | 'url' => env('AWS_URL'),
54 | 'endpoint' => env('AWS_ENDPOINT'),
55 | 'use_path_style_endpoint' => env('AWS_USE_PATH_STYLE_ENDPOINT', false),
56 | 'throw' => false,
57 | ],
58 |
59 | ],
60 |
61 | /*
62 | |--------------------------------------------------------------------------
63 | | Symbolic Links
64 | |--------------------------------------------------------------------------
65 | |
66 | | Here you may configure the symbolic links that will be created when the
67 | | `storage:link` Artisan command is executed. The array keys should be
68 | | the locations of the links and the values should be their targets.
69 | |
70 | */
71 |
72 | 'links' => [
73 | public_path('storage') => storage_path('app/public'),
74 | ],
75 |
76 | ];
77 |
--------------------------------------------------------------------------------
/app/Livewire/WordpressSeo.php:
--------------------------------------------------------------------------------
1 | openAIService = $openAIService;
40 | }
41 |
42 | public function load()
43 | {
44 | $this->validate([
45 | 'url' => 'required|url',
46 | ]);
47 | $response = WordpressModel::getPostsPerPage($this->url, 1, 1)[0];
48 |
49 | $this->firstPost = 'title: ' . $response->title->rendered . PHP_EOL . 'body: ' . $response->content->rendered;
50 | }
51 |
52 | public function ask()
53 | {
54 | $this->transactions[] = ['role' => 'system', 'content' => $this->chatBoxSystemInstruction];
55 | // If the user has typed something, then asking the ChatGPT API
56 | if (!empty($this->firstPost)) {
57 | $this->transactions[] = ['role' => 'user', 'content' => $this->firstPost];
58 | $response = $this->openAIService->ask(
59 | $this->chatBoxModel,
60 | $this->chatBoxMaxTokens,
61 | $this->chatBoxTemperature,
62 | $this->transactions
63 | );
64 | $this->totalTokens = $response->usage->totalTokens;
65 | $this->transactions[] = ['role' => 'assistant', 'content' => $response->choices[0]->message->content];
66 | $this->messages = collect($this->transactions)->reject(fn ($message) => $message['role'] === 'system');
67 | // dd($this->messages);
68 | }
69 | }
70 |
71 | public function resetUrl()
72 | {
73 | return redirect()->route('wordpress');
74 | }
75 |
76 | public function render()
77 | {
78 | return view('livewire.wordpress.seo');
79 | }
80 | }
81 |
--------------------------------------------------------------------------------
/resources/views/livewire/transcribe-box/transcribe-box.blade.php:
--------------------------------------------------------------------------------
1 |
2 | @if ($message)
3 |
4 | Your
5 | transcription
6 |
8 |
9 | @endif
10 |
11 |
Select a
12 | language
13 |
15 | Choose a Language
16 | @foreach ($availableLanguages as $key => $value)
17 | {{ $key }}
18 | @endforeach
19 |
20 |
21 | @error('language')
22 |
{{ $message }}
23 | @enderror
24 |
25 |
26 |
29 |
Please select an audio file
30 | (MP3, MP4, WAV, MPEG, MPGA, M4A, WEBM) with a maximum size of 25 MB.
31 |
32 | @error('file')
33 |
{{ $message }}
34 | @enderror
35 |
36 |
37 |
39 | Transcribe
40 |
41 |
42 |
43 |
--------------------------------------------------------------------------------
/app/Http/Kernel.php:
--------------------------------------------------------------------------------
1 |
15 | */
16 | protected $middleware = [
17 | // \App\Http\Middleware\TrustHosts::class,
18 | \App\Http\Middleware\TrustProxies::class,
19 | \Illuminate\Http\Middleware\HandleCors::class,
20 | \App\Http\Middleware\PreventRequestsDuringMaintenance::class,
21 | \Illuminate\Foundation\Http\Middleware\ValidatePostSize::class,
22 | \App\Http\Middleware\TrimStrings::class,
23 | \Illuminate\Foundation\Http\Middleware\ConvertEmptyStringsToNull::class,
24 | ];
25 |
26 | /**
27 | * The application's route middleware groups.
28 | *
29 | * @var array>
30 | */
31 | protected $middlewareGroups = [
32 | 'web' => [
33 | \App\Http\Middleware\EncryptCookies::class,
34 | \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
35 | \Illuminate\Session\Middleware\StartSession::class,
36 | \Illuminate\View\Middleware\ShareErrorsFromSession::class,
37 | \App\Http\Middleware\VerifyCsrfToken::class,
38 | \Illuminate\Routing\Middleware\SubstituteBindings::class,
39 | ],
40 |
41 | 'api' => [
42 | // \Laravel\Sanctum\Http\Middleware\EnsureFrontendRequestsAreStateful::class,
43 | \Illuminate\Routing\Middleware\ThrottleRequests::class.':api',
44 | \Illuminate\Routing\Middleware\SubstituteBindings::class,
45 | ],
46 | ];
47 |
48 | /**
49 | * The application's middleware aliases.
50 | *
51 | * Aliases may be used to conveniently assign middleware to routes and groups.
52 | *
53 | * @var array
54 | */
55 | protected $middlewareAliases = [
56 | 'auth' => \App\Http\Middleware\Authenticate::class,
57 | 'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
58 | 'auth.session' => \Illuminate\Session\Middleware\AuthenticateSession::class,
59 | 'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class,
60 | 'can' => \Illuminate\Auth\Middleware\Authorize::class,
61 | 'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
62 | 'password.confirm' => \Illuminate\Auth\Middleware\RequirePassword::class,
63 | 'signed' => \App\Http\Middleware\ValidateSignature::class,
64 | 'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class,
65 | 'verified' => \Illuminate\Auth\Middleware\EnsureEmailIsVerified::class,
66 | ];
67 | }
68 |
--------------------------------------------------------------------------------
/resources/views/components/application-logo.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/tests/Feature/ProfileTest.php:
--------------------------------------------------------------------------------
1 | create();
16 |
17 | $response = $this
18 | ->actingAs($user)
19 | ->get('/profile');
20 |
21 | $response->assertOk();
22 | }
23 |
24 | public function test_profile_information_can_be_updated(): void
25 | {
26 | $user = User::factory()->create();
27 |
28 | $response = $this
29 | ->actingAs($user)
30 | ->patch('/profile', [
31 | 'name' => 'Test User',
32 | 'email' => 'test@example.com',
33 | ]);
34 |
35 | $response
36 | ->assertSessionHasNoErrors()
37 | ->assertRedirect('/profile');
38 |
39 | $user->refresh();
40 |
41 | $this->assertSame('Test User', $user->name);
42 | $this->assertSame('test@example.com', $user->email);
43 | $this->assertNull($user->email_verified_at);
44 | }
45 |
46 | public function test_email_verification_status_is_unchanged_when_the_email_address_is_unchanged(): void
47 | {
48 | $user = User::factory()->create();
49 |
50 | $response = $this
51 | ->actingAs($user)
52 | ->patch('/profile', [
53 | 'name' => 'Test User',
54 | 'email' => $user->email,
55 | ]);
56 |
57 | $response
58 | ->assertSessionHasNoErrors()
59 | ->assertRedirect('/profile');
60 |
61 | $this->assertNotNull($user->refresh()->email_verified_at);
62 | }
63 |
64 | public function test_user_can_delete_their_account(): void
65 | {
66 | $user = User::factory()->create();
67 |
68 | $response = $this
69 | ->actingAs($user)
70 | ->delete('/profile', [
71 | 'password' => 'password',
72 | ]);
73 |
74 | $response
75 | ->assertSessionHasNoErrors()
76 | ->assertRedirect('/');
77 |
78 | $this->assertGuest();
79 | $this->assertNull($user->fresh());
80 | }
81 |
82 | public function test_correct_password_must_be_provided_to_delete_account(): void
83 | {
84 | $user = User::factory()->create();
85 |
86 | $response = $this
87 | ->actingAs($user)
88 | ->from('/profile')
89 | ->delete('/profile', [
90 | 'password' => 'wrong-password',
91 | ]);
92 |
93 | $response
94 | ->assertSessionHasErrorsIn('userDeletion', 'password')
95 | ->assertRedirect('/profile');
96 |
97 | $this->assertNotNull($user->fresh());
98 | }
99 | }
100 |
--------------------------------------------------------------------------------
/resources/views/profile/partials/update-profile-information-form.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
11 |
12 |
15 |
16 |
64 |
65 |
--------------------------------------------------------------------------------
/config/queue.php:
--------------------------------------------------------------------------------
1 | env('QUEUE_CONNECTION', 'sync'),
17 |
18 | /*
19 | |--------------------------------------------------------------------------
20 | | Queue Connections
21 | |--------------------------------------------------------------------------
22 | |
23 | | Here you may configure the connection information for each server that
24 | | is used by your application. A default configuration has been added
25 | | for each back-end shipped with Laravel. You are free to add more.
26 | |
27 | | Drivers: "sync", "database", "beanstalkd", "sqs", "redis", "null"
28 | |
29 | */
30 |
31 | 'connections' => [
32 |
33 | 'sync' => [
34 | 'driver' => 'sync',
35 | ],
36 |
37 | 'database' => [
38 | 'driver' => 'database',
39 | 'table' => 'jobs',
40 | 'queue' => 'default',
41 | 'retry_after' => 90,
42 | 'after_commit' => false,
43 | ],
44 |
45 | 'beanstalkd' => [
46 | 'driver' => 'beanstalkd',
47 | 'host' => 'localhost',
48 | 'queue' => 'default',
49 | 'retry_after' => 90,
50 | 'block_for' => 0,
51 | 'after_commit' => false,
52 | ],
53 |
54 | 'sqs' => [
55 | 'driver' => 'sqs',
56 | 'key' => env('AWS_ACCESS_KEY_ID'),
57 | 'secret' => env('AWS_SECRET_ACCESS_KEY'),
58 | 'prefix' => env('SQS_PREFIX', 'https://sqs.us-east-1.amazonaws.com/your-account-id'),
59 | 'queue' => env('SQS_QUEUE', 'default'),
60 | 'suffix' => env('SQS_SUFFIX'),
61 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'),
62 | 'after_commit' => false,
63 | ],
64 |
65 | 'redis' => [
66 | 'driver' => 'redis',
67 | 'connection' => 'default',
68 | 'queue' => env('REDIS_QUEUE', 'default'),
69 | 'retry_after' => 90,
70 | 'block_for' => null,
71 | 'after_commit' => false,
72 | ],
73 |
74 | ],
75 |
76 | /*
77 | |--------------------------------------------------------------------------
78 | | Failed Queue Jobs
79 | |--------------------------------------------------------------------------
80 | |
81 | | These options configure the behavior of failed queue job logging so you
82 | | can control which database and table are used to store the jobs that
83 | | have failed. You may change them to any database / table you wish.
84 | |
85 | */
86 |
87 | 'failed' => [
88 | 'driver' => env('QUEUE_FAILED_DRIVER', 'database-uuids'),
89 | 'database' => env('DB_CONNECTION', 'mysql'),
90 | 'table' => 'failed_jobs',
91 | ],
92 |
93 | ];
94 |
--------------------------------------------------------------------------------
/config/markdown.php:
--------------------------------------------------------------------------------
1 | [
5 | /*
6 | * To highlight code, we'll use Shiki under the hood. Make sure it's installed.
7 | *
8 | * More info: https://spatie.be/docs/laravel-markdown/v1/installation-setup
9 | */
10 | 'enabled' => true,
11 |
12 | /*
13 | * The name of or path to a Shiki theme
14 | *
15 | * More info: https://github.com/shikijs/shiki/blob/main/docs/themes.md
16 | */
17 | 'theme' => 'light-plus',
18 | ],
19 |
20 | /*
21 | * When enabled, anchor links will be added to all titles
22 | */
23 | 'add_anchors_to_headings' => true,
24 |
25 | /*
26 | * These options will be passed to the league/commonmark package which is
27 | * used under the hood to render markdown.
28 | *
29 | * More info: https://spatie.be/docs/laravel-markdown/v1/using-the-blade-component/passing-options-to-commonmark
30 | */
31 | 'commonmark_options' => [],
32 |
33 | /*
34 | * Rendering markdown to HTML can be resource intensive. By default
35 | * we'll cache the results.
36 | *
37 | * You can specify the name of a cache store here. When set to `null`
38 | * the default cache store will be used. If you do not want to use
39 | * caching set this value to `false`.
40 | */
41 | 'cache_store' => null,
42 |
43 | /*
44 | * This class will convert markdown to HTML
45 | *
46 | * You can change this to a class of your own to greatly
47 | * customize the rendering process
48 | *
49 | * More info: https://spatie.be/docs/laravel-markdown/v1/advanced-usage/customizing-the-rendering-process
50 | */
51 | 'renderer_class' => Spatie\LaravelMarkdown\MarkdownRenderer::class,
52 |
53 | /*
54 | * These extensions should be added to the markdown environment. A valid
55 | * extension implements League\CommonMark\Extension\ExtensionInterface
56 | *
57 | * More info: https://commonmark.thephpleague.com/2.1/extensions/overview/
58 | */
59 | 'extensions' => [
60 | //
61 | ],
62 |
63 | /*
64 | * These block renderers should be added to the markdown environment. A valid
65 | * renderer implements League\CommonMark\Renderer\NodeRendererInterface;
66 | *
67 | * More info: https://commonmark.thephpleague.com/2.1/customization/rendering/
68 | */
69 | 'block_renderers' => [
70 | // ['class' => FencedCode::class, 'renderer' => new MyCustomCodeRenderer(), 'priority' => 0]
71 | ],
72 |
73 | /*
74 | * These inline renderers should be added to the markdown environment. A valid
75 | * renderer implements League\CommonMark\Renderer\NodeRendererInterface;
76 | *
77 | * More info: https://commonmark.thephpleague.com/2.1/customization/rendering/
78 | */
79 | 'inline_renderers' => [
80 | // ['class' => FencedCode::class, 'renderer' => new MyCustomCodeRenderer(), 'priority' => 0]
81 | ],
82 |
83 | /*
84 | * These inline parsers should be added to the markdown environment. A valid
85 | * parser implements League\CommonMark\Renderer\InlineParserInterface;
86 | *
87 | * More info: https://commonmark.thephpleague.com/2.3/customization/inline-parsing/
88 | */
89 | 'inline_parsers' => [
90 | // ['parser' => new MyCustomInlineParser(), 'priority' => 0]
91 | ],
92 | ];
93 |
--------------------------------------------------------------------------------
/resources/views/components/modal.blade.php:
--------------------------------------------------------------------------------
1 | @props([
2 | 'name',
3 | 'show' => false,
4 | 'maxWidth' => '2xl'
5 | ])
6 |
7 | @php
8 | $maxWidth = [
9 | 'sm' => 'sm:max-w-sm',
10 | 'md' => 'sm:max-w-md',
11 | 'lg' => 'sm:max-w-lg',
12 | 'xl' => 'sm:max-w-xl',
13 | '2xl' => 'sm:max-w-2xl',
14 | ][$maxWidth];
15 | @endphp
16 |
17 |
51 |
64 |
65 |
75 | {{ $slot }}
76 |
77 |
78 |
--------------------------------------------------------------------------------
/config/cache.php:
--------------------------------------------------------------------------------
1 | env('CACHE_DRIVER', 'file'),
19 |
20 | /*
21 | |--------------------------------------------------------------------------
22 | | Cache Stores
23 | |--------------------------------------------------------------------------
24 | |
25 | | Here you may define all of the cache "stores" for your application as
26 | | well as their drivers. You may even define multiple stores for the
27 | | same cache driver to group types of items stored in your caches.
28 | |
29 | | Supported drivers: "apc", "array", "database", "file",
30 | | "memcached", "redis", "dynamodb", "octane", "null"
31 | |
32 | */
33 |
34 | 'stores' => [
35 |
36 | 'apc' => [
37 | 'driver' => 'apc',
38 | ],
39 |
40 | 'array' => [
41 | 'driver' => 'array',
42 | 'serialize' => false,
43 | ],
44 |
45 | 'database' => [
46 | 'driver' => 'database',
47 | 'table' => 'cache',
48 | 'connection' => null,
49 | 'lock_connection' => null,
50 | ],
51 |
52 | 'file' => [
53 | 'driver' => 'file',
54 | 'path' => storage_path('framework/cache/data'),
55 | ],
56 |
57 | 'memcached' => [
58 | 'driver' => 'memcached',
59 | 'persistent_id' => env('MEMCACHED_PERSISTENT_ID'),
60 | 'sasl' => [
61 | env('MEMCACHED_USERNAME'),
62 | env('MEMCACHED_PASSWORD'),
63 | ],
64 | 'options' => [
65 | // Memcached::OPT_CONNECT_TIMEOUT => 2000,
66 | ],
67 | 'servers' => [
68 | [
69 | 'host' => env('MEMCACHED_HOST', '127.0.0.1'),
70 | 'port' => env('MEMCACHED_PORT', 11211),
71 | 'weight' => 100,
72 | ],
73 | ],
74 | ],
75 |
76 | 'redis' => [
77 | 'driver' => 'redis',
78 | 'connection' => 'cache',
79 | 'lock_connection' => 'default',
80 | ],
81 |
82 | 'dynamodb' => [
83 | 'driver' => 'dynamodb',
84 | 'key' => env('AWS_ACCESS_KEY_ID'),
85 | 'secret' => env('AWS_SECRET_ACCESS_KEY'),
86 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'),
87 | 'table' => env('DYNAMODB_CACHE_TABLE', 'cache'),
88 | 'endpoint' => env('DYNAMODB_ENDPOINT'),
89 | ],
90 |
91 | 'octane' => [
92 | 'driver' => 'octane',
93 | ],
94 |
95 | ],
96 |
97 | /*
98 | |--------------------------------------------------------------------------
99 | | Cache Key Prefix
100 | |--------------------------------------------------------------------------
101 | |
102 | | When utilizing the APC, database, memcached, Redis, or DynamoDB cache
103 | | stores there might be other applications using the same cache. For
104 | | that reason, you may prefix every cache key to avoid collisions.
105 | |
106 | */
107 |
108 | 'prefix' => env('CACHE_PREFIX', Str::slug(env('APP_NAME', 'laravel'), '_').'_cache_'),
109 |
110 | ];
111 |
--------------------------------------------------------------------------------
/resources/views/dashboard.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | {{ __('Dashboard') }}
5 |
6 |
7 |
8 |
9 |
10 |
11 |
13 | @forelse ($chatboxes as $chatbox)
14 |
54 | @empty
55 |
No messages
56 | @endforelse
57 |
58 |
59 | {{ $chatboxes->links() }}
60 |
61 |
62 |
63 |
64 | @push('scripts')
65 |
70 | @endpush
71 |
--------------------------------------------------------------------------------
/app/Livewire/WordpressCreatePost.php:
--------------------------------------------------------------------------------
1 | openAIService = $openAIService;
52 | }
53 |
54 | public function ask()
55 | {
56 | $this->transactions[] = ['role' => 'system', 'content' => $this->chatBoxSystemInstruction];
57 | // If the user has typed something, then asking the ChatGPT API
58 | if (!empty($this->topic)) {
59 | $this->transactions[] = ['role' => 'user', 'content' => $this->topic];
60 | $response = $this->openAIService->ask(
61 | $this->chatBoxModel,
62 | $this->chatBoxMaxTokens,
63 | $this->chatBoxTemperature,
64 | $this->transactions
65 | );
66 | $this->totalTokens = $response->usage->totalTokens;
67 | $this->transactions[] = ['role' => 'assistant', 'content' => $response->choices[0]->message->content];
68 | $this->messages = collect($this->transactions)->reject(function ($message) {
69 | return $message['role'] === 'system' || $message['role'] === 'user';
70 | })->map(function ($message) {
71 | return $message['content'];
72 | })->toArray();
73 | $this->messages = json_decode(array_shift($this->messages), true);
74 | $this->title = $this->messages['title'];
75 | $this->body = $this->messages['body'];
76 | }
77 | }
78 |
79 | public function createPost()
80 | {
81 | $this->validate([
82 | 'url' => 'required|url',
83 | 'title' => 'required',
84 | 'body' => 'required',
85 | // 'status' => 'required',
86 | 'username' => 'required',
87 | 'password' => 'required',
88 | ]);
89 | $response = Wordpress::createPost(
90 | $this->url,
91 | $this->title,
92 | $this->body,
93 | $this->status,
94 | $this->username,
95 | $this->password
96 | );
97 | if ($response) {
98 | $this->alert('success', 'Your post was created successfully!', [
99 | 'position' => 'top-end',
100 | 'timer' => 3000,
101 | 'toast' => true,
102 | ]);
103 | $this->reset(['url', 'title', 'body', 'status', 'username', 'password']);
104 | } else {
105 | $this->alert('error', 'Your post was not created!', [
106 | 'position' => 'top-end',
107 | 'timer' => 3000,
108 | 'toast' => true,
109 | ]);
110 | }
111 | }
112 |
113 | public function render()
114 | {
115 | return view('livewire.wordpress.create-post');
116 | }
117 | }
118 |
--------------------------------------------------------------------------------
/config/mail.php:
--------------------------------------------------------------------------------
1 | env('MAIL_MAILER', 'smtp'),
17 |
18 | /*
19 | |--------------------------------------------------------------------------
20 | | Mailer Configurations
21 | |--------------------------------------------------------------------------
22 | |
23 | | Here you may configure all of the mailers used by your application plus
24 | | their respective settings. Several examples have been configured for
25 | | you and you are free to add your own as your application requires.
26 | |
27 | | Laravel supports a variety of mail "transport" drivers to be used while
28 | | sending an e-mail. You will specify which one you are using for your
29 | | mailers below. You are free to add additional mailers as required.
30 | |
31 | | Supported: "smtp", "sendmail", "mailgun", "ses", "ses-v2",
32 | | "postmark", "log", "array", "failover"
33 | |
34 | */
35 |
36 | 'mailers' => [
37 | 'smtp' => [
38 | 'transport' => 'smtp',
39 | 'host' => env('MAIL_HOST', 'smtp.mailgun.org'),
40 | 'port' => env('MAIL_PORT', 587),
41 | 'encryption' => env('MAIL_ENCRYPTION', 'tls'),
42 | 'username' => env('MAIL_USERNAME'),
43 | 'password' => env('MAIL_PASSWORD'),
44 | 'timeout' => null,
45 | 'local_domain' => env('MAIL_EHLO_DOMAIN'),
46 | ],
47 |
48 | 'ses' => [
49 | 'transport' => 'ses',
50 | ],
51 |
52 | 'mailgun' => [
53 | 'transport' => 'mailgun',
54 | // 'client' => [
55 | // 'timeout' => 5,
56 | // ],
57 | ],
58 |
59 | 'postmark' => [
60 | 'transport' => 'postmark',
61 | // 'client' => [
62 | // 'timeout' => 5,
63 | // ],
64 | ],
65 |
66 | 'sendmail' => [
67 | 'transport' => 'sendmail',
68 | 'path' => env('MAIL_SENDMAIL_PATH', '/usr/sbin/sendmail -bs -i'),
69 | ],
70 |
71 | 'log' => [
72 | 'transport' => 'log',
73 | 'channel' => env('MAIL_LOG_CHANNEL'),
74 | ],
75 |
76 | 'array' => [
77 | 'transport' => 'array',
78 | ],
79 |
80 | 'failover' => [
81 | 'transport' => 'failover',
82 | 'mailers' => [
83 | 'smtp',
84 | 'log',
85 | ],
86 | ],
87 | ],
88 |
89 | /*
90 | |--------------------------------------------------------------------------
91 | | Global "From" Address
92 | |--------------------------------------------------------------------------
93 | |
94 | | You may wish for all e-mails sent by your application to be sent from
95 | | the same address. Here, you may specify a name and address that is
96 | | used globally for all e-mails that are sent by your application.
97 | |
98 | */
99 |
100 | 'from' => [
101 | 'address' => env('MAIL_FROM_ADDRESS', 'hello@example.com'),
102 | 'name' => env('MAIL_FROM_NAME', 'Example'),
103 | ],
104 |
105 | /*
106 | |--------------------------------------------------------------------------
107 | | Markdown Mail Settings
108 | |--------------------------------------------------------------------------
109 | |
110 | | If you are using Markdown based email rendering, you may configure your
111 | | theme and component paths here, allowing you to customize the design
112 | | of the emails. Or, you may simply stick with the Laravel defaults!
113 | |
114 | */
115 |
116 | 'markdown' => [
117 | 'theme' => 'default',
118 |
119 | 'paths' => [
120 | resource_path('views/vendor/mail'),
121 | ],
122 | ],
123 |
124 | ];
125 |
--------------------------------------------------------------------------------
/config/logging.php:
--------------------------------------------------------------------------------
1 | env('LOG_CHANNEL', 'stack'),
21 |
22 | /*
23 | |--------------------------------------------------------------------------
24 | | Deprecations Log Channel
25 | |--------------------------------------------------------------------------
26 | |
27 | | This option controls the log channel that should be used to log warnings
28 | | regarding deprecated PHP and library features. This allows you to get
29 | | your application ready for upcoming major versions of dependencies.
30 | |
31 | */
32 |
33 | 'deprecations' => [
34 | 'channel' => env('LOG_DEPRECATIONS_CHANNEL', 'null'),
35 | 'trace' => false,
36 | ],
37 |
38 | /*
39 | |--------------------------------------------------------------------------
40 | | Log Channels
41 | |--------------------------------------------------------------------------
42 | |
43 | | Here you may configure the log channels for your application. Out of
44 | | the box, Laravel uses the Monolog PHP logging library. This gives
45 | | you a variety of powerful log handlers / formatters to utilize.
46 | |
47 | | Available Drivers: "single", "daily", "slack", "syslog",
48 | | "errorlog", "monolog",
49 | | "custom", "stack"
50 | |
51 | */
52 |
53 | 'channels' => [
54 | 'stack' => [
55 | 'driver' => 'stack',
56 | 'channels' => ['single'],
57 | 'ignore_exceptions' => false,
58 | ],
59 |
60 | 'single' => [
61 | 'driver' => 'single',
62 | 'path' => storage_path('logs/laravel.log'),
63 | 'level' => env('LOG_LEVEL', 'debug'),
64 | ],
65 |
66 | 'daily' => [
67 | 'driver' => 'daily',
68 | 'path' => storage_path('logs/laravel.log'),
69 | 'level' => env('LOG_LEVEL', 'debug'),
70 | 'days' => 14,
71 | ],
72 |
73 | 'slack' => [
74 | 'driver' => 'slack',
75 | 'url' => env('LOG_SLACK_WEBHOOK_URL'),
76 | 'username' => 'Laravel Log',
77 | 'emoji' => ':boom:',
78 | 'level' => env('LOG_LEVEL', 'critical'),
79 | ],
80 |
81 | 'papertrail' => [
82 | 'driver' => 'monolog',
83 | 'level' => env('LOG_LEVEL', 'debug'),
84 | 'handler' => env('LOG_PAPERTRAIL_HANDLER', SyslogUdpHandler::class),
85 | 'handler_with' => [
86 | 'host' => env('PAPERTRAIL_URL'),
87 | 'port' => env('PAPERTRAIL_PORT'),
88 | 'connectionString' => 'tls://'.env('PAPERTRAIL_URL').':'.env('PAPERTRAIL_PORT'),
89 | ],
90 | ],
91 |
92 | 'stderr' => [
93 | 'driver' => 'monolog',
94 | 'level' => env('LOG_LEVEL', 'debug'),
95 | 'handler' => StreamHandler::class,
96 | 'formatter' => env('LOG_STDERR_FORMATTER'),
97 | 'with' => [
98 | 'stream' => 'php://stderr',
99 | ],
100 | ],
101 |
102 | 'syslog' => [
103 | 'driver' => 'syslog',
104 | 'level' => env('LOG_LEVEL', 'debug'),
105 | 'facility' => LOG_USER,
106 | ],
107 |
108 | 'errorlog' => [
109 | 'driver' => 'errorlog',
110 | 'level' => env('LOG_LEVEL', 'debug'),
111 | ],
112 |
113 | 'null' => [
114 | 'driver' => 'monolog',
115 | 'handler' => NullHandler::class,
116 | ],
117 |
118 | 'emergency' => [
119 | 'path' => storage_path('logs/laravel.log'),
120 | ],
121 | ],
122 |
123 | ];
124 |
--------------------------------------------------------------------------------
/config/auth.php:
--------------------------------------------------------------------------------
1 | [
17 | 'guard' => 'web',
18 | 'passwords' => 'users',
19 | ],
20 |
21 | /*
22 | |--------------------------------------------------------------------------
23 | | Authentication Guards
24 | |--------------------------------------------------------------------------
25 | |
26 | | Next, you may define every authentication guard for your application.
27 | | Of course, a great default configuration has been defined for you
28 | | here which uses session storage and the Eloquent user provider.
29 | |
30 | | All authentication drivers have a user provider. This defines how the
31 | | users are actually retrieved out of your database or other storage
32 | | mechanisms used by this application to persist your user's data.
33 | |
34 | | Supported: "session"
35 | |
36 | */
37 |
38 | 'guards' => [
39 | 'web' => [
40 | 'driver' => 'session',
41 | 'provider' => 'users',
42 | ],
43 | ],
44 |
45 | /*
46 | |--------------------------------------------------------------------------
47 | | User Providers
48 | |--------------------------------------------------------------------------
49 | |
50 | | All authentication drivers have a user provider. This defines how the
51 | | users are actually retrieved out of your database or other storage
52 | | mechanisms used by this application to persist your user's data.
53 | |
54 | | If you have multiple user tables or models you may configure multiple
55 | | sources which represent each model / table. These sources may then
56 | | be assigned to any extra authentication guards you have defined.
57 | |
58 | | Supported: "database", "eloquent"
59 | |
60 | */
61 |
62 | 'providers' => [
63 | 'users' => [
64 | 'driver' => 'eloquent',
65 | 'model' => App\Models\User::class,
66 | ],
67 |
68 | // 'users' => [
69 | // 'driver' => 'database',
70 | // 'table' => 'users',
71 | // ],
72 | ],
73 |
74 | /*
75 | |--------------------------------------------------------------------------
76 | | Resetting Passwords
77 | |--------------------------------------------------------------------------
78 | |
79 | | You may specify multiple password reset configurations if you have more
80 | | than one user table or model in the application and you want to have
81 | | separate password reset settings based on the specific user types.
82 | |
83 | | The expiry time is the number of minutes that each reset token will be
84 | | considered valid. This security feature keeps tokens short-lived so
85 | | they have less time to be guessed. You may change this as needed.
86 | |
87 | | The throttle setting is the number of seconds a user must wait before
88 | | generating more password reset tokens. This prevents the user from
89 | | quickly generating a very large amount of password reset tokens.
90 | |
91 | */
92 |
93 | 'passwords' => [
94 | 'users' => [
95 | 'provider' => 'users',
96 | 'table' => 'password_reset_tokens',
97 | 'expire' => 60,
98 | 'throttle' => 60,
99 | ],
100 | ],
101 |
102 | /*
103 | |--------------------------------------------------------------------------
104 | | Password Confirmation Timeout
105 | |--------------------------------------------------------------------------
106 | |
107 | | Here you may define the amount of seconds before a password confirmation
108 | | times out and the user is prompted to re-enter their password via the
109 | | confirmation screen. By default, the timeout lasts for three hours.
110 | |
111 | */
112 |
113 | 'password_timeout' => 10800,
114 |
115 | ];
116 |
--------------------------------------------------------------------------------
/app/Console/Commands/Chat.php:
--------------------------------------------------------------------------------
1 | availableGPTModels();
33 |
34 | // Display an introduction message
35 | intro('Welcome to Chatwire!');
36 |
37 | // Ask the user to choose a GPT model
38 | $chosenModel = suggest(
39 | label: 'Choose a GPT model: ',
40 | options: array_values($availableGPTModels),
41 | required: 'This field is required. Choose a model to use for Chatwire.',
42 | hint: 'Choose a model to use for Chatwire.'
43 | );
44 |
45 | // Get the available GPT roles
46 | $chosenAvailableGPTRoles = $openAIService->availableGPTRoles();
47 |
48 | // Ask the user to choose a role
49 | $chosenRole = suggest(
50 | label: 'Choose a role: ',
51 | options: array_keys($chosenAvailableGPTRoles),
52 | hint: 'If it will be empty, then the default will be used: ' . "$chatBoxSystemInstruction"
53 | );
54 |
55 | // If no role is chosen, use the default system instruction
56 | if (empty($chosenRole)) {
57 | $transactions[] = ['role' => 'system', 'content' => $chatBoxSystemInstruction];
58 | info($chatBoxSystemInstruction);
59 | } else {
60 | // If a role is chosen, use the chosen role
61 | $transactions[] = ['role' => 'system', 'content' => $chosenAvailableGPTRoles[$chosenRole]];
62 | info($chosenAvailableGPTRoles[$chosenRole]);
63 | }
64 |
65 | // Ask the user to choose the max tokens
66 | $chosenMaxTokens = text(
67 | label: 'Choose max tokens: ',
68 | required: 'This field is required. Choose max tokens for Chatwire.',
69 | default: 600,
70 | hint: 'Choose max tokens for Chatwire.'
71 | );
72 |
73 | // Ask the user to choose the temperature
74 | $chosenTemperature = text(
75 | label: 'Choose temperature: ',
76 | required: 'This field is required. Choose temperature for Chatwire.',
77 | default: 0.6,
78 | hint: 'Choose temperature for Chatwire.'
79 | );
80 |
81 | // Get the key of the chosen model
82 | $chosenModelKey = array_search($chosenModel, $availableGPTModels);
83 |
84 | // Ask the user to ask a question to Chatwire
85 | $question = text(
86 | label: 'Ask Chatwire: ',
87 | required: 'This field is required. Type "exit" to leave the chat.',
88 | hint: 'Ask whatever you want Chatwire or type "exit" to exit the chat.'
89 | );
90 |
91 | // Keep asking questions until the user types 'exit'
92 | while ($question !== 'exit') {
93 |
94 | // Add the question to the transactions
95 | $transactions[] = ['role' => 'user', 'content' => $question];
96 |
97 | // Get the response from the AI service
98 | $response = spin(function ()
99 | use ($openAIService, $transactions, $chosenModelKey, $chosenMaxTokens, $chosenTemperature) {
100 | return $openAIService->ask($chosenModelKey, $chosenMaxTokens, $chosenTemperature, $transactions);
101 | });
102 |
103 | // Display the AI's response
104 | info($response->choices[0]->message->content);
105 |
106 | // Ask the next question
107 | $question = text(
108 | label: 'Ask Chatwire: ',
109 | required: 'This field is required. Type "exit" to leave the chat.',
110 | hint: 'Ask whatever you want Chatwire or type "exit" to exit the chat.'
111 | );
112 | }
113 |
114 | // Display a goodbye message
115 | outro('Thank you for using Chatwire. See you next time!');
116 | }
117 | }
118 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Chatwire
2 |
3 | Chatwire is a clone of ChatGPT made with Laravel Breeze using Livewire and the OpenAI PHP client (Chat and Audio Resource). It also supports a command interface for asking questions to ChatGPT using Laravel Prompts.
4 |
5 | ## Description
6 |
7 | In this project, we have a Laravel 11 Breeze with Livewire, a login session, and access to OpenAI's ChatGPT API. The entire Laravel project is based on TailwindCSS using Flowbite, Vite.js to compile scripts, MySQL. As far as the extra packages Flowbite, Laravel-Livewire, openai-php, livewire-alert, Laravel Prompts are used.
8 |
9 | ## Purpose
10 |
11 | Chatwire is a Laravel-based project that calls OpenAI's API and displays the responses in Laravel Breeze Dashboard using Livewire. When you finish your ChatBot conversation, you can send it in your email or save it for future use. You can see your saved conversations on the dashboard page using pagination.
12 |
13 | Now, you can upload your audio files to create transcription using Audio Resource of OpenAI.
14 |
15 | ## Features
16 |
17 | #### Model Configuration
18 |
19 | - All ChatGPT Models (GPT-4)
20 |
21 | - Audio Resource using Whisper Model
22 |
23 | - Custom System Instruction
24 |
25 | - Temperature Control
26 |
27 | - Maximum Token Control
28 |
29 | #### Chat Experience
30 |
31 | - Prompt Library
32 |
33 | #### Wordpress Integration
34 |
35 | - Propose and create a Wordpress Post using Wordpress API
36 |
37 | - SEO optimization proposal (focus keywords, meta title, meta description) for latest post
38 |
39 | #### Chat Management
40 |
41 | - Share Chat by email
42 |
43 | - Save Chat to Database
44 |
45 | - Paginate Chats
46 |
47 | #### Security and Privacy
48 |
49 | - Private By Default
50 |
51 | - Self-hosted
52 |
53 | ## Technologies Used
54 |
55 | This project requires the following technologies:
56 |
57 | - PHP 8.2+
58 |
59 | - MySQL
60 |
61 | - OpenAI account for API Key
62 |
63 | The project uses the following technologies:
64 |
65 | - Laravel 11 Breeze
66 |
67 | - Livewire
68 |
69 | - TailwindCSS
70 |
71 | - Flowbite
72 |
73 | - Vite.js
74 |
75 | - MySQL
76 |
77 | - openai-php
78 |
79 | - livewire-alert
80 |
81 | ## Getting Started
82 |
83 | To run Chatwire locally, follow these steps:
84 |
85 | 1. Clone this repository
86 |
87 | 2. Run `composer install`
88 |
89 | 3. Copy `.env.example` to `.env`
90 |
91 | 4. Add your OpenAI API key in `.env` file at `OPENAI_API_KEY`
92 |
93 | 
94 |
95 | 5. Add your SMTP configuration. This will be used to send your conversation to your email.
96 |
97 | 
98 |
99 | 6. Run `php artisan key:generate`
100 |
101 | 7. Run `php artisan migrate`
102 |
103 | 8. Run `php artisan storage:link`
104 |
105 | 9. Run `npm install`
106 |
107 | 10. Run `npm run build`
108 |
109 | 11. Run `php artisan serve`
110 |
111 | 12. Add the following line to your cronjobs:
112 |
113 | `* * * * * cd /path-to-your-project && php artisan schedule:run >> /dev/null 2>&1`
114 |
115 | The above will be used for the email queue job that it is used for sending the emails
116 |
117 | 13. Open the link: http://127.0.0.1:8000
118 |
119 | ## Usage
120 |
121 | To use Chatwire, follow these steps:
122 |
123 | 1. Start a conversation with the chatbot on the dashboard page.
124 |
125 | 2. When you finish the conversation, you can save it or send it to your email.
126 |
127 | 3. You can view your saved conversations on the dashboard page using pagination.
128 |
129 | 4. You can upload an audio file and ChatGPT will create the trancription for you.
130 |
131 | 5. You can also use `php artisan chat` command in order to ask ChatGPT from CLI.
132 |
133 | ## Screenshots
134 |
135 | Here are some screenshots of Chatwire in action:
136 |
137 | 
138 |
139 | 
140 |
141 | 
142 |
143 | 
144 |
145 | 
146 |
147 | 
148 |
149 | 
150 |
151 | 
152 |
153 | 
154 |
155 | 
156 |
157 | ## Contributing
158 |
159 | If you'd like to contribute to Chatwire, here's how you can get started:
160 |
161 | 1. Fork this repository
162 |
163 | 2. Create a new branch (`git checkout -b my-new-branch`)
164 |
165 | 3. Make your changes
166 |
167 | 4. Commit your changes (`git commit -am 'Add some feature'`)
168 |
169 | 5. Push to the branch (`git push origin my-new-branch`)
170 |
171 | 6. Create a new Pull Request
172 |
173 | We welcome contributions from anyone interested in improving Chatwire
174 |
--------------------------------------------------------------------------------
/app/Livewire/ChatBox.php:
--------------------------------------------------------------------------------
1 | openAIService = $openAIService;
41 | }
42 |
43 | public function mount(ChatBoxModel $chatbox)
44 | {
45 | if ($chatbox->exists) {
46 | $this->messages = json_decode($chatbox->messages, true);
47 | // Preparing saved messages to be loaded in transactions array
48 | $saved_messages = array_values(json_decode($chatbox->messages, true));
49 | foreach ($saved_messages as $saved_message) {
50 | $this->transactions[] = ['role' => $saved_message['role'], 'content' => $saved_message['content']];
51 | }
52 | }
53 | }
54 |
55 | public function ask()
56 | {
57 | $this->transactions[] = ['role' => 'system', 'content' => $this->chatBoxSystemInstruction];
58 | // If the user has typed something, then asking the ChatGPT API
59 | if (!empty($this->message)) {
60 | $this->transactions[] = ['role' => 'user', 'content' => $this->message];
61 | $response = $this->openAIService->ask(
62 | $this->chatBoxModel,
63 | $this->chatBoxMaxTokens,
64 | $this->chatBoxTemperature,
65 | $this->transactions
66 | );
67 | $this->totalTokens = $response->usage->totalTokens;
68 | $this->transactions[] = ['role' => 'assistant', 'content' => $response->choices[0]->message->content];
69 | $this->messages = collect($this->transactions)->reject(fn ($message) => $message['role'] === 'system');
70 | $this->message = '';
71 | }
72 | }
73 |
74 | public function sendChatToEmail()
75 | {
76 | if ($this->messages === []) {
77 | $this->alert('error', 'You have not started a conversation yet!', [
78 | 'position' => 'top-end',
79 | 'timer' => 3000,
80 | 'toast' => true,
81 | ]);
82 | } else {
83 | $details = [
84 | 'email' => auth()->user()->email,
85 | 'messages' => $this->messages,
86 | ];
87 | dispatch(new \App\Jobs\SendEmailJob($details));
88 | $this->alert('success', 'Your email was sent successfully!', [
89 | 'position' => 'top-end',
90 | 'timer' => 3000,
91 | 'toast' => true,
92 | ]);
93 | }
94 | }
95 |
96 | public function updatedChatBoxRole($value)
97 | {
98 | $this->message = $value;
99 | }
100 |
101 | public function resetChatBox()
102 | {
103 | return redirect()->route('chatbox');
104 | }
105 |
106 | public function saveChat()
107 | {
108 | if ($this->messages === []) {
109 | $this->alert('error', 'You have not started a conversation yet!', [
110 | 'position' => 'top-end',
111 | 'timer' => 3000,
112 | 'toast' => true,
113 | ]);
114 | } else {
115 | if ($this->chatbox->exists) {
116 | $this->chatbox->update([
117 | 'messages' => json_encode($this->messages),
118 | 'total_tokens' => $this->totalTokens,
119 | ]);
120 | $this->message = '';
121 | $this->alert('success', 'Your chat was updated successfully!', [
122 | 'position' => 'top-end',
123 | 'timer' => 3000,
124 | 'toast' => true,
125 | ]);
126 | } else {
127 | $chatBox = new ChatBoxModel();
128 | $chatBox->user_id = auth()->user()->id;
129 | $chatBox->messages = $this->messages;
130 | $chatBox->total_tokens = $this->totalTokens;
131 | $chatBox->save();
132 | $this->message = '';
133 | $this->alert('success', 'Your chat was saved successfully!', [
134 | 'position' => 'top-end',
135 | 'timer' => 3000,
136 | 'toast' => true,
137 | ]);
138 | }
139 | }
140 | }
141 |
142 | public function render()
143 | {
144 | return view('livewire.chat-box.chat-box', [
145 | 'availableGPTModels' => $this->openAIService->availableGPTModels(),
146 | 'availableGPTRoles' => $this->openAIService->availableGPTRoles(),
147 | ]);
148 | }
149 | }
150 |
--------------------------------------------------------------------------------
/CODE_OF_CONDUCT.md:
--------------------------------------------------------------------------------
1 | # Contributor Covenant Code of Conduct
2 |
3 | ## Our Pledge
4 |
5 | We as members, contributors, and leaders pledge to make participation in our
6 | community a harassment-free experience for everyone, regardless of age, body
7 | size, visible or invisible disability, ethnicity, sex characteristics, gender
8 | identity and expression, level of experience, education, socio-economic status,
9 | nationality, personal appearance, race, religion, or sexual identity
10 | and orientation.
11 |
12 | We pledge to act and interact in ways that contribute to an open, welcoming,
13 | diverse, inclusive, and healthy community.
14 |
15 | ## Our Standards
16 |
17 | Examples of behavior that contributes to a positive environment for our
18 | community include:
19 |
20 | * Demonstrating empathy and kindness toward other people
21 | * Being respectful of differing opinions, viewpoints, and experiences
22 | * Giving and gracefully accepting constructive feedback
23 | * Accepting responsibility and apologizing to those affected by our mistakes,
24 | and learning from the experience
25 | * Focusing on what is best not just for us as individuals, but for the
26 | overall community
27 |
28 | Examples of unacceptable behavior include:
29 |
30 | * The use of sexualized language or imagery, and sexual attention or
31 | advances of any kind
32 | * Trolling, insulting or derogatory comments, and personal or political attacks
33 | * Public or private harassment
34 | * Publishing others' private information, such as a physical or email
35 | address, without their explicit permission
36 | * Other conduct which could reasonably be considered inappropriate in a
37 | professional setting
38 |
39 | ## Enforcement Responsibilities
40 |
41 | Community leaders are responsible for clarifying and enforcing our standards of
42 | acceptable behavior and will take appropriate and fair corrective action in
43 | response to any behavior that they deem inappropriate, threatening, offensive,
44 | or harmful.
45 |
46 | Community leaders have the right and responsibility to remove, edit, or reject
47 | comments, commits, code, wiki edits, issues, and other contributions that are
48 | not aligned to this Code of Conduct, and will communicate reasons for moderation
49 | decisions when appropriate.
50 |
51 | ## Scope
52 |
53 | This Code of Conduct applies within all community spaces, and also applies when
54 | an individual is officially representing the community in public spaces.
55 | Examples of representing our community include using an official e-mail address,
56 | posting via an official social media account, or acting as an appointed
57 | representative at an online or offline event.
58 |
59 | ## Enforcement
60 |
61 | Instances of abusive, harassing, or otherwise unacceptable behavior may be
62 | reported to the community leaders responsible for enforcement at
63 | .
64 | All complaints will be reviewed and investigated promptly and fairly.
65 |
66 | All community leaders are obligated to respect the privacy and security of the
67 | reporter of any incident.
68 |
69 | ## Enforcement Guidelines
70 |
71 | Community leaders will follow these Community Impact Guidelines in determining
72 | the consequences for any action they deem in violation of this Code of Conduct:
73 |
74 | ### 1. Correction
75 |
76 | **Community Impact**: Use of inappropriate language or other behavior deemed
77 | unprofessional or unwelcome in the community.
78 |
79 | **Consequence**: A private, written warning from community leaders, providing
80 | clarity around the nature of the violation and an explanation of why the
81 | behavior was inappropriate. A public apology may be requested.
82 |
83 | ### 2. Warning
84 |
85 | **Community Impact**: A violation through a single incident or series
86 | of actions.
87 |
88 | **Consequence**: A warning with consequences for continued behavior. No
89 | interaction with the people involved, including unsolicited interaction with
90 | those enforcing the Code of Conduct, for a specified period of time. This
91 | includes avoiding interactions in community spaces as well as external channels
92 | like social media. Violating these terms may lead to a temporary or
93 | permanent ban.
94 |
95 | ### 3. Temporary Ban
96 |
97 | **Community Impact**: A serious violation of community standards, including
98 | sustained inappropriate behavior.
99 |
100 | **Consequence**: A temporary ban from any sort of interaction or public
101 | communication with the community for a specified period of time. No public or
102 | private interaction with the people involved, including unsolicited interaction
103 | with those enforcing the Code of Conduct, is allowed during this period.
104 | Violating these terms may lead to a permanent ban.
105 |
106 | ### 4. Permanent Ban
107 |
108 | **Community Impact**: Demonstrating a pattern of violation of community
109 | standards, including sustained inappropriate behavior, harassment of an
110 | individual, or aggression toward or disparagement of classes of individuals.
111 |
112 | **Consequence**: A permanent ban from any sort of public interaction within
113 | the community.
114 |
115 | ## Attribution
116 |
117 | This Code of Conduct is adapted from the [Contributor Covenant][homepage],
118 | version 2.0, available at
119 | https://www.contributor-covenant.org/version/2/0/code_of_conduct.html.
120 |
121 | Community Impact Guidelines were inspired by [Mozilla's code of conduct
122 | enforcement ladder](https://github.com/mozilla/diversity).
123 |
124 | [homepage]: https://www.contributor-covenant.org
125 |
126 | For answers to common questions about this code of conduct, see the FAQ at
127 | https://www.contributor-covenant.org/faq. Translations are available at
128 | https://www.contributor-covenant.org/translations.
129 |
--------------------------------------------------------------------------------
/config/database.php:
--------------------------------------------------------------------------------
1 | env('DB_CONNECTION', 'mysql'),
19 |
20 | /*
21 | |--------------------------------------------------------------------------
22 | | Database Connections
23 | |--------------------------------------------------------------------------
24 | |
25 | | Here are each of the database connections setup for your application.
26 | | Of course, examples of configuring each database platform that is
27 | | supported by Laravel is shown below to make development simple.
28 | |
29 | |
30 | | All database work in Laravel is done through the PHP PDO facilities
31 | | so make sure you have the driver for your particular database of
32 | | choice installed on your machine before you begin development.
33 | |
34 | */
35 |
36 | 'connections' => [
37 |
38 | 'sqlite' => [
39 | 'driver' => 'sqlite',
40 | 'url' => env('DATABASE_URL'),
41 | 'database' => env('DB_DATABASE', database_path('database.sqlite')),
42 | 'prefix' => '',
43 | 'foreign_key_constraints' => env('DB_FOREIGN_KEYS', true),
44 | ],
45 |
46 | 'mysql' => [
47 | 'driver' => 'mysql',
48 | 'url' => env('DATABASE_URL'),
49 | 'host' => env('DB_HOST', '127.0.0.1'),
50 | 'port' => env('DB_PORT', '3306'),
51 | 'database' => env('DB_DATABASE', 'forge'),
52 | 'username' => env('DB_USERNAME', 'forge'),
53 | 'password' => env('DB_PASSWORD', ''),
54 | 'unix_socket' => env('DB_SOCKET', ''),
55 | 'charset' => 'utf8mb4',
56 | 'collation' => 'utf8mb4_unicode_ci',
57 | 'prefix' => '',
58 | 'prefix_indexes' => true,
59 | 'strict' => true,
60 | 'engine' => null,
61 | 'options' => extension_loaded('pdo_mysql') ? array_filter([
62 | PDO::MYSQL_ATTR_SSL_CA => env('MYSQL_ATTR_SSL_CA'),
63 | ]) : [],
64 | ],
65 |
66 | 'pgsql' => [
67 | 'driver' => 'pgsql',
68 | 'url' => env('DATABASE_URL'),
69 | 'host' => env('DB_HOST', '127.0.0.1'),
70 | 'port' => env('DB_PORT', '5432'),
71 | 'database' => env('DB_DATABASE', 'forge'),
72 | 'username' => env('DB_USERNAME', 'forge'),
73 | 'password' => env('DB_PASSWORD', ''),
74 | 'charset' => 'utf8',
75 | 'prefix' => '',
76 | 'prefix_indexes' => true,
77 | 'search_path' => 'public',
78 | 'sslmode' => 'prefer',
79 | ],
80 |
81 | 'sqlsrv' => [
82 | 'driver' => 'sqlsrv',
83 | 'url' => env('DATABASE_URL'),
84 | 'host' => env('DB_HOST', 'localhost'),
85 | 'port' => env('DB_PORT', '1433'),
86 | 'database' => env('DB_DATABASE', 'forge'),
87 | 'username' => env('DB_USERNAME', 'forge'),
88 | 'password' => env('DB_PASSWORD', ''),
89 | 'charset' => 'utf8',
90 | 'prefix' => '',
91 | 'prefix_indexes' => true,
92 | // 'encrypt' => env('DB_ENCRYPT', 'yes'),
93 | // 'trust_server_certificate' => env('DB_TRUST_SERVER_CERTIFICATE', 'false'),
94 | ],
95 |
96 | ],
97 |
98 | /*
99 | |--------------------------------------------------------------------------
100 | | Migration Repository Table
101 | |--------------------------------------------------------------------------
102 | |
103 | | This table keeps track of all the migrations that have already run for
104 | | your application. Using this information, we can determine which of
105 | | the migrations on disk haven't actually been run in the database.
106 | |
107 | */
108 |
109 | 'migrations' => 'migrations',
110 |
111 | /*
112 | |--------------------------------------------------------------------------
113 | | Redis Databases
114 | |--------------------------------------------------------------------------
115 | |
116 | | Redis is an open source, fast, and advanced key-value store that also
117 | | provides a richer body of commands than a typical key-value system
118 | | such as APC or Memcached. Laravel makes it easy to dig right in.
119 | |
120 | */
121 |
122 | 'redis' => [
123 |
124 | 'client' => env('REDIS_CLIENT', 'phpredis'),
125 |
126 | 'options' => [
127 | 'cluster' => env('REDIS_CLUSTER', 'redis'),
128 | 'prefix' => env('REDIS_PREFIX', Str::slug(env('APP_NAME', 'laravel'), '_').'_database_'),
129 | ],
130 |
131 | 'default' => [
132 | 'url' => env('REDIS_URL'),
133 | 'host' => env('REDIS_HOST', '127.0.0.1'),
134 | 'username' => env('REDIS_USERNAME'),
135 | 'password' => env('REDIS_PASSWORD'),
136 | 'port' => env('REDIS_PORT', '6379'),
137 | 'database' => env('REDIS_DB', '0'),
138 | ],
139 |
140 | 'cache' => [
141 | 'url' => env('REDIS_URL'),
142 | 'host' => env('REDIS_HOST', '127.0.0.1'),
143 | 'username' => env('REDIS_USERNAME'),
144 | 'password' => env('REDIS_PASSWORD'),
145 | 'port' => env('REDIS_PORT', '6379'),
146 | 'database' => env('REDIS_CACHE_DB', '1'),
147 | ],
148 |
149 | ],
150 |
151 | ];
152 |
--------------------------------------------------------------------------------
/resources/views/layouts/navigation.blade.php:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
12 |
13 |
14 |
15 |
16 | {{ __('Dashboard') }}
17 |
18 |
19 | {{ __('ChatBox') }}
20 |
21 |
22 | {{ __('Wordpress') }}
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
33 | {{ Auth::user()->name }}
34 |
35 |
43 |
44 |
45 |
46 |
47 |
48 | {{ __('Profile') }}
49 |
50 |
51 |
52 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
68 |
69 |
72 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 | {{ __('Dashboard') }}
85 |
86 |
87 | {{ __('Chat Box') }}
88 |
89 |
90 | {{ __('Wordpress') }}
91 |
92 |
93 |
94 |
95 |
96 |
97 |
{{ Auth::user()->name }}
98 |
{{ Auth::user()->email }}
99 |
100 |
101 |
102 |
103 | {{ __('Profile') }}
104 |
105 |
106 |
107 |
115 |
116 |
117 |
118 |
--------------------------------------------------------------------------------