diff --git a/composer.json b/composer.json index 4440336..6aedd02 100644 --- a/composer.json +++ b/composer.json @@ -32,7 +32,7 @@ "react/async": "^4.2", "react/http": "^1.9", "symfony/psr-http-message-bridge": "^7.0", - "team-reflex/discord-php": "dev-master" + "team-reflex/discord-php": "v10.0.0-RC10" }, "require-dev": { "laravel/pint": "^1.15" diff --git a/config/database.php b/config/database.php index c4cc182..6893f2f 100644 --- a/config/database.php +++ b/config/database.php @@ -10,8 +10,9 @@ |-------------------------------------------------------------------------- | | Here you may specify which of the database connections below you wish - | to use as your default connection for all database work. Of course - | you may use many connections at once using the Database library. + | to use as your default connection for database operations. This is + | the connection which will be utilized unless another connection + | is explicitly specified when you execute a query / statement. | */ @@ -22,14 +23,9 @@ | Database Connections |-------------------------------------------------------------------------- | - | Here are each of the database connections setup for your application. - | Of course, examples of configuring each database platform that is - | supported by Laravel is shown below to make development simple. - | - | - | All database work in Laravel is done through the PHP PDO facilities - | so make sure you have the driver for your particular database of - | choice installed on your machine before you begin development. + | Below are all of the database connections defined for your application. + | An example configuration is provided for each database system which + | is supported by Laravel. You're free to add / remove connections. | */ @@ -37,26 +33,49 @@ 'sqlite' => [ 'driver' => 'sqlite', - 'url' => env('DATABASE_URL'), + 'url' => env('DB_URL'), 'database' => env('DB_DATABASE', config('app.env') === 'production' ? laracord_path('database.sqlite', basePath: false) : database_path('database.sqlite') ), 'prefix' => '', 'foreign_key_constraints' => env('DB_FOREIGN_KEYS', true), + 'busy_timeout' => null, + 'journal_mode' => null, + 'synchronous' => null, ], 'mysql' => [ 'driver' => 'mysql', - 'url' => env('DATABASE_URL'), + 'url' => env('DB_URL'), + 'host' => env('DB_HOST', '127.0.0.1'), + 'port' => env('DB_PORT', '3306'), + 'database' => env('DB_DATABASE', 'laracord'), + 'username' => env('DB_USERNAME', 'root'), + 'password' => env('DB_PASSWORD', ''), + 'unix_socket' => env('DB_SOCKET', ''), + 'charset' => env('DB_CHARSET', 'utf8mb4'), + 'collation' => env('DB_COLLATION', 'utf8mb4_unicode_ci'), + 'prefix' => '', + 'prefix_indexes' => true, + 'strict' => true, + 'engine' => null, + 'options' => extension_loaded('pdo_mysql') ? array_filter([ + PDO::MYSQL_ATTR_SSL_CA => env('MYSQL_ATTR_SSL_CA'), + ]) : [], + ], + + 'mariadb' => [ + 'driver' => 'mariadb', + 'url' => env('DB_URL'), 'host' => env('DB_HOST', '127.0.0.1'), 'port' => env('DB_PORT', '3306'), 'database' => env('DB_DATABASE', 'laracord'), - 'username' => env('DB_USERNAME', 'laracord'), + 'username' => env('DB_USERNAME', 'root'), 'password' => env('DB_PASSWORD', ''), 'unix_socket' => env('DB_SOCKET', ''), - 'charset' => 'utf8mb4', - 'collation' => 'utf8mb4_unicode_ci', + 'charset' => env('DB_CHARSET', 'utf8mb4'), + 'collation' => env('DB_COLLATION', 'utf8mb4_unicode_ci'), 'prefix' => '', 'prefix_indexes' => true, 'strict' => true, @@ -66,6 +85,36 @@ ]) : [], ], + 'pgsql' => [ + 'driver' => 'pgsql', + 'url' => env('DB_URL'), + 'host' => env('DB_HOST', '127.0.0.1'), + 'port' => env('DB_PORT', '5432'), + 'database' => env('DB_DATABASE', 'laracord'), + 'username' => env('DB_USERNAME', 'root'), + 'password' => env('DB_PASSWORD', ''), + 'charset' => env('DB_CHARSET', 'utf8'), + 'prefix' => '', + 'prefix_indexes' => true, + 'search_path' => 'public', + 'sslmode' => 'prefer', + ], + + 'sqlsrv' => [ + 'driver' => 'sqlsrv', + 'url' => env('DB_URL'), + 'host' => env('DB_HOST', 'localhost'), + 'port' => env('DB_PORT', '1433'), + 'database' => env('DB_DATABASE', 'laracord'), + 'username' => env('DB_USERNAME', 'root'), + 'password' => env('DB_PASSWORD', ''), + 'charset' => env('DB_CHARSET', 'utf8'), + 'prefix' => '', + 'prefix_indexes' => true, + // 'encrypt' => env('DB_ENCRYPT', 'yes'), + // 'trust_server_certificate' => env('DB_TRUST_SERVER_CERTIFICATE', 'false'), + ], + ], /* @@ -75,11 +124,14 @@ | | This table keeps track of all the migrations that have already run for | your application. Using this information, we can determine which of - | the migrations on disk haven't actually been run in the database. + | the migrations on disk haven't actually been run on the database. | */ - 'migrations' => 'migrations', + 'migrations' => [ + 'table' => 'migrations', + 'update_date_on_publish' => true, + ], /* |-------------------------------------------------------------------------- @@ -88,7 +140,7 @@ | | Redis is an open source, fast, and advanced key-value store that also | provides a richer body of commands than a typical key-value system - | such as APC or Memcached. Laravel makes it easy to dig right in. + | such as Memcached. You may define your connection settings here. | */ @@ -98,13 +150,14 @@ 'options' => [ 'cluster' => env('REDIS_CLUSTER', 'redis'), - 'prefix' => env('REDIS_PREFIX', Str::slug(env('APP_NAME', 'laravel'), '_').'_database_'), + 'prefix' => env('REDIS_PREFIX', Str::slug(env('APP_NAME', 'laracord'), '_').'_database_'), ], 'default' => [ 'url' => env('REDIS_URL'), 'host' => env('REDIS_HOST', '127.0.0.1'), - 'password' => env('REDIS_PASSWORD', null), + 'username' => env('REDIS_USERNAME'), + 'password' => env('REDIS_PASSWORD'), 'port' => env('REDIS_PORT', '6379'), 'database' => env('REDIS_DB', '0'), ], @@ -112,7 +165,8 @@ 'cache' => [ 'url' => env('REDIS_URL'), 'host' => env('REDIS_HOST', '127.0.0.1'), - 'password' => env('REDIS_PASSWORD', null), + 'username' => env('REDIS_USERNAME'), + 'password' => env('REDIS_PASSWORD'), 'port' => env('REDIS_PORT', '6379'), 'database' => env('REDIS_CACHE_DB', '1'), ], diff --git a/src/Commands/ApplicationCommand.php b/src/Commands/ApplicationCommand.php index ca50709..e0f27ca 100644 --- a/src/Commands/ApplicationCommand.php +++ b/src/Commands/ApplicationCommand.php @@ -2,6 +2,8 @@ namespace Laracord\Commands; +use Discord\Parts\Permissions\RolePermission; + abstract class ApplicationCommand extends AbstractCommand { /** diff --git a/src/Console/Commands/stubs/event.stub b/src/Console/Commands/stubs/event.stub index 7d11bf9..0f03934 100644 --- a/src/Console/Commands/stubs/event.stub +++ b/src/Console/Commands/stubs/event.stub @@ -17,7 +17,7 @@ class {{ class }} extends Event /** * Handle the event. */ - public function handle({{ attributes }}): mixed + public function handle({{ attributes }}) { $this->console()->log('The {{ eventName }} event has fired!'); } diff --git a/src/Console/Concerns/ResolvesUser.php b/src/Console/Concerns/ResolvesUser.php index 6be2f55..0f2d218 100644 --- a/src/Console/Concerns/ResolvesUser.php +++ b/src/Console/Concerns/ResolvesUser.php @@ -2,6 +2,7 @@ namespace Laracord\Console\Concerns; +use Exception; use Illuminate\Support\Facades\Http; use Illuminate\Support\Str; diff --git a/src/Discord/Message.php b/src/Discord/Message.php index b37caf9..e81dd78 100644 --- a/src/Discord/Message.php +++ b/src/Discord/Message.php @@ -16,11 +16,14 @@ use Discord\Parts\Channel\Message as ChannelMessage; use Discord\Parts\Channel\Poll\Poll; use Discord\Parts\Channel\Webhook; +use Discord\Parts\Guild\Sticker; use Discord\Parts\Interactions\Interaction; use Discord\Parts\User\User; use Discord\Repository\Channel\WebhookRepository; use Exception; use Illuminate\Support\Carbon; +use Illuminate\Support\Collection; +use Illuminate\Support\Facades\Storage; use Illuminate\Support\Str; use Laracord\Laracord; use React\Promise\ExtendedPromiseInterface; @@ -151,6 +154,16 @@ class Message */ protected ?Poll $poll = null; + /** + * The message attachments. + */ + protected ?Collection $attachments = null; + + /** + * The message stickers. + */ + protected array $stickers = []; + /** * The message webhook. */ @@ -204,9 +217,10 @@ public function build(): MessageBuilder { $message = MessageBuilder::new() ->setUsername($this->username) - ->setAvatarUrl($this->avatarUrl) + ->setAvatarUrl($this->getAttachment($this->avatarUrl)) ->setTts($this->tts) ->setContent($this->body) + ->setStickers($this->stickers) ->setComponents($this->getComponents()); if ($this->hasContent() || $this->hasFields()) { @@ -220,7 +234,9 @@ public function build(): MessageBuilder } if ($this->hasButtons()) { - $message->addComponent($this->getButtons()); + foreach ($this->getButtons() as $button) { + $message->addComponent($button); + } } if ($this->hasPoll()) { @@ -391,18 +407,18 @@ public function getEmbed(): array 'color' => $this->color, 'footer' => [ 'text' => $this->footerText, - 'icon_url' => $this->footerIcon, + 'icon_url' => $this->getAttachment($this->footerIcon), ], 'thumbnail' => [ - 'url' => $this->thumbnailUrl, + 'url' => $this->getAttachment($this->thumbnailUrl), ], 'image' => [ - 'url' => $this->imageUrl, + 'url' => $this->getAttachment($this->imageUrl), ], 'author' => [ 'name' => $this->authorName, 'url' => $this->authorUrl, - 'icon_url' => $this->authorIcon, + 'icon_url' => $this->getAttachment($this->authorIcon), ], 'fields' => $this->fields, ])->filter()->all(); @@ -417,21 +433,23 @@ public function getComponents(): array } /** - * Get the buttons. + * Get the button components. */ - public function getButtons() + public function getButtons(): array { if (! $this->hasButtons()) { - return; + return []; } - $buttons = ActionRow::new(); + return collect($this->buttons)->chunk(5)->map(function ($buttons) { + $row = ActionRow::new(); - foreach ($this->buttons as $button) { - $buttons->addComponent($button); - } + foreach ($buttons as $button) { + $row->addComponent($button); + } - return $buttons; + return $row; + })->all(); } /** @@ -545,11 +563,29 @@ public function body(string $body = ''): self } /** - * Add a file from content to the message. + * Add a file using raw input, local storage, or a remote URL to the message. */ - public function file(string $content = '', string $filename = ''): self + public function file(string $input = '', ?string $filename = null): self { - $filename = $filename ?? 'file.txt'; + $isPath = (! $isUrl = Str::isUrl($input)) + && Str::length($input) <= 1024 + && ! Str::contains($input, [DIRECTORY_SEPARATOR, "\n"]) + && Str::isMatch('/\.\w+$/', $input); + + $isPath = $isPath && Storage::drive('local')->exists($input); + + $content = match (true) { + $isUrl => file_get_contents($input), + $isPath => Storage::drive('local')->get($input), + default => $input, + }; + + $filename = match (true) { + filled($filename) => $filename, + $isUrl => basename(parse_url($input, PHP_URL_PATH)), + $isPath => basename($input), + default => 'file.txt', + }; $this->files[] = [ 'content' => $content, @@ -562,7 +598,7 @@ public function file(string $content = '', string $filename = ''): self /** * Add a file to the message. */ - public function filePath(string $path, string $filename = ''): self + public function filePath(string $path, ?string $filename = null): self { if (! file_exists($path)) { $this->bot->console()->error("File {$path} does not exist"); @@ -585,6 +621,28 @@ public function hasFiles(): bool return ! empty($this->files); } + /** + * Retrieve the message file attachments. + */ + protected function getAttachments(): Collection + { + return $this->attachments ??= collect($this->files)->mapWithKeys(fn ($file) => [ + $file['filename'] => "attachment://{$file['filename']}", + ]); + } + + /** + * Retrieve a message file attachment. + */ + protected function getAttachment(?string $filename = null): ?string + { + if (blank($filename)) { + return null; + } + + return $this->getAttachments()->get($filename, $filename); + } + /** * Set the message color. */ @@ -666,9 +724,7 @@ public function footerText(?string $footerText): self */ public function thumbnail(?string $thumbnailUrl): self { - $this->thumbnailUrl = $thumbnailUrl; - - return $this; + return $this->thumbnailUrl($thumbnailUrl); } /** @@ -696,9 +752,7 @@ public function url(?string $url): self */ public function image(?string $imageUrl): self { - $this->imageUrl = $imageUrl; - - return $this; + return $this->imageUrl($imageUrl); } /** @@ -711,6 +765,40 @@ public function imageUrl(?string $imageUrl): self return $this; } + /** + * Add a sticker to the message. + */ + public function sticker(string|Sticker $sticker): self + { + $this->stickers[] = $sticker instanceof Sticker + ? $sticker->id + : $sticker; + + return $this; + } + + /** + * Add stickers to the message. + */ + public function stickers(array $stickers): self + { + foreach ($stickers as $sticker) { + $this->sticker($sticker); + } + + return $this; + } + + /** + * Clear the stickers from the message. + */ + public function clearStickers(): self + { + $this->stickers = []; + + return $this; + } + /** * Set the message timestamp. */ diff --git a/src/Http/Middleware/AuthorizeToken.php b/src/Http/Middleware/AuthorizeToken.php index 4c787a3..5faf016 100644 --- a/src/Http/Middleware/AuthorizeToken.php +++ b/src/Http/Middleware/AuthorizeToken.php @@ -15,7 +15,7 @@ class AuthorizeToken */ public function handle(Request $request, Closure $next) { - $token = $request->bearerToken() ?? $request->query('token'); + $token = $request->bearerToken() ?? $request->get('token'); if (! $token) { return response()->json(['message' => 'You must specify a token.'], 401); diff --git a/src/Http/Server.php b/src/Http/Server.php index d45c5db..d451493 100644 --- a/src/Http/Server.php +++ b/src/Http/Server.php @@ -13,6 +13,7 @@ use React\Http\HttpServer; use React\Http\Message\Response; use React\Socket\SocketServer; +use Symfony\Component\HttpFoundation\BinaryFileResponse; use Throwable; class Server @@ -141,7 +142,7 @@ public function getServer() return new Response( $response->getStatusCode(), $response->headers->allPreserveCase(), - $response->getContent() ?: $response->getFile()?->getContent() ?: '' + $response->getContent() ?: ($response instanceof BinaryFileResponse ? $response->getFile()->getContent() : false) ?: '' ); }); } diff --git a/src/Laracord.php b/src/Laracord.php index bddd94d..75a8a02 100644 --- a/src/Laracord.php +++ b/src/Laracord.php @@ -1311,7 +1311,10 @@ public function handleSafe(string $name, callable $callback): mixed return $callback(); } catch (Throwable $e) { $this->console()->error("An error occurred in {$name}."); - $this->console()->outputComponents()->bulletList([$e->getMessage()]); + + $this->console()->outputComponents()->bulletList([ + sprintf('%s in %s:%d', $e->getMessage(), $e->getFile(), $e->getLine()), + ]); } return null;