some initial backend scaffolding

This commit is contained in:
Lara 2024-11-06 20:40:37 +02:00
parent c12d47b749
commit 48f8459b58
Signed by: laratheprotogen
GPG key ID: 5C0296EB3165F98B
21 changed files with 1991 additions and 321 deletions

View file

@ -0,0 +1,64 @@
<?php
namespace App\Http\Controllers;
use Illuminate\Http\Request;
class AllSearchController extends Controller
{
/**
* Display a listing of the resource.
*/
public function index()
{
//
}
/**
* Show the form for creating a new resource.
*/
public function create()
{
//
}
/**
* Store a newly created resource in storage.
*/
public function store(Request $request)
{
//
}
/**
* Display the specified resource.
*/
public function show(string $id)
{
//
}
/**
* Show the form for editing the specified resource.
*/
public function edit(string $id)
{
//
}
/**
* Update the specified resource in storage.
*/
public function update(Request $request, string $id)
{
//
}
/**
* Remove the specified resource from storage.
*/
public function destroy(string $id)
{
//
}
}

View file

@ -0,0 +1,19 @@
<?php
namespace App\Http\Resources;
use Illuminate\Http\Request;
use Illuminate\Http\Resources\Json\JsonResource;
class AllSearchResource extends JsonResource
{
/**
* Transform the resource into an array.
*
* @return array<string, mixed>
*/
public function toArray(Request $request): array
{
return parent::toArray($request);
}
}

11
app/Models/Album.php Normal file
View file

@ -0,0 +1,11 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Album extends Model
{
use Searchable;
}

11
app/Models/Artist.php Normal file
View file

@ -0,0 +1,11 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Artist extends Model
{
use Searchable;
}

11
app/Models/Playlist.php Normal file
View file

@ -0,0 +1,11 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Playlist extends Model
{
use Searchable;
}

11
app/Models/Song.php Normal file
View file

@ -0,0 +1,11 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Song extends Model
{
use Searchable;
}

View file

@ -2,6 +2,7 @@
namespace App\Providers; namespace App\Providers;
use Illuminate\Http\Resources\Json\JsonResource;
use Illuminate\Support\ServiceProvider; use Illuminate\Support\ServiceProvider;
class AppServiceProvider extends ServiceProvider class AppServiceProvider extends ServiceProvider
@ -19,6 +20,6 @@ class AppServiceProvider extends ServiceProvider
*/ */
public function boot(): void public function boot(): void
{ {
// JsonResource::withoutWrapping();
} }
} }

View file

@ -2,4 +2,5 @@
return [ return [
App\Providers\AppServiceProvider::class, App\Providers\AppServiceProvider::class,
TeamTNT\Scout\TNTSearchScoutServiceProvider::class,
]; ];

View file

@ -11,9 +11,13 @@
"php": "^8.3", "php": "^8.3",
"ext-pdo": "*", "ext-pdo": "*",
"inertiajs/inertia-laravel": "2.x-dev", "inertiajs/inertia-laravel": "2.x-dev",
"james-heinrich/getid3": "^1.9",
"laravel/framework": "^11.9", "laravel/framework": "^11.9",
"laravel/scout": "^10.11",
"laravel/tinker": "^2.9", "laravel/tinker": "^2.9",
"php-ffmpeg/php-ffmpeg": "^1.2", "php-ffmpeg/php-ffmpeg": "^1.2",
"teamtnt/laravel-scout-tntsearch-driver": "^14.0",
"teamtnt/tntsearch": "^4.3",
"tightenco/ziggy": "^2.3" "tightenco/ziggy": "^2.3"
}, },
"require-dev": { "require-dev": {

1816
composer.lock generated

File diff suppressed because it is too large Load diff

217
config/scout.php Normal file
View file

@ -0,0 +1,217 @@
<?php
return [
/*
|--------------------------------------------------------------------------
| Default Search Engine
|--------------------------------------------------------------------------
|
| This option controls the default search connection that gets used while
| using Laravel Scout. This connection is used when syncing all models
| to the search service. You should adjust this based on your needs.
|
| Supported: "algolia", "meilisearch", "typesense",
| "database", "collection", "null"
|
*/
'driver' => env('SCOUT_DRIVER', 'algolia'),
/*
|--------------------------------------------------------------------------
| Index Prefix
|--------------------------------------------------------------------------
|
| Here you may specify a prefix that will be applied to all search index
| names used by Scout. This prefix may be useful if you have multiple
| "tenants" or applications sharing the same search infrastructure.
|
*/
'prefix' => env('SCOUT_PREFIX', ''),
/*
|--------------------------------------------------------------------------
| Queue Data Syncing
|--------------------------------------------------------------------------
|
| This option allows you to control if the operations that sync your data
| with your search engines are queued. When this is set to "true" then
| all automatic data syncing will get queued for better performance.
|
*/
'queue' => env('SCOUT_QUEUE', false),
/*
|--------------------------------------------------------------------------
| Database Transactions
|--------------------------------------------------------------------------
|
| This configuration option determines if your data will only be synced
| with your search indexes after every open database transaction has
| been committed, thus preventing any discarded data from syncing.
|
*/
'after_commit' => false,
/*
|--------------------------------------------------------------------------
| Chunk Sizes
|--------------------------------------------------------------------------
|
| These options allow you to control the maximum chunk size when you are
| mass importing data into the search engine. This allows you to fine
| tune each of these chunk sizes based on the power of the servers.
|
*/
'chunk' => [
'searchable' => 500,
'unsearchable' => 500,
],
/*
|--------------------------------------------------------------------------
| Soft Deletes
|--------------------------------------------------------------------------
|
| This option allows to control whether to keep soft deleted records in
| the search indexes. Maintaining soft deleted records can be useful
| if your application still needs to search for the records later.
|
*/
'soft_delete' => false,
/*
|--------------------------------------------------------------------------
| Identify User
|--------------------------------------------------------------------------
|
| This option allows you to control whether to notify the search engine
| of the user performing the search. This is sometimes useful if the
| engine supports any analytics based on this application's users.
|
| Supported engines: "algolia"
|
*/
'identify' => env('SCOUT_IDENTIFY', false),
/*
|--------------------------------------------------------------------------
| Algolia Configuration
|--------------------------------------------------------------------------
|
| Here you may configure your Algolia settings. Algolia is a cloud hosted
| search engine which works great with Scout out of the box. Just plug
| in your application ID and admin API key to get started searching.
|
*/
'algolia' => [
'id' => env('ALGOLIA_APP_ID', ''),
'secret' => env('ALGOLIA_SECRET', ''),
],
/*
|--------------------------------------------------------------------------
| Meilisearch Configuration
|--------------------------------------------------------------------------
|
| Here you may configure your Meilisearch settings. Meilisearch is an open
| source search engine with minimal configuration. Below, you can state
| the host and key information for your own Meilisearch installation.
|
| See: https://www.meilisearch.com/docs/learn/configuration/instance_options#all-instance-options
|
*/
'meilisearch' => [
'host' => env('MEILISEARCH_HOST', 'http://localhost:7700'),
'key' => env('MEILISEARCH_KEY'),
'index-settings' => [
// 'users' => [
// 'filterableAttributes'=> ['id', 'name', 'email'],
// ],
],
],
/*
|--------------------------------------------------------------------------
| Typesense Configuration
|--------------------------------------------------------------------------
|
| Here you may configure your Typesense settings. Typesense is an open
| source search engine using minimal configuration. Below, you will
| state the host, key, and schema configuration for the instance.
|
*/
'typesense' => [
'client-settings' => [
'api_key' => env('TYPESENSE_API_KEY', 'xyz'),
'nodes' => [
[
'host' => env('TYPESENSE_HOST', 'localhost'),
'port' => env('TYPESENSE_PORT', '8108'),
'path' => env('TYPESENSE_PATH', ''),
'protocol' => env('TYPESENSE_PROTOCOL', 'http'),
],
],
'nearest_node' => [
'host' => env('TYPESENSE_HOST', 'localhost'),
'port' => env('TYPESENSE_PORT', '8108'),
'path' => env('TYPESENSE_PATH', ''),
'protocol' => env('TYPESENSE_PROTOCOL', 'http'),
],
'connection_timeout_seconds' => env('TYPESENSE_CONNECTION_TIMEOUT_SECONDS', 2),
'healthcheck_interval_seconds' => env('TYPESENSE_HEALTHCHECK_INTERVAL_SECONDS', 30),
'num_retries' => env('TYPESENSE_NUM_RETRIES', 3),
'retry_interval_seconds' => env('TYPESENSE_RETRY_INTERVAL_SECONDS', 1),
],
// 'max_total_results' => env('TYPESENSE_MAX_TOTAL_RESULTS', 1000),
'model-settings' => [
// User::class => [
// 'collection-schema' => [
// 'fields' => [
// [
// 'name' => 'id',
// 'type' => 'string',
// ],
// [
// 'name' => 'name',
// 'type' => 'string',
// ],
// [
// 'name' => 'created_at',
// 'type' => 'int64',
// ],
// ],
// 'default_sorting_field' => 'created_at',
// ],
// 'search-parameters' => [
// 'query_by' => 'name'
// ],
// ],
],
],
'tntsearch' => [
'storage' => storage_path(), //place where the index files will be stored
'fuzziness' => env('TNTSEARCH_FUZZINESS', false),
'fuzzy' => [
'prefix_length' => 2,
'max_expansions' => 50,
'distance' => 2,
'no_limit' => true
],
'asYouType' => false,
'searchBoolean' => env('TNTSEARCH_BOOLEAN', false),
'maxDocs' => env('TNTSEARCH_MAX_DOCS', 500),
],
];

View file

@ -0,0 +1,27 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('songs', function (Blueprint $table) {
$table->id();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('songs');
}
};

View file

@ -0,0 +1,27 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('albums', function (Blueprint $table) {
$table->id();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('albums');
}
};

View file

@ -0,0 +1,27 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('artists', function (Blueprint $table) {
$table->id();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('artists');
}
};

View file

@ -0,0 +1,27 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('playlists', function (Blueprint $table) {
$table->id();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('playlists');
}
};

View file

@ -4,7 +4,7 @@
"scripts": { "scripts": {
"build": "tsc && vite build && vite build --ssr", "build": "tsc && vite build && vite build --ssr",
"php:serve": "php artisan serve", "php:serve": "php artisan serve",
"php:queue": "php artisan queue:listen --tries=1", "php:queue": "php artisan queue:listen --tries=1",
"php:pail": "php artisan pail", "php:pail": "php artisan pail",
"vite:dev": "vite", "vite:dev": "vite",
"dev": "concurrently -k -p \"[{name}]\" -c \"blue.bold,green.bold,red.bold,orange.bold\" \"pnpm:php:serve\" \"pnpm:php:pail\" \"pnpm:vite:dev\" \"pnpm:php:queue\"", "dev": "concurrently -k -p \"[{name}]\" -c \"blue.bold,green.bold,red.bold,orange.bold\" \"pnpm:php:serve\" \"pnpm:php:pail\" \"pnpm:vite:dev\" \"pnpm:php:queue\"",
@ -16,6 +16,7 @@
"@inertiajs/svelte": "2.0.0-beta.2", "@inertiajs/svelte": "2.0.0-beta.2",
"@sveltejs/vite-plugin-svelte": "^4.0.0", "@sveltejs/vite-plugin-svelte": "^4.0.0",
"@types/eslint": "^9.6.0", "@types/eslint": "^9.6.0",
"@types/lodash": "^4.17.13",
"@types/node": "^22.9.0", "@types/node": "^22.9.0",
"autoprefixer": "^10.4.20", "autoprefixer": "^10.4.20",
"axios": "^1.7.4", "axios": "^1.7.4",
@ -25,6 +26,7 @@
"eslint-plugin-svelte": "^2.36.0", "eslint-plugin-svelte": "^2.36.0",
"globals": "^15.0.0", "globals": "^15.0.0",
"laravel-vite-plugin": "^1.0", "laravel-vite-plugin": "^1.0",
"lodash": "^4.17.21",
"material-icons": "^1.13.12", "material-icons": "^1.13.12",
"postcss": "^8.4.47", "postcss": "^8.4.47",
"prettier": "^3.3.2", "prettier": "^3.3.2",

11
pnpm-lock.yaml generated
View file

@ -20,6 +20,9 @@ importers:
'@types/eslint': '@types/eslint':
specifier: ^9.6.0 specifier: ^9.6.0
version: 9.6.1 version: 9.6.1
'@types/lodash':
specifier: ^4.17.13
version: 4.17.13
'@types/node': '@types/node':
specifier: ^22.9.0 specifier: ^22.9.0
version: 22.9.0 version: 22.9.0
@ -47,6 +50,9 @@ importers:
laravel-vite-plugin: laravel-vite-plugin:
specifier: ^1.0 specifier: ^1.0
version: 1.0.5(vite@5.4.10(@types/node@22.9.0)(sass-embedded@1.80.6)) version: 1.0.5(vite@5.4.10(@types/node@22.9.0)(sass-embedded@1.80.6))
lodash:
specifier: ^4.17.21
version: 4.17.21
material-icons: material-icons:
specifier: ^1.13.12 specifier: ^1.13.12
version: 1.13.12 version: 1.13.12
@ -446,6 +452,9 @@ packages:
'@types/json-schema@7.0.15': '@types/json-schema@7.0.15':
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
'@types/lodash@4.17.13':
resolution: {integrity: sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==}
'@types/node@22.9.0': '@types/node@22.9.0':
resolution: {integrity: sha512-vuyHg81vvWA1Z1ELfvLko2c8f34gyA0zaic0+Rllc5lbCnbSyuvb2Oxpm6TAUAC/2xZN3QGqxBNggD1nNR2AfQ==} resolution: {integrity: sha512-vuyHg81vvWA1Z1ELfvLko2c8f34gyA0zaic0+Rllc5lbCnbSyuvb2Oxpm6TAUAC/2xZN3QGqxBNggD1nNR2AfQ==}
@ -1957,6 +1966,8 @@ snapshots:
'@types/json-schema@7.0.15': {} '@types/json-schema@7.0.15': {}
'@types/lodash@4.17.13': {}
'@types/node@22.9.0': '@types/node@22.9.0':
dependencies: dependencies:
undici-types: 6.19.8 undici-types: 6.19.8

View file

@ -6,18 +6,19 @@
//eslint-disable-next-line //eslint-disable-next-line
let { children }: { children: Snippet } = $props(); let { children }: { children: Snippet } = $props();
const searchChanged = () => { const searchChanged = (event: InputEvent) => {
if ($page.url !== '/search') { if (!$page.url.includes('/search')) {
globalState.lastPage = $page.url globalState.lastPage = $page.url
router.visit('/search', { router.visit('/search/', {
preserveState: true, preserveState: true,
}) })
} }
if ($page.url === '/search' && globalState.searchQuery === '') { if ($page.url.includes('/search') && event.data === '') {
router.visit(globalState.lastPage ?? '/', { router.visit(globalState.lastPage ?? '/', {
preserveState: true, preserveState: true,
}) })
} }
globalState.onSearchChanged?.(event)
} }
</script> </script>

View file

@ -4,6 +4,7 @@ export const globalState: {
title?: Snippet; title?: Snippet;
searchQuery?: string; searchQuery?: string;
lastPage?: string; lastPage?: string;
onSearchChanged?: (event: InputEvent) => void;
} = $state({ } = $state({
}) })

View file

@ -4,9 +4,16 @@
<script lang="ts"> <script lang="ts">
import { globalState } from "@/Library/PassState.svelte"; import { globalState } from "@/Library/PassState.svelte";
import {default as _} from 'lodash';
let { appName } = $props(); let { appName } = $props();
globalState.title = title; globalState.title = title;
const onSearchChanged = (event: InputEvent) => {
}
globalState.onSearchChanged = _.throttle(onSearchChanged, 500);
</script> </script>
<svelte:head> <svelte:head>

View file

@ -7,6 +7,6 @@ Route::get('/', function () {
return Inertia::render('Home'); return Inertia::render('Home');
})->name('home'); })->name('home');
Route::get('/search', function () { Route::get('/search/{query?}', function (?string $query = null) {
return Inertia::render('Search'); return Inertia::render('Search');
})->name('search'); })->name('search')->where('search', '.*');