You've already forked Php-Template
Compare commits
12 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
7fe2722fc1
|
|||
|
5542ad1e75
|
|||
|
e4a55af694
|
|||
|
2879cbe203
|
|||
|
eeb46bc982
|
|||
|
7d0b00fb89
|
|||
|
13445a0719
|
|||
|
54ea22c49a
|
|||
|
f8d3462cb7
|
|||
|
68614958a9
|
|||
|
413145f479
|
|||
|
d2bd9d2d1b
|
@@ -1,3 +1,5 @@
|
|||||||
.idea/
|
.idea/
|
||||||
|
.DS_Store
|
||||||
vendor/
|
vendor/
|
||||||
.phpunit.cache/
|
.phpunit.cache/
|
||||||
|
tests/
|
||||||
@@ -246,10 +246,23 @@ jobs:
|
|||||||
siteworxpro/composer \
|
siteworxpro/composer \
|
||||||
install --ignore-platform-reqs --no-interaction --prefer-dist --optimize-autoloader
|
install --ignore-platform-reqs --no-interaction --prefer-dist --optimize-autoloader
|
||||||
|
|
||||||
- name: Run Unit Tests
|
- name: 🧪 ✅ Run Unit Tests
|
||||||
|
uses: addnab/docker-run-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKER_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
image: siteworxpro/composer
|
||||||
|
options: --volumes-from ${{ env.JOB_CONTAINER_NAME }} -w ${{ gitea.workspace }}
|
||||||
run: |
|
run: |
|
||||||
docker run --rm \
|
bin/pcov.sh
|
||||||
--volumes-from ${{ env.JOB_CONTAINER_NAME }} \
|
composer run tests:unit:coverage
|
||||||
-w ${{ github.workspace }} \
|
|
||||||
siteworxpro/composer \
|
# - name: 📦 Publish Build Artifacts
|
||||||
run tests:unit
|
# env:
|
||||||
|
# NODE_TLS_REJECT_UNAUTHORIZED: 0
|
||||||
|
# uses: christopherhx/gitea-upload-artifact@v4
|
||||||
|
# with:
|
||||||
|
# options: --volumes-from ${{ env.JOB_CONTAINER_NAME }} -w ${{ gitea.workspace }}
|
||||||
|
# name: junit-coverage.xml
|
||||||
|
# path: tests/reports/junit.xml
|
||||||
|
# retention-days: 1
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,3 +1,6 @@
|
|||||||
.idea/
|
.idea/
|
||||||
|
.DS_Store
|
||||||
vendor/
|
vendor/
|
||||||
.phpunit.cache/
|
.phpunit.cache/
|
||||||
|
|
||||||
|
tests/reports/
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
include:
|
|
||||||
- local: .gitlab/ci/stages.yml
|
|
||||||
- local: .gitlab/ci/tests.yml
|
|
||||||
- local: .gitlab/ci/libraries.yml
|
|
||||||
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
Install Composer Libraries:
|
|
||||||
stage: libraries
|
|
||||||
image: siteworxpro/composer:latest
|
|
||||||
rules:
|
|
||||||
- if: '$CI_COMMIT_TAG'
|
|
||||||
when: never
|
|
||||||
- if: '$CI_PIPELINE_SOURCE == "push"'
|
|
||||||
when: always
|
|
||||||
- when: never
|
|
||||||
script:
|
|
||||||
- composer install --ignore-platform-reqs
|
|
||||||
artifacts:
|
|
||||||
paths:
|
|
||||||
- vendor/
|
|
||||||
expire_in: 1 hour
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
stages:
|
|
||||||
- libraries
|
|
||||||
- tests
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
Unit Tests:
|
|
||||||
stage: tests
|
|
||||||
needs:
|
|
||||||
- Install Composer Libraries
|
|
||||||
rules:
|
|
||||||
- if: '$CI_COMMIT_TAG'
|
|
||||||
when: never
|
|
||||||
- if: '$CI_PIPELINE_SOURCE == "push"'
|
|
||||||
when: always
|
|
||||||
- when: never
|
|
||||||
image: siteworxpro/composer
|
|
||||||
before_script: |
|
|
||||||
bin/pcov.sh
|
|
||||||
script: |
|
|
||||||
echo "Running unit tests..."
|
|
||||||
composer run tests:unit:coverage
|
|
||||||
coverage: '/^\s*Lines:\s*\d+.\d+\%/'
|
|
||||||
artifacts:
|
|
||||||
expire_in: 1 day
|
|
||||||
reports:
|
|
||||||
junit: tests/reports/junit.xml
|
|
||||||
paths:
|
|
||||||
- tests/reports/
|
|
||||||
|
|
||||||
Run License Check:
|
|
||||||
stage: tests
|
|
||||||
needs:
|
|
||||||
- Install Composer Libraries
|
|
||||||
rules:
|
|
||||||
- if: '$CI_COMMIT_TAG'
|
|
||||||
when: never
|
|
||||||
- if: '$CI_PIPELINE_SOURCE == "push"'
|
|
||||||
when: on_success
|
|
||||||
- when: never
|
|
||||||
image: siteworxpro/composer
|
|
||||||
script:
|
|
||||||
- composer run tests:license
|
|
||||||
|
|
||||||
Run Code Lint:
|
|
||||||
stage: tests
|
|
||||||
needs:
|
|
||||||
- Install Composer Libraries
|
|
||||||
rules:
|
|
||||||
- if: '$CI_COMMIT_TAG'
|
|
||||||
when: never
|
|
||||||
- if: '$CI_PIPELINE_SOURCE == "push"'
|
|
||||||
when: on_success
|
|
||||||
- when: never
|
|
||||||
image: siteworxpro/composer
|
|
||||||
script:
|
|
||||||
- composer run tests:lint
|
|
||||||
|
|
||||||
Run Code Sniffer:
|
|
||||||
stage: tests
|
|
||||||
needs:
|
|
||||||
- Install Composer Libraries
|
|
||||||
rules:
|
|
||||||
- if: '$CI_COMMIT_TAG'
|
|
||||||
when: never
|
|
||||||
- if: '$CI_PIPELINE_SOURCE == "push"'
|
|
||||||
when: on_success
|
|
||||||
- when: never
|
|
||||||
image: siteworxpro/composer
|
|
||||||
script:
|
|
||||||
- composer run tests:phpstan
|
|
||||||
2
.rr.yaml
2
.rr.yaml
@@ -21,4 +21,4 @@ http:
|
|||||||
logs:
|
logs:
|
||||||
encoding: json
|
encoding: json
|
||||||
level: ${LOG_LEVEL:-info}
|
level: ${LOG_LEVEL:-info}
|
||||||
mode: production
|
mode: ${LOG_MODE:-production}
|
||||||
10
.run/ Compose Deployment.run.xml
Normal file
10
.run/ Compose Deployment.run.xml
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name=" Compose Deployment" type="docker-deploy" factoryName="docker-compose.yml" server-name="Docker">
|
||||||
|
<deployment type="docker-compose.yml">
|
||||||
|
<settings>
|
||||||
|
<option name="sourceFilePath" value="docker-compose.yml" />
|
||||||
|
</settings>
|
||||||
|
</deployment>
|
||||||
|
<method v="2" />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
8
.run/All.run.xml
Normal file
8
.run/All.run.xml
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="All" type="ComposerRunConfigurationType" factoryName="Composer Script">
|
||||||
|
<option name="commandLineParameters" value="" />
|
||||||
|
<option name="pathToComposerJson" value="$PROJECT_DIR$/composer.json" />
|
||||||
|
<option name="script" value="tests:all" />
|
||||||
|
<method v="2" />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
8
.run/Lint_fix.run.xml
Normal file
8
.run/Lint_fix.run.xml
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="Lint:fix" type="ComposerRunConfigurationType" factoryName="Composer Script">
|
||||||
|
<option name="commandLineParameters" value="" />
|
||||||
|
<option name="pathToComposerJson" value="$PROJECT_DIR$/composer.json" />
|
||||||
|
<option name="script" value="tests:lint:fix" />
|
||||||
|
<method v="2" />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
11
.run/Main.run.xml
Normal file
11
.run/Main.run.xml
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="Main" type="PHPUnitRunConfigurationType" factoryName="PHPUnit">
|
||||||
|
<CommandLine>
|
||||||
|
<PhpTestInterpreterSettings>
|
||||||
|
<option name="interpreterName" value="composer-runtime" />
|
||||||
|
</PhpTestInterpreterSettings>
|
||||||
|
</CommandLine>
|
||||||
|
<TestRunner configuration_file="$PROJECT_DIR$/phpunit.xml" scope="XML" use_alternative_configuration_file="true" />
|
||||||
|
<method v="2" />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
17
Dockerfile
17
Dockerfile
@@ -1,5 +1,5 @@
|
|||||||
# Use the RoadRunner image as a base for the first stage
|
# Use the RoadRunner image as a base for the first stage
|
||||||
FROM ghcr.io/roadrunner-server/roadrunner:2025.1.1 AS roadrunner
|
FROM ghcr.io/roadrunner-server/roadrunner:2025.1.4 AS roadrunner
|
||||||
|
|
||||||
# Use the official Composer image as the base for the library stage
|
# Use the official Composer image as the base for the library stage
|
||||||
FROM siteworxpro/composer AS library
|
FROM siteworxpro/composer AS library
|
||||||
@@ -12,14 +12,25 @@ RUN composer install --optimize-autoloader --ignore-platform-reqs --no-dev
|
|||||||
|
|
||||||
|
|
||||||
# Use the official PHP CLI image with Alpine Linux for the second stage
|
# Use the official PHP CLI image with Alpine Linux for the second stage
|
||||||
FROM php:8.4.6-alpine AS php
|
FROM php:8.4.14-alpine AS php
|
||||||
|
|
||||||
|
ARG KAFKA_ENABLED=0
|
||||||
|
|
||||||
# Move the production PHP configuration file to the default location
|
# Move the production PHP configuration file to the default location
|
||||||
RUN mv /usr/local/etc/php/php.ini-production /usr/local/etc/php/php.ini \
|
RUN mv /usr/local/etc/php/php.ini-production /usr/local/etc/php/php.ini \
|
||||||
&& apk add libpq-dev linux-headers --no-cache \
|
&& apk add libpq-dev linux-headers --no-cache \
|
||||||
&& docker-php-ext-install pdo_pgsql sockets \
|
&& docker-php-ext-install pdo_pgsql sockets pcntl \
|
||||||
&& rm -rf /var/cache/apk/*
|
&& rm -rf /var/cache/apk/*
|
||||||
|
|
||||||
|
RUN if [ "$KAFKA_ENABLED" -eq 1 ] ; then \
|
||||||
|
echo "Kafka support enabled" ; \
|
||||||
|
apk add autoconf g++ librdkafka-dev make --no-cache ; \
|
||||||
|
pecl install rdkafka && docker-php-ext-enable rdkafka ; \
|
||||||
|
else \
|
||||||
|
echo "Kafka support disabled" ; \
|
||||||
|
exit 0 ; \
|
||||||
|
fi
|
||||||
|
|
||||||
# Set the working directory to /app
|
# Set the working directory to /app
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ You can access the api at `http://localhost:9501/`
|
|||||||
|
|
||||||
xdebug needs to be built into the container before it will work
|
xdebug needs to be built into the container before it will work
|
||||||
```shell
|
```shell
|
||||||
docker exec -it template-runtime-1 bin/xdebug.sh
|
docker exec -it php-template-composer-runtime-1 bin/xdebug.sh
|
||||||
```
|
```
|
||||||
|
|
||||||
### Install the dependencies
|
### Install the dependencies
|
||||||
|
|||||||
4
bin/migrate.sh
Executable file
4
bin/migrate.sh
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
eval #!/bin/sh
|
||||||
|
set -e
|
||||||
|
|
||||||
|
migrate -path /app/db/migrations -database "postgres://$DB_USERNAME:$DB_PASSWORD@$DB_HOST:$DB_PORT/$DB_DATABASE?sslmode=disable" up
|
||||||
11
cli.php
Executable file
11
cli.php
Executable file
@@ -0,0 +1,11 @@
|
|||||||
|
#!/usr/local/bin/php
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
require __DIR__ . '/vendor/autoload.php';
|
||||||
|
|
||||||
|
use Siteworxpro\App\Cli\App;
|
||||||
|
|
||||||
|
$cliApp = new App();
|
||||||
|
exit($cliApp->run());
|
||||||
@@ -16,14 +16,20 @@
|
|||||||
"illuminate/support": "^v12.10.2",
|
"illuminate/support": "^v12.10.2",
|
||||||
"roadrunner-php/app-logger": "^1.2.0",
|
"roadrunner-php/app-logger": "^1.2.0",
|
||||||
"siteworxpro/config": "^1.1.1",
|
"siteworxpro/config": "^1.1.1",
|
||||||
"predis/predis": "^v3.2.0"
|
"predis/predis": "^v3.2.0",
|
||||||
|
"siteworxpro/http-status": "0.0.2",
|
||||||
|
"lcobucci/jwt": "^5.6",
|
||||||
|
"adhocore/cli": "^1.9",
|
||||||
|
"robinvdvleuten/ulid": "^5.0",
|
||||||
|
"monolog/monolog": "^3.9"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"phpunit/phpunit": "^12.4",
|
"phpunit/phpunit": "^12.4",
|
||||||
"mockery/mockery": "^1.6",
|
"mockery/mockery": "^1.6",
|
||||||
"squizlabs/php_codesniffer": "^3.12",
|
"squizlabs/php_codesniffer": "^3.12",
|
||||||
"lendable/composer-license-checker": "^1.2",
|
"lendable/composer-license-checker": "^1.2",
|
||||||
"phpstan/phpstan": "^2.1.31"
|
"phpstan/phpstan": "^2.1.31",
|
||||||
|
"kwn/php-rdkafka-stubs": "^2.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"tests:all": [
|
"tests:all": [
|
||||||
|
|||||||
521
composer.lock
generated
521
composer.lock
generated
@@ -4,8 +4,81 @@
|
|||||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
||||||
"This file is @generated automatically"
|
"This file is @generated automatically"
|
||||||
],
|
],
|
||||||
"content-hash": "df98926488dc1be80080ae38a55b6f97",
|
"content-hash": "7c2d40400d6f4d0469324dc1645eba3c",
|
||||||
"packages": [
|
"packages": [
|
||||||
|
{
|
||||||
|
"name": "adhocore/cli",
|
||||||
|
"version": "v1.9.4",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/adhocore/php-cli.git",
|
||||||
|
"reference": "474dc3d7ab139796be98b104d891476e3916b6f4"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/adhocore/php-cli/zipball/474dc3d7ab139796be98b104d891476e3916b6f4",
|
||||||
|
"reference": "474dc3d7ab139796be98b104d891476e3916b6f4",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": ">=8.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"phpunit/phpunit": "^9.0"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"files": [
|
||||||
|
"src/functions.php"
|
||||||
|
],
|
||||||
|
"psr-4": {
|
||||||
|
"Ahc\\Cli\\": "src/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Jitendra Adhikari",
|
||||||
|
"email": "jiten.adhikary@gmail.com"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "Command line interface library for PHP",
|
||||||
|
"keywords": [
|
||||||
|
"argument-parser",
|
||||||
|
"argv-parser",
|
||||||
|
"cli",
|
||||||
|
"cli-action",
|
||||||
|
"cli-app",
|
||||||
|
"cli-color",
|
||||||
|
"cli-option",
|
||||||
|
"cli-writer",
|
||||||
|
"command",
|
||||||
|
"console",
|
||||||
|
"console-app",
|
||||||
|
"php-cli",
|
||||||
|
"php8",
|
||||||
|
"stream-input",
|
||||||
|
"stream-output"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/adhocore/php-cli/issues",
|
||||||
|
"source": "https://github.com/adhocore/php-cli/tree/v1.9.4"
|
||||||
|
},
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"url": "https://paypal.me/ji10",
|
||||||
|
"type": "custom"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://github.com/adhocore",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"time": "2025-05-11T13:23:54+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "brick/math",
|
"name": "brick/math",
|
||||||
"version": "0.14.0",
|
"version": "0.14.0",
|
||||||
@@ -227,16 +300,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "google/protobuf",
|
"name": "google/protobuf",
|
||||||
"version": "v4.32.1",
|
"version": "v4.33.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/protocolbuffers/protobuf-php.git",
|
"url": "https://github.com/protocolbuffers/protobuf-php.git",
|
||||||
"reference": "c4ed1c1f9bbc1e91766e2cd6c0af749324fe87cb"
|
"reference": "b50269e23204e5ae859a326ec3d90f09efe3047d"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/protocolbuffers/protobuf-php/zipball/c4ed1c1f9bbc1e91766e2cd6c0af749324fe87cb",
|
"url": "https://api.github.com/repos/protocolbuffers/protobuf-php/zipball/b50269e23204e5ae859a326ec3d90f09efe3047d",
|
||||||
"reference": "c4ed1c1f9bbc1e91766e2cd6c0af749324fe87cb",
|
"reference": "b50269e23204e5ae859a326ec3d90f09efe3047d",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -265,22 +338,22 @@
|
|||||||
"proto"
|
"proto"
|
||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"source": "https://github.com/protocolbuffers/protobuf-php/tree/v4.32.1"
|
"source": "https://github.com/protocolbuffers/protobuf-php/tree/v4.33.0"
|
||||||
},
|
},
|
||||||
"time": "2025-09-14T05:14:52+00:00"
|
"time": "2025-10-15T20:10:28+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/collections",
|
"name": "illuminate/collections",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/collections.git",
|
"url": "https://github.com/illuminate/collections.git",
|
||||||
"reference": "b323866d9e571f8c444f3ccca6f645c05fadf568"
|
"reference": "deb291b109b6f7fd776a3550a120771137b3c5d1"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/illuminate/collections/zipball/b323866d9e571f8c444f3ccca6f645c05fadf568",
|
"url": "https://api.github.com/repos/illuminate/collections/zipball/deb291b109b6f7fd776a3550a120771137b3c5d1",
|
||||||
"reference": "b323866d9e571f8c444f3ccca6f645c05fadf568",
|
"reference": "deb291b109b6f7fd776a3550a120771137b3c5d1",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -326,11 +399,11 @@
|
|||||||
"issues": "https://github.com/laravel/framework/issues",
|
"issues": "https://github.com/laravel/framework/issues",
|
||||||
"source": "https://github.com/laravel/framework"
|
"source": "https://github.com/laravel/framework"
|
||||||
},
|
},
|
||||||
"time": "2025-10-10T13:31:43+00:00"
|
"time": "2025-10-30T12:22:05+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/conditionable",
|
"name": "illuminate/conditionable",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/conditionable.git",
|
"url": "https://github.com/illuminate/conditionable.git",
|
||||||
@@ -376,7 +449,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/container",
|
"name": "illuminate/container",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/container.git",
|
"url": "https://github.com/illuminate/container.git",
|
||||||
@@ -437,7 +510,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/contracts",
|
"name": "illuminate/contracts",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/contracts.git",
|
"url": "https://github.com/illuminate/contracts.git",
|
||||||
@@ -485,16 +558,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/database",
|
"name": "illuminate/database",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/database.git",
|
"url": "https://github.com/illuminate/database.git",
|
||||||
"reference": "3ad07bda64019d18fc6fda97fec0b3b7cb6ecae1"
|
"reference": "eacbdddf31f655fba5406fdf31bd264d880dd1a8"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/illuminate/database/zipball/3ad07bda64019d18fc6fda97fec0b3b7cb6ecae1",
|
"url": "https://api.github.com/repos/illuminate/database/zipball/eacbdddf31f655fba5406fdf31bd264d880dd1a8",
|
||||||
"reference": "3ad07bda64019d18fc6fda97fec0b3b7cb6ecae1",
|
"reference": "eacbdddf31f655fba5406fdf31bd264d880dd1a8",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -552,11 +625,11 @@
|
|||||||
"issues": "https://github.com/laravel/framework/issues",
|
"issues": "https://github.com/laravel/framework/issues",
|
||||||
"source": "https://github.com/laravel/framework"
|
"source": "https://github.com/laravel/framework"
|
||||||
},
|
},
|
||||||
"time": "2025-10-10T13:33:40+00:00"
|
"time": "2025-11-11T14:13:21+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/macroable",
|
"name": "illuminate/macroable",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/macroable.git",
|
"url": "https://github.com/illuminate/macroable.git",
|
||||||
@@ -602,16 +675,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "illuminate/support",
|
"name": "illuminate/support",
|
||||||
"version": "v12.34.0",
|
"version": "v12.38.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/illuminate/support.git",
|
"url": "https://github.com/illuminate/support.git",
|
||||||
"reference": "89291f59ef6c170c00f10a41c566c49ee32ca09a"
|
"reference": "008b6c0d45f548de0f801d60a5854a7f9e4dd32f"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/illuminate/support/zipball/89291f59ef6c170c00f10a41c566c49ee32ca09a",
|
"url": "https://api.github.com/repos/illuminate/support/zipball/008b6c0d45f548de0f801d60a5854a7f9e4dd32f",
|
||||||
"reference": "89291f59ef6c170c00f10a41c566c49ee32ca09a",
|
"reference": "008b6c0d45f548de0f801d60a5854a7f9e4dd32f",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -677,7 +750,7 @@
|
|||||||
"issues": "https://github.com/laravel/framework/issues",
|
"issues": "https://github.com/laravel/framework/issues",
|
||||||
"source": "https://github.com/laravel/framework"
|
"source": "https://github.com/laravel/framework"
|
||||||
},
|
},
|
||||||
"time": "2025-10-13T21:11:33+00:00"
|
"time": "2025-11-06T14:27:18+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "laravel/serializable-closure",
|
"name": "laravel/serializable-closure",
|
||||||
@@ -740,6 +813,79 @@
|
|||||||
},
|
},
|
||||||
"time": "2025-10-09T13:42:30+00:00"
|
"time": "2025-10-09T13:42:30+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "lcobucci/jwt",
|
||||||
|
"version": "5.6.0",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/lcobucci/jwt.git",
|
||||||
|
"reference": "bb3e9f21e4196e8afc41def81ef649c164bca25e"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/lcobucci/jwt/zipball/bb3e9f21e4196e8afc41def81ef649c164bca25e",
|
||||||
|
"reference": "bb3e9f21e4196e8afc41def81ef649c164bca25e",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"ext-openssl": "*",
|
||||||
|
"ext-sodium": "*",
|
||||||
|
"php": "~8.2.0 || ~8.3.0 || ~8.4.0 || ~8.5.0",
|
||||||
|
"psr/clock": "^1.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"infection/infection": "^0.29",
|
||||||
|
"lcobucci/clock": "^3.2",
|
||||||
|
"lcobucci/coding-standard": "^11.0",
|
||||||
|
"phpbench/phpbench": "^1.2",
|
||||||
|
"phpstan/extension-installer": "^1.2",
|
||||||
|
"phpstan/phpstan": "^1.10.7",
|
||||||
|
"phpstan/phpstan-deprecation-rules": "^1.1.3",
|
||||||
|
"phpstan/phpstan-phpunit": "^1.3.10",
|
||||||
|
"phpstan/phpstan-strict-rules": "^1.5.0",
|
||||||
|
"phpunit/phpunit": "^11.1"
|
||||||
|
},
|
||||||
|
"suggest": {
|
||||||
|
"lcobucci/clock": ">= 3.2"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Lcobucci\\JWT\\": "src"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"BSD-3-Clause"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Luís Cobucci",
|
||||||
|
"email": "lcobucci@gmail.com",
|
||||||
|
"role": "Developer"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "A simple library to work with JSON Web Token and JSON Web Signature",
|
||||||
|
"keywords": [
|
||||||
|
"JWS",
|
||||||
|
"jwt"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/lcobucci/jwt/issues",
|
||||||
|
"source": "https://github.com/lcobucci/jwt/tree/5.6.0"
|
||||||
|
},
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"url": "https://github.com/lcobucci",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://www.patreon.com/lcobucci",
|
||||||
|
"type": "patreon"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"time": "2025-10-17T11:30:53+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "league/route",
|
"name": "league/route",
|
||||||
"version": "6.2.0",
|
"version": "6.2.0",
|
||||||
@@ -830,6 +976,109 @@
|
|||||||
],
|
],
|
||||||
"time": "2024-11-25T08:10:15+00:00"
|
"time": "2024-11-25T08:10:15+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "monolog/monolog",
|
||||||
|
"version": "3.9.0",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/Seldaek/monolog.git",
|
||||||
|
"reference": "10d85740180ecba7896c87e06a166e0c95a0e3b6"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/Seldaek/monolog/zipball/10d85740180ecba7896c87e06a166e0c95a0e3b6",
|
||||||
|
"reference": "10d85740180ecba7896c87e06a166e0c95a0e3b6",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": ">=8.1",
|
||||||
|
"psr/log": "^2.0 || ^3.0"
|
||||||
|
},
|
||||||
|
"provide": {
|
||||||
|
"psr/log-implementation": "3.0.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"aws/aws-sdk-php": "^3.0",
|
||||||
|
"doctrine/couchdb": "~1.0@dev",
|
||||||
|
"elasticsearch/elasticsearch": "^7 || ^8",
|
||||||
|
"ext-json": "*",
|
||||||
|
"graylog2/gelf-php": "^1.4.2 || ^2.0",
|
||||||
|
"guzzlehttp/guzzle": "^7.4.5",
|
||||||
|
"guzzlehttp/psr7": "^2.2",
|
||||||
|
"mongodb/mongodb": "^1.8",
|
||||||
|
"php-amqplib/php-amqplib": "~2.4 || ^3",
|
||||||
|
"php-console/php-console": "^3.1.8",
|
||||||
|
"phpstan/phpstan": "^2",
|
||||||
|
"phpstan/phpstan-deprecation-rules": "^2",
|
||||||
|
"phpstan/phpstan-strict-rules": "^2",
|
||||||
|
"phpunit/phpunit": "^10.5.17 || ^11.0.7",
|
||||||
|
"predis/predis": "^1.1 || ^2",
|
||||||
|
"rollbar/rollbar": "^4.0",
|
||||||
|
"ruflin/elastica": "^7 || ^8",
|
||||||
|
"symfony/mailer": "^5.4 || ^6",
|
||||||
|
"symfony/mime": "^5.4 || ^6"
|
||||||
|
},
|
||||||
|
"suggest": {
|
||||||
|
"aws/aws-sdk-php": "Allow sending log messages to AWS services like DynamoDB",
|
||||||
|
"doctrine/couchdb": "Allow sending log messages to a CouchDB server",
|
||||||
|
"elasticsearch/elasticsearch": "Allow sending log messages to an Elasticsearch server via official client",
|
||||||
|
"ext-amqp": "Allow sending log messages to an AMQP server (1.0+ required)",
|
||||||
|
"ext-curl": "Required to send log messages using the IFTTTHandler, the LogglyHandler, the SendGridHandler, the SlackWebhookHandler or the TelegramBotHandler",
|
||||||
|
"ext-mbstring": "Allow to work properly with unicode symbols",
|
||||||
|
"ext-mongodb": "Allow sending log messages to a MongoDB server (via driver)",
|
||||||
|
"ext-openssl": "Required to send log messages using SSL",
|
||||||
|
"ext-sockets": "Allow sending log messages to a Syslog server (via UDP driver)",
|
||||||
|
"graylog2/gelf-php": "Allow sending log messages to a GrayLog2 server",
|
||||||
|
"mongodb/mongodb": "Allow sending log messages to a MongoDB server (via library)",
|
||||||
|
"php-amqplib/php-amqplib": "Allow sending log messages to an AMQP server using php-amqplib",
|
||||||
|
"rollbar/rollbar": "Allow sending log messages to Rollbar",
|
||||||
|
"ruflin/elastica": "Allow sending log messages to an Elastic Search server"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"extra": {
|
||||||
|
"branch-alias": {
|
||||||
|
"dev-main": "3.x-dev"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Monolog\\": "src/Monolog"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Jordi Boggiano",
|
||||||
|
"email": "j.boggiano@seld.be",
|
||||||
|
"homepage": "https://seld.be"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "Sends your logs to files, sockets, inboxes, databases and various web services",
|
||||||
|
"homepage": "https://github.com/Seldaek/monolog",
|
||||||
|
"keywords": [
|
||||||
|
"log",
|
||||||
|
"logging",
|
||||||
|
"psr-3"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/Seldaek/monolog/issues",
|
||||||
|
"source": "https://github.com/Seldaek/monolog/tree/3.9.0"
|
||||||
|
},
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"url": "https://github.com/Seldaek",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://tidelift.com/funding/github/packagist/monolog/monolog",
|
||||||
|
"type": "tidelift"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"time": "2025-03-24T10:02:05+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "nesbot/carbon",
|
"name": "nesbot/carbon",
|
||||||
"version": "3.10.3",
|
"version": "3.10.3",
|
||||||
@@ -1608,16 +1857,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "roadrunner-php/roadrunner-api-dto",
|
"name": "roadrunner-php/roadrunner-api-dto",
|
||||||
"version": "v1.13.0",
|
"version": "v1.14.0",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/roadrunner-php/roadrunner-api-dto.git",
|
"url": "https://github.com/roadrunner-php/roadrunner-api-dto.git",
|
||||||
"reference": "8a683f5057005bef742916847c0befbf9a00c543"
|
"reference": "e6efb759f0a73b8516b7f28317230ecd4010005e"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/roadrunner-php/roadrunner-api-dto/zipball/8a683f5057005bef742916847c0befbf9a00c543",
|
"url": "https://api.github.com/repos/roadrunner-php/roadrunner-api-dto/zipball/e6efb759f0a73b8516b7f28317230ecd4010005e",
|
||||||
"reference": "8a683f5057005bef742916847c0befbf9a00c543",
|
"reference": "e6efb759f0a73b8516b7f28317230ecd4010005e",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -1663,7 +1912,7 @@
|
|||||||
"docs": "https://docs.roadrunner.dev",
|
"docs": "https://docs.roadrunner.dev",
|
||||||
"forum": "https://forum.roadrunner.dev",
|
"forum": "https://forum.roadrunner.dev",
|
||||||
"issues": "https://github.com/roadrunner-server/roadrunner/issues",
|
"issues": "https://github.com/roadrunner-server/roadrunner/issues",
|
||||||
"source": "https://github.com/roadrunner-php/roadrunner-api-dto/tree/v1.13.0"
|
"source": "https://github.com/roadrunner-php/roadrunner-api-dto/tree/v1.14.0"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -1671,15 +1920,61 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-08-12T14:04:38+00:00"
|
"time": "2025-11-06T13:03:11+00:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "robinvdvleuten/ulid",
|
||||||
|
"version": "v5.0.0",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/robinvdvleuten/php-ulid.git",
|
||||||
|
"reference": "5389c9a2ff020815cc1f2b840334fdcb84ae3f35"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/robinvdvleuten/php-ulid/zipball/5389c9a2ff020815cc1f2b840334fdcb84ae3f35",
|
||||||
|
"reference": "5389c9a2ff020815cc1f2b840334fdcb84ae3f35",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": "^7.2|^8.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"phpbench/phpbench": "^1.0.0-alpha3",
|
||||||
|
"phpunit/phpunit": "^8.5",
|
||||||
|
"symfony/phpunit-bridge": "^5.1"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Ulid\\": "src"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Robin van der Vleuten",
|
||||||
|
"email": "robin@webstronauts.co"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "Universally Unique Lexicographically Sortable Identifier (ULID) implementation for PHP.",
|
||||||
|
"homepage": "https://github.com/robinvdvleuten/php-ulid",
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/robinvdvleuten/php-ulid/issues",
|
||||||
|
"source": "https://github.com/robinvdvleuten/php-ulid/tree/v5.0.0"
|
||||||
|
},
|
||||||
|
"time": "2020-12-06T19:13:21+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "siteworxpro/config",
|
"name": "siteworxpro/config",
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "",
|
"type": "git",
|
||||||
"url": "",
|
"url": "https://gitea.siteworxpro.com/php-packages/config",
|
||||||
"reference": ""
|
"reference": "1.1.1"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
@@ -1739,6 +2034,33 @@
|
|||||||
],
|
],
|
||||||
"time": "2025-08-15T19:08:49+00:00"
|
"time": "2025-08-15T19:08:49+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "siteworxpro/http-status",
|
||||||
|
"version": "0.0.2",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://gitea.siteworxpro.com/php-packages/http-status",
|
||||||
|
"reference": "0.0.2"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://gitea.siteworxpro.com/api/packages/php-packages/composer/files/siteworxpro%2Fhttp-status/0.0.2/siteworxpro-http-status.0.0.2.zip",
|
||||||
|
"shasum": "2eee4cd2605aa4b64ce18d18eb651764e9e88dbf"
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": ">=8.4"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Siteworxpro\\HttpStatus\\": "src/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"time": "2025-06-20T12:46:36+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "spiral/goridge",
|
"name": "spiral/goridge",
|
||||||
"version": "4.2.1",
|
"version": "4.2.1",
|
||||||
@@ -2619,16 +2941,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/translation-contracts",
|
"name": "symfony/translation-contracts",
|
||||||
"version": "v3.6.0",
|
"version": "v3.6.1",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/translation-contracts.git",
|
"url": "https://github.com/symfony/translation-contracts.git",
|
||||||
"reference": "df210c7a2573f1913b2d17cc95f90f53a73d8f7d"
|
"reference": "65a8bc82080447fae78373aa10f8d13b38338977"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/translation-contracts/zipball/df210c7a2573f1913b2d17cc95f90f53a73d8f7d",
|
"url": "https://api.github.com/repos/symfony/translation-contracts/zipball/65a8bc82080447fae78373aa10f8d13b38338977",
|
||||||
"reference": "df210c7a2573f1913b2d17cc95f90f53a73d8f7d",
|
"reference": "65a8bc82080447fae78373aa10f8d13b38338977",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -2677,7 +2999,7 @@
|
|||||||
"standards"
|
"standards"
|
||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"source": "https://github.com/symfony/translation-contracts/tree/v3.6.0"
|
"source": "https://github.com/symfony/translation-contracts/tree/v3.6.1"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -2688,12 +3010,16 @@
|
|||||||
"url": "https://github.com/fabpot",
|
"url": "https://github.com/fabpot",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"url": "https://github.com/nicolas-grekas",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2024-09-27T08:32:26+00:00"
|
"time": "2025-07-15T13:41:35+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "voku/portable-ascii",
|
"name": "voku/portable-ascii",
|
||||||
@@ -2822,6 +3148,44 @@
|
|||||||
},
|
},
|
||||||
"time": "2025-04-30T06:54:44+00:00"
|
"time": "2025-04-30T06:54:44+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "kwn/php-rdkafka-stubs",
|
||||||
|
"version": "v2.2.1",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/kwn/php-rdkafka-stubs.git",
|
||||||
|
"reference": "23b865d6b3e8fe1f080aa7371dc1da3339361996"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/kwn/php-rdkafka-stubs/zipball/23b865d6b3e8fe1f080aa7371dc1da3339361996",
|
||||||
|
"reference": "23b865d6b3e8fe1f080aa7371dc1da3339361996",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"ext-rdkafka": ">=4.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"phpunit/phpunit": "^8.2.4"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Karol Wnuk",
|
||||||
|
"email": "k.wnuk@ascetic.pl"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "Rdkafka extension stubs for your IDE",
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/kwn/php-rdkafka-stubs/issues",
|
||||||
|
"source": "https://github.com/kwn/php-rdkafka-stubs/tree/v2.2.1"
|
||||||
|
},
|
||||||
|
"time": "2022-08-16T15:27:51+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "lendable/composer-license-checker",
|
"name": "lendable/composer-license-checker",
|
||||||
"version": "1.2.2",
|
"version": "1.2.2",
|
||||||
@@ -3025,16 +3389,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "nikic/php-parser",
|
"name": "nikic/php-parser",
|
||||||
"version": "v5.6.1",
|
"version": "v5.6.2",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/nikic/PHP-Parser.git",
|
"url": "https://github.com/nikic/PHP-Parser.git",
|
||||||
"reference": "f103601b29efebd7ff4a1ca7b3eeea9e3336a2a2"
|
"reference": "3a454ca033b9e06b63282ce19562e892747449bb"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/f103601b29efebd7ff4a1ca7b3eeea9e3336a2a2",
|
"url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/3a454ca033b9e06b63282ce19562e892747449bb",
|
||||||
"reference": "f103601b29efebd7ff4a1ca7b3eeea9e3336a2a2",
|
"reference": "3a454ca033b9e06b63282ce19562e892747449bb",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -3077,9 +3441,9 @@
|
|||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"issues": "https://github.com/nikic/PHP-Parser/issues",
|
"issues": "https://github.com/nikic/PHP-Parser/issues",
|
||||||
"source": "https://github.com/nikic/PHP-Parser/tree/v5.6.1"
|
"source": "https://github.com/nikic/PHP-Parser/tree/v5.6.2"
|
||||||
},
|
},
|
||||||
"time": "2025-08-13T20:13:15+00:00"
|
"time": "2025-10-21T19:32:17+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "phar-io/manifest",
|
"name": "phar-io/manifest",
|
||||||
@@ -3201,11 +3565,11 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "phpstan/phpstan",
|
"name": "phpstan/phpstan",
|
||||||
"version": "2.1.31",
|
"version": "2.1.32",
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/phpstan/phpstan/zipball/ead89849d879fe203ce9292c6ef5e7e76f867b96",
|
"url": "https://api.github.com/repos/phpstan/phpstan/zipball/e126cad1e30a99b137b8ed75a85a676450ebb227",
|
||||||
"reference": "ead89849d879fe203ce9292c6ef5e7e76f867b96",
|
"reference": "e126cad1e30a99b137b8ed75a85a676450ebb227",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -3250,7 +3614,7 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-10-10T14:14:11+00:00"
|
"time": "2025-11-11T15:18:17+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "phpunit/php-code-coverage",
|
"name": "phpunit/php-code-coverage",
|
||||||
@@ -3588,16 +3952,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "phpunit/phpunit",
|
"name": "phpunit/phpunit",
|
||||||
"version": "12.4.1",
|
"version": "12.4.2",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/sebastianbergmann/phpunit.git",
|
"url": "https://github.com/sebastianbergmann/phpunit.git",
|
||||||
"reference": "fc5413a2e6d240d2f6d9317bdf7f0a24e73de194"
|
"reference": "a94ea4d26d865875803b23aaf78c3c2c670ea2ea"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/fc5413a2e6d240d2f6d9317bdf7f0a24e73de194",
|
"url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/a94ea4d26d865875803b23aaf78c3c2c670ea2ea",
|
||||||
"reference": "fc5413a2e6d240d2f6d9317bdf7f0a24e73de194",
|
"reference": "a94ea4d26d865875803b23aaf78c3c2c670ea2ea",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -3665,7 +4029,7 @@
|
|||||||
"support": {
|
"support": {
|
||||||
"issues": "https://github.com/sebastianbergmann/phpunit/issues",
|
"issues": "https://github.com/sebastianbergmann/phpunit/issues",
|
||||||
"security": "https://github.com/sebastianbergmann/phpunit/security/policy",
|
"security": "https://github.com/sebastianbergmann/phpunit/security/policy",
|
||||||
"source": "https://github.com/sebastianbergmann/phpunit/tree/12.4.1"
|
"source": "https://github.com/sebastianbergmann/phpunit/tree/12.4.2"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -3689,7 +4053,7 @@
|
|||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-10-09T14:08:29+00:00"
|
"time": "2025-10-30T08:41:39+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "sebastian/cli-parser",
|
"name": "sebastian/cli-parser",
|
||||||
@@ -4590,16 +4954,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "squizlabs/php_codesniffer",
|
"name": "squizlabs/php_codesniffer",
|
||||||
"version": "3.13.4",
|
"version": "3.13.5",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/PHPCSStandards/PHP_CodeSniffer.git",
|
"url": "https://github.com/PHPCSStandards/PHP_CodeSniffer.git",
|
||||||
"reference": "ad545ea9c1b7d270ce0fc9cbfb884161cd706119"
|
"reference": "0ca86845ce43291e8f5692c7356fccf3bcf02bf4"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/ad545ea9c1b7d270ce0fc9cbfb884161cd706119",
|
"url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/0ca86845ce43291e8f5692c7356fccf3bcf02bf4",
|
||||||
"reference": "ad545ea9c1b7d270ce0fc9cbfb884161cd706119",
|
"reference": "0ca86845ce43291e8f5692c7356fccf3bcf02bf4",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -4616,11 +4980,6 @@
|
|||||||
"bin/phpcs"
|
"bin/phpcs"
|
||||||
],
|
],
|
||||||
"type": "library",
|
"type": "library",
|
||||||
"extra": {
|
|
||||||
"branch-alias": {
|
|
||||||
"dev-master": "3.x-dev"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"notification-url": "https://packagist.org/downloads/",
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
"license": [
|
"license": [
|
||||||
"BSD-3-Clause"
|
"BSD-3-Clause"
|
||||||
@@ -4670,7 +5029,7 @@
|
|||||||
"type": "thanks_dev"
|
"type": "thanks_dev"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-09-05T05:47:09+00:00"
|
"time": "2025-11-04T16:30:35+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "staabm/side-effects-detector",
|
"name": "staabm/side-effects-detector",
|
||||||
@@ -4726,16 +5085,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/console",
|
"name": "symfony/console",
|
||||||
"version": "v7.3.4",
|
"version": "v7.3.6",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/console.git",
|
"url": "https://github.com/symfony/console.git",
|
||||||
"reference": "2b9c5fafbac0399a20a2e82429e2bd735dcfb7db"
|
"reference": "c28ad91448f86c5f6d9d2c70f0cf68bf135f252a"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/console/zipball/2b9c5fafbac0399a20a2e82429e2bd735dcfb7db",
|
"url": "https://api.github.com/repos/symfony/console/zipball/c28ad91448f86c5f6d9d2c70f0cf68bf135f252a",
|
||||||
"reference": "2b9c5fafbac0399a20a2e82429e2bd735dcfb7db",
|
"reference": "c28ad91448f86c5f6d9d2c70f0cf68bf135f252a",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -4800,7 +5159,7 @@
|
|||||||
"terminal"
|
"terminal"
|
||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"source": "https://github.com/symfony/console/tree/v7.3.4"
|
"source": "https://github.com/symfony/console/tree/v7.3.6"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -4820,7 +5179,7 @@
|
|||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-09-22T15:31:00+00:00"
|
"time": "2025-11-04T01:21:42+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/polyfill-ctype",
|
"name": "symfony/polyfill-ctype",
|
||||||
@@ -5139,16 +5498,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/service-contracts",
|
"name": "symfony/service-contracts",
|
||||||
"version": "v3.6.0",
|
"version": "v3.6.1",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/symfony/service-contracts.git",
|
"url": "https://github.com/symfony/service-contracts.git",
|
||||||
"reference": "f021b05a130d35510bd6b25fe9053c2a8a15d5d4"
|
"reference": "45112560a3ba2d715666a509a0bc9521d10b6c43"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/symfony/service-contracts/zipball/f021b05a130d35510bd6b25fe9053c2a8a15d5d4",
|
"url": "https://api.github.com/repos/symfony/service-contracts/zipball/45112560a3ba2d715666a509a0bc9521d10b6c43",
|
||||||
"reference": "f021b05a130d35510bd6b25fe9053c2a8a15d5d4",
|
"reference": "45112560a3ba2d715666a509a0bc9521d10b6c43",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -5202,7 +5561,7 @@
|
|||||||
"standards"
|
"standards"
|
||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"source": "https://github.com/symfony/service-contracts/tree/v3.6.0"
|
"source": "https://github.com/symfony/service-contracts/tree/v3.6.1"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -5213,12 +5572,16 @@
|
|||||||
"url": "https://github.com/fabpot",
|
"url": "https://github.com/fabpot",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"url": "https://github.com/nicolas-grekas",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
|
||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2025-04-25T09:37:31+00:00"
|
"time": "2025-07-15T11:30:57+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "symfony/string",
|
"name": "symfony/string",
|
||||||
|
|||||||
48
config.php
48
config.php
@@ -21,8 +21,14 @@ return [
|
|||||||
'database' => Env::get('DB_DATABASE', 'siteworxpro'),
|
'database' => Env::get('DB_DATABASE', 'siteworxpro'),
|
||||||
'username' => Env::get('DB_USERNAME', 'siteworxpro'),
|
'username' => Env::get('DB_USERNAME', 'siteworxpro'),
|
||||||
'password' => Env::get('DB_PASSWORD', 'password'),
|
'password' => Env::get('DB_PASSWORD', 'password'),
|
||||||
|
'port' => Env::get('DB_PORT', 5432, 'int'),
|
||||||
|
'charset' => Env::get('DB_CHARSET', 'utf8'),
|
||||||
|
'collation' => Env::get('DB_COLLATION', 'utf8_unicode_ci'),
|
||||||
|
'prefix' => Env::get('DB_PREFIX', ''),
|
||||||
|
'options' => [
|
||||||
|
// Add any additional PDO options here
|
||||||
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
||||||
'cors' => [
|
'cors' => [
|
||||||
'allowed_origins' => Env::get('CORS_ALLOWED_ORIGINS', 'localhost:3000'),
|
'allowed_origins' => Env::get('CORS_ALLOWED_ORIGINS', 'localhost:3000'),
|
||||||
@@ -34,5 +40,45 @@ return [
|
|||||||
'host' => Env::get('REDIS_HOST', 'localhost'),
|
'host' => Env::get('REDIS_HOST', 'localhost'),
|
||||||
'port' => Env::get('REDIS_PORT', 6379, 'int'),
|
'port' => Env::get('REDIS_PORT', 6379, 'int'),
|
||||||
'database' => Env::get('REDIS_DATABASE', 0, 'int'),
|
'database' => Env::get('REDIS_DATABASE', 0, 'int'),
|
||||||
|
'password' => Env::get('REDIS_PASSWORD'),
|
||||||
|
],
|
||||||
|
|
||||||
|
'jwt' => [
|
||||||
|
'signing_key' => Env::get('JWT_SIGNING_KEY', 'a_super_secret_key'),
|
||||||
|
'audience' => Env::get('JWT_AUDIENCE', 'my_audience'),
|
||||||
|
'issuer' => Env::get('JWT_ISSUER', 'my_issuer'),
|
||||||
|
'strict_validation' => Env::get('JWT_STRICT_VALIDATION', true, 'bool'),
|
||||||
|
],
|
||||||
|
|
||||||
|
'queue' => [
|
||||||
|
'broker' => Env::get('QUEUE_BROKER', 'redis'),
|
||||||
|
|
||||||
|
'broker_config' => [
|
||||||
|
|
||||||
|
'redis' => [
|
||||||
|
'consumerGroup' => Env::get('QUEUE_REDIS_CONSUMER_GROUP', ''),
|
||||||
|
],
|
||||||
|
|
||||||
|
'kafka' => [
|
||||||
|
'brokers' => Env::get('QUEUE_KAFKA_BROKERS', 'kafka:9092'),
|
||||||
|
'consumerGroup' => Env::get('QUEUE_KAFKA_CONSUMER_GROUP', 'default_group'),
|
||||||
|
],
|
||||||
|
|
||||||
|
'rabbitmq' => [
|
||||||
|
'host' => Env::get('QUEUE_RABBITMQ_HOST', 'localhost'),
|
||||||
|
'port' => Env::get('QUEUE_RABBITMQ_PORT', 5672, 'int'),
|
||||||
|
'username' => Env::get('QUEUE_RABBITMQ_USERNAME', 'guest'),
|
||||||
|
'password' => Env::get('QUEUE_RABBITMQ_PASSWORD', 'guest'),
|
||||||
|
'vhost' => Env::get('QUEUE_RABBITMQ_VHOST', '/'),
|
||||||
|
],
|
||||||
|
|
||||||
|
'sqs' => [
|
||||||
|
'key' => Env::get('QUEUE_SQS_KEY', ''),
|
||||||
|
'secret' => Env::get('QUEUE_SQS_SECRET', ''),
|
||||||
|
'region' => Env::get('QUEUE_SQS_REGION', 'us-east-1'),
|
||||||
|
'version' => Env::get('QUEUE_SQS_VERSION', 'latest'),
|
||||||
|
'queue_url' => Env::get('QUEUE_SQS_QUEUE_URL', ''),
|
||||||
|
]
|
||||||
|
]
|
||||||
]
|
]
|
||||||
];
|
];
|
||||||
@@ -4,6 +4,31 @@ volumes:
|
|||||||
|
|
||||||
services:
|
services:
|
||||||
|
|
||||||
|
traefik:
|
||||||
|
image: traefik:latest
|
||||||
|
container_name: traefik
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "traefik", "healthcheck", "--ping"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
ports:
|
||||||
|
- "80:80"
|
||||||
|
- "443:443"
|
||||||
|
volumes:
|
||||||
|
- "/var/run/docker.sock:/var/run/docker.sock"
|
||||||
|
restart: always
|
||||||
|
command:
|
||||||
|
- "--providers.docker=true"
|
||||||
|
- "--ping"
|
||||||
|
- "--providers.docker.exposedByDefault=false"
|
||||||
|
- "--entrypoints.web.address=:80"
|
||||||
|
- "--entrypoints.web-secure.address=:443"
|
||||||
|
- "--accesslog=true"
|
||||||
|
- "--entrypoints.web.http.redirections.entryPoint.to=web-secure"
|
||||||
|
- "--entrypoints.web.http.redirections.entryPoint.scheme=https"
|
||||||
|
- "--entrypoints.web.http.redirections.entrypoint.permanent=true"
|
||||||
|
|
||||||
composer-runtime:
|
composer-runtime:
|
||||||
volumes:
|
volumes:
|
||||||
- .:/app
|
- .:/app
|
||||||
@@ -12,30 +37,113 @@ services:
|
|||||||
environment:
|
environment:
|
||||||
PHP_IDE_CONFIG: serverName=localhost
|
PHP_IDE_CONFIG: serverName=localhost
|
||||||
|
|
||||||
|
migration-container:
|
||||||
|
volumes:
|
||||||
|
- ./db/migrations:/app/db/migrations
|
||||||
|
- ./bin:/app/bin
|
||||||
|
image: siteworxpro/migrate:v4.18.3
|
||||||
|
working_dir: /app
|
||||||
|
# entrypoint: "/bin/sh -c 'while true; do sleep 30; done;'"
|
||||||
|
entrypoint: /bin/sh -c '/app/bin/migrate.sh'
|
||||||
|
depends_on:
|
||||||
|
postgres:
|
||||||
|
condition: service_healthy
|
||||||
|
environment:
|
||||||
|
DB_USERNAME: ${DB_USERNAME:-siteworxpro}
|
||||||
|
DB_PASSWORD: ${DB_PASSWORD:-password}
|
||||||
|
DB_DATABASE: ${DB_DATABASE:-siteworxpro}
|
||||||
|
DB_HOST: ${DB_HOST-postgres}
|
||||||
|
DB_PORT: ${DB_PORT-5432}
|
||||||
|
|
||||||
dev-runtime:
|
dev-runtime:
|
||||||
ports:
|
labels:
|
||||||
- "9501:9501"
|
- "traefik.enable=true"
|
||||||
|
- "traefik.http.routers.api.entrypoints=web-secure"
|
||||||
|
- "traefik.http.routers.api.rule=Host(`localhost`) || Host(`127.0.0.1`)"
|
||||||
|
- "traefik.http.routers.api.tls=true"
|
||||||
|
- "traefik.http.routers.api.service=api"
|
||||||
|
- "traefik.http.services.api.loadbalancer.healthcheck.path=/healthz"
|
||||||
|
- "traefik.http.services.api.loadbalancer.healthcheck.interval=5s"
|
||||||
|
- "traefik.http.services.api.loadbalancer.healthcheck.timeout=60s"
|
||||||
volumes:
|
volumes:
|
||||||
- .:/app
|
- .:/app
|
||||||
build:
|
build:
|
||||||
|
args:
|
||||||
|
KAFKA_ENABLED: "1"
|
||||||
context: .
|
context: .
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
entrypoint: "/bin/sh -c 'while true; do sleep 30; done;'"
|
entrypoint: "/bin/sh -c 'while true; do sleep 30; done;'"
|
||||||
|
depends_on:
|
||||||
|
migration-container:
|
||||||
|
condition: service_completed_successfully
|
||||||
|
traefik:
|
||||||
|
condition: service_healthy
|
||||||
|
redis:
|
||||||
|
condition: service_healthy
|
||||||
|
postgres:
|
||||||
|
condition: service_healthy
|
||||||
environment:
|
environment:
|
||||||
|
QUEUE_BROKER: redis
|
||||||
PHP_IDE_CONFIG: serverName=localhost
|
PHP_IDE_CONFIG: serverName=localhost
|
||||||
WORKERS: 1
|
WORKERS: 1
|
||||||
DEBUG: 1
|
DEBUG: 1
|
||||||
REDIS_HOST: redis
|
REDIS_HOST: redis
|
||||||
|
DB_HOST: postgres
|
||||||
|
JWT_SIGNING_KEY: a-string-secret-at-least-256-bits-long
|
||||||
|
|
||||||
|
## Kafka and Zookeeper for local development
|
||||||
|
kafka-ui:
|
||||||
|
image: kafbat/kafka-ui:latest # Or kafbat/kafka-ui:latest for newer Kafka
|
||||||
|
container_name: kafka-ui
|
||||||
|
ports:
|
||||||
|
- "8080:8080" # Expose the UI port
|
||||||
|
environment:
|
||||||
|
KAFKA_CLUSTERS_0_NAME: local-kafka-cluster
|
||||||
|
KAFKA_CLUSTERS_0_BOOTSTRAPSERVERS: kafka:9092
|
||||||
|
depends_on:
|
||||||
|
kafka:
|
||||||
|
condition: service_started
|
||||||
|
zookeeper:
|
||||||
|
condition: service_started
|
||||||
|
zookeeper:
|
||||||
|
image: ubuntu/zookeeper:latest
|
||||||
|
environment:
|
||||||
|
ALLOW_ANONYMOUS_LOGIN: "yes"
|
||||||
|
ports:
|
||||||
|
- "2181:2181"
|
||||||
|
kafka:
|
||||||
|
image: ubuntu/kafka:latest
|
||||||
|
environment:
|
||||||
|
KAFKA_BROKER_ID: 1
|
||||||
|
KAFKA_LISTENERS: PLAINTEXT://kafka:9092
|
||||||
|
KAFKA_ADVERTISED_LISTENERS: PLAINTEXT://kafka:9092
|
||||||
|
KAFKA_ZOOKEEPER_CONNECT: zookeeper:2181
|
||||||
|
ALLOW_PLAINTEXT_LISTENER: "yes"
|
||||||
|
ports:
|
||||||
|
- "9092:9092"
|
||||||
|
depends_on:
|
||||||
|
zookeeper:
|
||||||
|
condition: service_started
|
||||||
|
|
||||||
redis:
|
redis:
|
||||||
image: redis:latest
|
image: redis:latest
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "redis-cli", "ping"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
ports:
|
ports:
|
||||||
- "6379:6379"
|
- "6379:6379"
|
||||||
volumes:
|
volumes:
|
||||||
- redisdata:/data
|
- redisdata:/data
|
||||||
|
|
||||||
postgres:
|
postgres:
|
||||||
image: postgres:latest
|
image: postgres:18
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "pg_isready -U ${DB_USERNAME:-siteworxpro}"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: ${DB_USERNAME:-siteworxpro}
|
POSTGRES_USER: ${DB_USERNAME:-siteworxpro}
|
||||||
POSTGRES_PASSWORD: ${DB_PASSWORD:-password}
|
POSTGRES_PASSWORD: ${DB_PASSWORD:-password}
|
||||||
@@ -43,4 +151,4 @@ services:
|
|||||||
ports:
|
ports:
|
||||||
- "5432:5432"
|
- "5432:5432"
|
||||||
volumes:
|
volumes:
|
||||||
- pgdata:/var/lib/postgresql/data
|
- pgdata:/var/lib/postgresql
|
||||||
@@ -1,5 +1,8 @@
|
|||||||
FROM siteworxpro/migrate:v4.18.3
|
FROM siteworxpro/migrate:v4.18.3
|
||||||
|
|
||||||
ADD db/migrations /app/db/migrations
|
ADD db/migrations /app/db/migrations
|
||||||
|
ADD bin/migrate.sh /app/bin/migrate.sh
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
ENTRYPOINT ["/app/bin/migrate.sh"]
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
use Siteworxpro\App\Server;
|
use Siteworxpro\App\Api;
|
||||||
|
|
||||||
require __DIR__ . '/vendor/autoload.php';
|
require __DIR__ . '/vendor/autoload.php';
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Instantiate the ExternalServer class
|
// Instantiate the ExternalServer class
|
||||||
$server = new Server();
|
$server = new Api();
|
||||||
|
|
||||||
// Start the server
|
// Start the server
|
||||||
$server->startServer();
|
$server->startServer();
|
||||||
|
|||||||
36
src/Annotations/Async/HandlesMessage.php
Normal file
36
src/Annotations/Async/HandlesMessage.php
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Annotations\Async;
|
||||||
|
|
||||||
|
use Attribute;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attribute to mark a class as a handler for a specific message class in an async workflow.
|
||||||
|
*
|
||||||
|
* Repeatable: attach multiple times to handle multiple message classes.
|
||||||
|
*/
|
||||||
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::IS_REPEATABLE)]
|
||||||
|
readonly class HandlesMessage
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Create a new HandlesMessage attribute.
|
||||||
|
*
|
||||||
|
* @param class-string $messageClass Fully-qualified class name of the message handled.
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
public string $messageClass,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the fully-qualified message class this handler processes.
|
||||||
|
*
|
||||||
|
* @return class-string
|
||||||
|
*/
|
||||||
|
public function getMessageClass(): string
|
||||||
|
{
|
||||||
|
return $this->messageClass;
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/Annotations/Events/ListensFor.php
Normal file
28
src/Annotations/Events/ListensFor.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Annotations\Events;
|
||||||
|
|
||||||
|
use Attribute;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attribute to mark a class as an event listener for a specific event class.
|
||||||
|
*
|
||||||
|
* Apply this attribute to classes that subscribe to domain or application events.
|
||||||
|
* Repeatable: can be attached multiple times to the same class to listen for multiple events.
|
||||||
|
*
|
||||||
|
* Targets: class only.
|
||||||
|
*/
|
||||||
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::IS_REPEATABLE)]
|
||||||
|
readonly class ListensFor
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Initialize the ListensFor attribute.
|
||||||
|
*
|
||||||
|
* @param class-string $eventClass Fully-qualified class name of the event to listen for.
|
||||||
|
*/
|
||||||
|
public function __construct(public string $eventClass)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
76
src/Annotations/Guards/Jwt.php
Normal file
76
src/Annotations/Guards/Jwt.php
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Annotations\Guards;
|
||||||
|
|
||||||
|
use Attribute;
|
||||||
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attribute to guard classes or methods with JWT claim requirements.
|
||||||
|
*
|
||||||
|
* Apply this attribute to a class or method to declare the expected JWT issuer and/or audience.
|
||||||
|
* If either the issuer or audience is an empty string, the value will be resolved from configuration:
|
||||||
|
* - `jwt.issuer`
|
||||||
|
* - `jwt.audience`
|
||||||
|
*
|
||||||
|
* Targets: class or method.
|
||||||
|
*/
|
||||||
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_METHOD)]
|
||||||
|
readonly class Jwt
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Initialize the Jwt attribute with optional overrides for expected JWT claims.
|
||||||
|
*
|
||||||
|
* @param string $issuer Optional expected JWT issuer (`iss`). Empty string uses `Config::get('jwt.issuer')`.
|
||||||
|
* @param string $audience Optional expected JWT audience (`aud`). Empty string uses `Config::get('jwt.audience')`.
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
private string $issuer = '',
|
||||||
|
private string $audience = '',
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the required audience from configuration, ignoring any local override.
|
||||||
|
*
|
||||||
|
* @return string The globally configured audience or an empty string if not set.
|
||||||
|
*/
|
||||||
|
public function getRequiredAudience(): string
|
||||||
|
{
|
||||||
|
return Config::get('jwt.audience') ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the expected audience for validation.
|
||||||
|
*
|
||||||
|
* Returns the constructor-provided audience when non-empty; otherwise falls back to `jwt.audience` config.
|
||||||
|
*
|
||||||
|
* @return string The audience value to enforce.
|
||||||
|
*/
|
||||||
|
public function getAudience(): string
|
||||||
|
{
|
||||||
|
if ($this->audience === '') {
|
||||||
|
return Config::get('jwt.audience') ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->audience;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the expected issuer for validation.
|
||||||
|
*
|
||||||
|
* Returns the constructor-provided issuer when non-empty; otherwise falls back to `jwt.issuer` config.
|
||||||
|
*
|
||||||
|
* @return string The issuer value to enforce.
|
||||||
|
*/
|
||||||
|
public function getIssuer(): string
|
||||||
|
{
|
||||||
|
if ($this->issuer === '') {
|
||||||
|
return Config::get('jwt.issuer') ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->issuer;
|
||||||
|
}
|
||||||
|
}
|
||||||
21
src/Annotations/Guards/Scope.php
Normal file
21
src/Annotations/Guards/Scope.php
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Annotations\Guards;
|
||||||
|
|
||||||
|
use Attribute;
|
||||||
|
|
||||||
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_METHOD)]
|
||||||
|
readonly class Scope
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private array $scopes = []
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getScopes(): array
|
||||||
|
{
|
||||||
|
return $this->scopes;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,21 +4,19 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Siteworxpro\App;
|
namespace Siteworxpro\App;
|
||||||
|
|
||||||
use Illuminate\Container\Container;
|
|
||||||
use Illuminate\Support\ServiceProvider;
|
|
||||||
use League\Route\Http\Exception\MethodNotAllowedException;
|
use League\Route\Http\Exception\MethodNotAllowedException;
|
||||||
use League\Route\Http\Exception\NotFoundException;
|
use League\Route\Http\Exception\NotFoundException;
|
||||||
use League\Route\Router;
|
use League\Route\Router;
|
||||||
use Nyholm\Psr7\Factory\Psr17Factory;
|
use Nyholm\Psr7\Factory\Psr17Factory;
|
||||||
use Siteworx\Config\Config as SWConfig;
|
use Siteworxpro\App\Controllers\HealthcheckController;
|
||||||
use Siteworxpro\App\Controllers\IndexController;
|
use Siteworxpro\App\Controllers\IndexController;
|
||||||
use Siteworxpro\App\Http\JsonResponseFactory;
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
use Siteworxpro\App\Http\Middleware\CorsMiddleware;
|
use Siteworxpro\App\Http\Middleware\CorsMiddleware;
|
||||||
use Siteworxpro\App\Services\Facade;
|
use Siteworxpro\App\Http\Middleware\JwtMiddleware;
|
||||||
|
use Siteworxpro\App\Http\Middleware\ScopeMiddleware;
|
||||||
use Siteworxpro\App\Services\Facades\Config;
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
use Siteworxpro\App\Services\Facades\Logger;
|
use Siteworxpro\App\Services\Facades\Logger;
|
||||||
use Siteworxpro\App\Services\ServiceProviders\LoggerServiceProvider;
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
use Siteworxpro\App\Services\ServiceProviders\RedisServiceProvider;
|
|
||||||
use Spiral\RoadRunner\Http\PSR7Worker;
|
use Spiral\RoadRunner\Http\PSR7Worker;
|
||||||
use Spiral\RoadRunner\Worker;
|
use Spiral\RoadRunner\Worker;
|
||||||
|
|
||||||
@@ -31,7 +29,7 @@ use Spiral\RoadRunner\Worker;
|
|||||||
*
|
*
|
||||||
* @package Siteworxpro\App
|
* @package Siteworxpro\App
|
||||||
*/
|
*/
|
||||||
class Server
|
class Api
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @var Router The router instance for handling routes.
|
* @var Router The router instance for handling routes.
|
||||||
@@ -43,97 +41,13 @@ class Server
|
|||||||
*/
|
*/
|
||||||
protected PSR7Worker $worker;
|
protected PSR7Worker $worker;
|
||||||
|
|
||||||
public static array $serviceProviders = [
|
|
||||||
LoggerServiceProvider::class,
|
|
||||||
RedisServiceProvider::class
|
|
||||||
];
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Server constructor.
|
|
||||||
*
|
|
||||||
* Initializes the server by booting the PSR-7 worker and router.
|
|
||||||
* @throws \ReflectionException
|
* @throws \ReflectionException
|
||||||
*/
|
*/
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
$this->boot();
|
Kernel::boot();
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Bootstraps the server by initializing the PSR-7 worker and router.
|
|
||||||
*
|
|
||||||
* This method sets up the PSR-7 worker and router instances, and registers
|
|
||||||
* the routes for the server. It should be called in the constructor of
|
|
||||||
* subclasses to ensure proper initialization.
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
* @throws \ReflectionException
|
|
||||||
*/
|
|
||||||
private function boot(): void
|
|
||||||
{
|
|
||||||
$container = new Container();
|
|
||||||
Facade::setFacadeContainer($container);
|
|
||||||
|
|
||||||
// Bind the container to the Config facade first so that it can be used by service providers
|
|
||||||
$container->bind(SWConfig::class, function () {
|
|
||||||
return SWConfig::load(__DIR__ . '/../config.php');
|
|
||||||
});
|
|
||||||
|
|
||||||
foreach (self::$serviceProviders as $serviceProvider) {
|
|
||||||
if (class_exists($serviceProvider)) {
|
|
||||||
$provider = new $serviceProvider($container);
|
|
||||||
if ($provider instanceof ServiceProvider) {
|
|
||||||
$provider->register();
|
|
||||||
} else {
|
|
||||||
throw new \RuntimeException(sprintf(
|
|
||||||
'Service provider %s is not an instance of ServiceProvider.',
|
|
||||||
$serviceProvider
|
|
||||||
));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new \RuntimeException(sprintf('Service provider %s not found.', $serviceProvider));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->worker = new PSR7Worker(
|
|
||||||
Worker::create(),
|
|
||||||
new Psr17Factory(),
|
|
||||||
new Psr17Factory(),
|
|
||||||
new Psr17Factory()
|
|
||||||
);
|
|
||||||
|
|
||||||
$this->router = new Router();
|
|
||||||
|
|
||||||
$this->registerRoutes();
|
$this->registerRoutes();
|
||||||
$this->bootModelCapsule();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Bootstraps the model capsule for database connections.
|
|
||||||
*
|
|
||||||
* This method sets up the database connection using the Eloquent ORM.
|
|
||||||
* It retrieves the database configuration from the Config facade and
|
|
||||||
* initializes the Eloquent capsule manager.
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function bootModelCapsule(): void
|
|
||||||
{
|
|
||||||
$capsule = new \Illuminate\Database\Capsule\Manager();
|
|
||||||
$capsule->addConnection([
|
|
||||||
'driver' => Config::get('db.driver'),
|
|
||||||
'host' => Config::get('db.host'),
|
|
||||||
'database' => Config::get('db.database'),
|
|
||||||
'username' => Config::get('db.username'),
|
|
||||||
'password' => Config::get('db.password'),
|
|
||||||
'charset' => 'utf8',
|
|
||||||
'collation' => 'utf8_unicode_ci',
|
|
||||||
'prefix' => '',
|
|
||||||
]);
|
|
||||||
|
|
||||||
$capsule->setAsGlobal();
|
|
||||||
$capsule->bootEloquent();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -144,10 +58,22 @@ class Server
|
|||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
protected function registerRoutes(): void
|
public function registerRoutes(): void
|
||||||
{
|
{
|
||||||
|
$this->worker = new PSR7Worker(
|
||||||
|
Worker::create(),
|
||||||
|
new Psr17Factory(),
|
||||||
|
new Psr17Factory(),
|
||||||
|
new Psr17Factory()
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->router = new Router();
|
||||||
$this->router->get('/', IndexController::class . '::get');
|
$this->router->get('/', IndexController::class . '::get');
|
||||||
|
$this->router->get('/healthz', HealthcheckController::class . '::get');
|
||||||
|
|
||||||
$this->router->middleware(new CorsMiddleware());
|
$this->router->middleware(new CorsMiddleware());
|
||||||
|
$this->router->middleware(new JwtMiddleware());
|
||||||
|
$this->router->middleware(new ScopeMiddleware());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -181,7 +107,7 @@ class Server
|
|||||||
$this->worker->respond(
|
$this->worker->respond(
|
||||||
JsonResponseFactory::createJsonResponse(
|
JsonResponseFactory::createJsonResponse(
|
||||||
['status_code' => 404, 'reason_phrase' => 'Not Found'],
|
['status_code' => 404, 'reason_phrase' => 'Not Found'],
|
||||||
404
|
CodesEnum::NOT_FOUND
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
} catch (\Throwable $e) {
|
} catch (\Throwable $e) {
|
||||||
@@ -198,7 +124,9 @@ class Server
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->worker->respond(JsonResponseFactory::createJsonResponse($json, 500));
|
$this->worker->respond(
|
||||||
|
JsonResponseFactory::createJsonResponse($json, CodesEnum::INTERNAL_SERVER_ERROR)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
19
src/Async/Brokers/Broker.php
Normal file
19
src/Async/Brokers/Broker.php
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
abstract class Broker implements BrokerInterface
|
||||||
|
{
|
||||||
|
public const array BROKER_TYPES = [
|
||||||
|
'redis' => Redis::class,
|
||||||
|
'rabbitmq' => RabbitMQ::class,
|
||||||
|
'kafka' => Kafka::class,
|
||||||
|
'sqs' => Sqs::class,
|
||||||
|
];
|
||||||
|
|
||||||
|
public function __construct(protected $config = [])
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
21
src/Async/Brokers/BrokerInterface.php
Normal file
21
src/Async/Brokers/BrokerInterface.php
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
|
||||||
|
interface BrokerInterface
|
||||||
|
{
|
||||||
|
public function publish(Queue $queue, Message $message, ?int $delay = null): void;
|
||||||
|
|
||||||
|
public function consume(Queue $queue): Message | null;
|
||||||
|
|
||||||
|
public function acknowledge(Queue $queue, Message $message): void;
|
||||||
|
|
||||||
|
public function reject(Queue $queue, Message $message, bool $requeue = false): void;
|
||||||
|
|
||||||
|
public function purge(Queue $queue): void;
|
||||||
|
}
|
||||||
94
src/Async/Brokers/Kafka.php
Normal file
94
src/Async/Brokers/Kafka.php
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
use RdKafka\Conf;
|
||||||
|
use RdKafka\Exception;
|
||||||
|
use RdKafka\KafkaConsumer;
|
||||||
|
use RdKafka\Producer;
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
|
||||||
|
class Kafka extends Broker
|
||||||
|
{
|
||||||
|
private Producer $producer;
|
||||||
|
|
||||||
|
private KafkaConsumer $consumer;
|
||||||
|
|
||||||
|
public function __construct($config = [])
|
||||||
|
{
|
||||||
|
parent::__construct($config);
|
||||||
|
|
||||||
|
|
||||||
|
$conf = new Conf();
|
||||||
|
$conf->set('bootstrap.servers', $config['brokers'] ?? 'localhost:9092');
|
||||||
|
|
||||||
|
$this->producer = new Producer($conf);
|
||||||
|
$this->producer->addBrokers($config['brokers'] ?? 'localhost:9092');
|
||||||
|
|
||||||
|
$conf->set('group.id', $config['consumerGroup'] ?? 'default');
|
||||||
|
$conf->set('auto.offset.reset', 'earliest');
|
||||||
|
$this->consumer = new KafkaConsumer($conf);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __destruct()
|
||||||
|
{
|
||||||
|
$this->producer->flush(1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \Exception
|
||||||
|
*/
|
||||||
|
public function publish(Queue $queue, Message $message, ?int $delay = null): void
|
||||||
|
{
|
||||||
|
$topic = $this->producer->newTopic($queue->queueName());
|
||||||
|
$topic->produce(RD_KAFKA_PARTITION_UA, 0, $message->serialize(), $message->getId());
|
||||||
|
$this->producer->flush(1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function consume(Queue $queue): Message|null
|
||||||
|
{
|
||||||
|
$this->consumer->subscribe([$queue->queueName()]);
|
||||||
|
$kafkaMessage = $this->consumer->consume(1000);
|
||||||
|
|
||||||
|
if ($kafkaMessage->err === RD_KAFKA_RESP_ERR__TIMED_OUT) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($kafkaMessage->err === RD_KAFKA_RESP_ERR_UNKNOWN_TOPIC_OR_PART) {
|
||||||
|
throw new \RuntimeException(
|
||||||
|
"Topic '{$queue->queueName()}' or partition does not exist. Kafka does not auto-create topics" .
|
||||||
|
" unless configured to do so."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var string | null $messageData */
|
||||||
|
$messageData = $kafkaMessage->payload;
|
||||||
|
if ($messageData !== null) {
|
||||||
|
/** @var Message $message */
|
||||||
|
$message = unserialize($messageData, ['allowed_classes' => true]);
|
||||||
|
$message->setId((string)$kafkaMessage->offset);
|
||||||
|
|
||||||
|
return $message;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function acknowledge(Queue $queue, Message $message): void
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reject(Queue $queue, Message $message, bool $requeue = false): void
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public function purge(Queue $queue): void
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
36
src/Async/Brokers/RabbitMQ.php
Normal file
36
src/Async/Brokers/RabbitMQ.php
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
|
||||||
|
class RabbitMQ extends Broker
|
||||||
|
{
|
||||||
|
public function publish(Queue $queue, Message $message, ?int $delay = null): void
|
||||||
|
{
|
||||||
|
// TODO: Implement publish() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function consume(Queue $queue): Message | null
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function acknowledge(Queue $queue, Message $message): void
|
||||||
|
{
|
||||||
|
// TODO: Implement acknowledge() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reject(Queue $queue, Message $message, bool $requeue = false): void
|
||||||
|
{
|
||||||
|
// TODO: Implement reject() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function purge(Queue $queue): void
|
||||||
|
{
|
||||||
|
// TODO: Implement purge() method.
|
||||||
|
}
|
||||||
|
}
|
||||||
190
src/Async/Brokers/Redis.php
Normal file
190
src/Async/Brokers/Redis.php
Normal file
@@ -0,0 +1,190 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
use Predis\Client;
|
||||||
|
use Predis\Command\RawCommand;
|
||||||
|
use Siteworxpro\App\Async\Messages\SayHelloMessage;
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
use Siteworxpro\App\Helpers\Ulid;
|
||||||
|
|
||||||
|
class Redis extends Broker
|
||||||
|
{
|
||||||
|
private Client $client;
|
||||||
|
|
||||||
|
private string $consumerId;
|
||||||
|
|
||||||
|
private string $consumerGroup;
|
||||||
|
|
||||||
|
private const string CONSUMER_ID_PREFIX = 'consumer-group:';
|
||||||
|
private const string QUEUE_PREFIX = 'queue:';
|
||||||
|
|
||||||
|
private array $queueNames = [];
|
||||||
|
|
||||||
|
public function __construct($config = [])
|
||||||
|
{
|
||||||
|
parent::__construct($config);
|
||||||
|
|
||||||
|
$this->client = \Siteworxpro\App\Services\Facades\Redis::getFacadeRoot();
|
||||||
|
$this->consumerId = php_uname('n') . ':' . getmypid();
|
||||||
|
$this->consumerGroup = $config['consumerGroup'] ?? 'default';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ensureQueue(string $queueName): void
|
||||||
|
{
|
||||||
|
if (in_array($queueName, $this->queueNames, true)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->client->executeCommand(
|
||||||
|
new RawCommand(
|
||||||
|
'XGROUP',
|
||||||
|
[
|
||||||
|
'CREATE',
|
||||||
|
self::QUEUE_PREFIX . $queueName,
|
||||||
|
self::CONSUMER_ID_PREFIX . $this->consumerGroup,
|
||||||
|
'$',
|
||||||
|
'MKSTREAM'
|
||||||
|
]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (\Exception) {
|
||||||
|
// If the group already exists, we catch the exception and ignore it
|
||||||
|
// This is because Redis will throw an error if the group already exists
|
||||||
|
// We can safely ignore this error as it means the group is already set up
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->client->executeCommand(
|
||||||
|
new RawCommand(
|
||||||
|
'XGROUP',
|
||||||
|
[
|
||||||
|
'CREATECONSUMER',
|
||||||
|
self::QUEUE_PREFIX . $queueName,
|
||||||
|
self::CONSUMER_ID_PREFIX . $this->consumerGroup,
|
||||||
|
$this->consumerId
|
||||||
|
]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->queueNames[] = $queueName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __destruct()
|
||||||
|
{
|
||||||
|
foreach ($this->queueNames as $queueName) {
|
||||||
|
try {
|
||||||
|
$this->client->executeCommand(
|
||||||
|
new RawCommand(
|
||||||
|
'XGROUP',
|
||||||
|
[
|
||||||
|
'DELCONSUMER',
|
||||||
|
self::QUEUE_PREFIX . $queueName,
|
||||||
|
self::CONSUMER_ID_PREFIX . $this->consumerGroup,
|
||||||
|
$this->consumerId
|
||||||
|
]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (\Exception) {
|
||||||
|
// Ignore exceptions during cleanup
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \Exception
|
||||||
|
*/
|
||||||
|
public function publish(Queue $queue, Message $message, ?int $delay = null): void
|
||||||
|
{
|
||||||
|
$command = '%s * data %s';
|
||||||
|
$command = sprintf(
|
||||||
|
$command,
|
||||||
|
self::QUEUE_PREFIX .
|
||||||
|
$queue->queueName(),
|
||||||
|
base64_encode($message->serialize())
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var string $result */
|
||||||
|
$result = $this
|
||||||
|
->client
|
||||||
|
->executeCommand(
|
||||||
|
new RawCommand('XADD', explode(' ', $command)),
|
||||||
|
);
|
||||||
|
|
||||||
|
$message->setId($result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function consume(Queue $queue): Message|null
|
||||||
|
{
|
||||||
|
$this->ensureQueue($queue->queueName());
|
||||||
|
|
||||||
|
$command = 'GROUP %s %s COUNT 1 STREAMS %s >';
|
||||||
|
$command = sprintf(
|
||||||
|
$command,
|
||||||
|
self::CONSUMER_ID_PREFIX . $this->consumerGroup,
|
||||||
|
$this->consumerId,
|
||||||
|
self::QUEUE_PREFIX . $queue->queueName(),
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var array | null $response */
|
||||||
|
$response = $this
|
||||||
|
->client
|
||||||
|
->executeCommand(
|
||||||
|
new RawCommand(
|
||||||
|
'XREADGROUP',
|
||||||
|
explode(' ', $command)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response === null || !isset($response[0][1][0][1][1])) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$messageData = base64_decode($response[0][1][0][1][1]);
|
||||||
|
$messageId = $response[0][1][0][0];
|
||||||
|
|
||||||
|
if ($messageData === 'NOOP') {
|
||||||
|
// If the message is a NOOP, we return null to indicate no actual message
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$value = unserialize($messageData, ['allowed_classes' => true]);
|
||||||
|
if (!$value instanceof Message) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$value->setId($messageId);
|
||||||
|
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function acknowledge(Queue $queue, Message $message): void
|
||||||
|
{
|
||||||
|
$response = $this
|
||||||
|
->client
|
||||||
|
->executeCommand(
|
||||||
|
new RawCommand(
|
||||||
|
'XACK',
|
||||||
|
[
|
||||||
|
self::QUEUE_PREFIX . $queue->queueName(),
|
||||||
|
self::CONSUMER_ID_PREFIX . $this->consumerGroup,
|
||||||
|
$message->getId()
|
||||||
|
]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reject(Queue $queue, Message $message, bool $requeue = false): void
|
||||||
|
{
|
||||||
|
// TODO: Implement reject() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function purge(Queue $queue): void
|
||||||
|
{
|
||||||
|
|
||||||
|
// TODO: Implement purge() method.
|
||||||
|
}
|
||||||
|
}
|
||||||
36
src/Async/Brokers/Sqs.php
Normal file
36
src/Async/Brokers/Sqs.php
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Brokers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
|
||||||
|
class Sqs extends Broker
|
||||||
|
{
|
||||||
|
public function publish(Queue $queue, Message $message, ?int $delay = null): void
|
||||||
|
{
|
||||||
|
// TODO: Implement publish() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function consume(Queue $queue): Message | null
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function acknowledge(Queue $queue, Message $message): void
|
||||||
|
{
|
||||||
|
// TODO: Implement acknowledge() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reject(Queue $queue, Message $message, bool $requeue = false): void
|
||||||
|
{
|
||||||
|
// TODO: Implement reject() method.
|
||||||
|
}
|
||||||
|
|
||||||
|
public function purge(Queue $queue): void
|
||||||
|
{
|
||||||
|
// TODO: Implement purge() method.
|
||||||
|
}
|
||||||
|
}
|
||||||
173
src/Async/Consumer.php
Normal file
173
src/Async/Consumer.php
Normal file
@@ -0,0 +1,173 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(ticks=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Annotations\Async\HandlesMessage;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Services\Facades\Broker;
|
||||||
|
use Siteworxpro\App\Services\Facades\Logger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Long-running process that listens to queues, pops messages, and dispatches them to handlers.
|
||||||
|
*/
|
||||||
|
class Consumer
|
||||||
|
{
|
||||||
|
private static bool $shutDown = false;
|
||||||
|
|
||||||
|
/** @var array<string,string> */
|
||||||
|
private const array QUEUES = [
|
||||||
|
'default' => Queues\DefaultQueue::class,
|
||||||
|
];
|
||||||
|
|
||||||
|
/** @var Queue[] */
|
||||||
|
private array $queues = [];
|
||||||
|
|
||||||
|
/** @var array<string, string[]> message FQCN => handler FQCNs */
|
||||||
|
private array $handlers = [];
|
||||||
|
|
||||||
|
private const string HANDLER_NAMESPACE = 'Siteworxpro\\App\\Async\\Handlers\\';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string[] $queues Optional list of queue names (keys from self::QUEUES)
|
||||||
|
*/
|
||||||
|
public function __construct(array $queues = [])
|
||||||
|
{
|
||||||
|
$queueClasses = $queues === []
|
||||||
|
? array_values(self::QUEUES)
|
||||||
|
: array_map(
|
||||||
|
static function (string $name): string {
|
||||||
|
if (!isset(self::QUEUES[$name])) {
|
||||||
|
throw new \InvalidArgumentException("Queue '$name' is not defined.");
|
||||||
|
}
|
||||||
|
return self::QUEUES[$name];
|
||||||
|
},
|
||||||
|
$queues
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach ($queueClasses as $class) {
|
||||||
|
$this->queues[] = new $class();
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->registerHandlers();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Discover handler classes under `Handlers` and register them via HandlesMessage attributes.
|
||||||
|
*/
|
||||||
|
private function registerHandlers(): void
|
||||||
|
{
|
||||||
|
$it = new \RecursiveIteratorIterator(
|
||||||
|
new \RecursiveDirectoryIterator(__DIR__ . '/Handlers/')
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var \SplFileInfo $file */
|
||||||
|
foreach ($it as $file) {
|
||||||
|
if (!$file->isFile() || $file->getExtension() !== 'php') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$relative = str_replace(__DIR__ . '/Handlers/', '', $file->getPathname());
|
||||||
|
$class = self::HANDLER_NAMESPACE . str_replace('/', '\\', substr($relative, 0, -4));
|
||||||
|
|
||||||
|
if (!class_exists($class)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ref = new \ReflectionClass($class);
|
||||||
|
foreach ($ref->getAttributes(HandlesMessage::class) as $attr) {
|
||||||
|
$messageClass = $attr->newInstance()->getMessageClass();
|
||||||
|
$this->handlers[$messageClass][] = $class;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Signal handler used to initiate graceful or immediate shutdown.
|
||||||
|
*/
|
||||||
|
public static function handleSignal(int $signal): void
|
||||||
|
{
|
||||||
|
switch ($signal) {
|
||||||
|
case SIGINT:
|
||||||
|
case SIGTERM:
|
||||||
|
case SIGHUP:
|
||||||
|
self::$shutDown = true;
|
||||||
|
return;
|
||||||
|
case SIGKILL:
|
||||||
|
exit(9);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function shouldShutDown(): bool
|
||||||
|
{
|
||||||
|
return self::$shutDown;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start the consumer main loop.
|
||||||
|
*/
|
||||||
|
public function start(): void
|
||||||
|
{
|
||||||
|
if (!\function_exists('pcntl_signal')) {
|
||||||
|
throw new \RuntimeException('The pcntl extension is required to handle signals.');
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger::info('Starting queue consumer...');
|
||||||
|
Logger::info('Using Broker: ' . Broker::getFacadeRoot()::class);
|
||||||
|
|
||||||
|
foreach ([SIGINT, SIGTERM, SIGHUP] as $sig) {
|
||||||
|
\pcntl_signal($sig, [self::class, 'handleSignal']);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if ($this->shouldShutDown()) {
|
||||||
|
Logger::info('Shutting down queue consumer...');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var Queue $queue */
|
||||||
|
foreach ($this->queues as $queue) {
|
||||||
|
Logger::info('Listening to queue: ' . $queue->queueName());
|
||||||
|
$message = $queue->pop();
|
||||||
|
if (!$message) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger::info('Processing message of type: ' . get_class($message));
|
||||||
|
|
||||||
|
foreach ($this->getHandlersForMessage($message) as $handler) {
|
||||||
|
$handler($message);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Continue polling from the top of the loop after processing a message.
|
||||||
|
continue 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid busy-looping when no messages are available.
|
||||||
|
sleep(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return callable[] Handler instances invokable with the message
|
||||||
|
*/
|
||||||
|
private function getHandlersForMessage(Message $message): array
|
||||||
|
{
|
||||||
|
$messageClass = get_class($message);
|
||||||
|
|
||||||
|
if (!isset($this->handlers[$messageClass])) {
|
||||||
|
throw new \RuntimeException("No handler found for message class: $messageClass");
|
||||||
|
}
|
||||||
|
|
||||||
|
$callables = [];
|
||||||
|
foreach ($this->handlers[$messageClass] as $handlerClass) {
|
||||||
|
if (class_exists($handlerClass)) {
|
||||||
|
$callables[] = new $handlerClass();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $callables;
|
||||||
|
}
|
||||||
|
}
|
||||||
12
src/Async/Handlers/HandlerInterface.php
Normal file
12
src/Async/Handlers/HandlerInterface.php
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Handlers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
|
||||||
|
interface HandlerInterface
|
||||||
|
{
|
||||||
|
public function __invoke(Message $message): void;
|
||||||
|
}
|
||||||
21
src/Async/Handlers/SayHelloHandler.php
Normal file
21
src/Async/Handlers/SayHelloHandler.php
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Handlers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Annotations\Async\HandlesMessage;
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
use Siteworxpro\App\Async\Messages\SayHelloMessage;
|
||||||
|
use Siteworxpro\App\Services\Facades\Logger;
|
||||||
|
|
||||||
|
#[HandlesMessage(SayHelloMessage::class)]
|
||||||
|
class SayHelloHandler implements HandlerInterface
|
||||||
|
{
|
||||||
|
public function __invoke(Message | SayHelloMessage $message): void
|
||||||
|
{
|
||||||
|
$name = $message->getPayload()['name'] ?? 'Guest';
|
||||||
|
|
||||||
|
Logger::info(sprintf("Hello, %s!", $name));
|
||||||
|
}
|
||||||
|
}
|
||||||
102
src/Async/Messages/Message.php
Normal file
102
src/Async/Messages/Message.php
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Messages;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Queues\DefaultQueue;
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Helpers\Ulid;
|
||||||
|
|
||||||
|
abstract class Message implements \Serializable
|
||||||
|
{
|
||||||
|
protected string $id = '';
|
||||||
|
|
||||||
|
protected string $uniqueId;
|
||||||
|
|
||||||
|
protected array $payload;
|
||||||
|
|
||||||
|
protected int $timestamp;
|
||||||
|
|
||||||
|
protected string $queue = '';
|
||||||
|
|
||||||
|
protected const string DEFAULT_QUEUE = DefaultQueue::class;
|
||||||
|
|
||||||
|
abstract public static function dispatch(...$args): void;
|
||||||
|
|
||||||
|
abstract public static function dispatchLater(int $delay, ...$args): void;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
$this->uniqueId = Ulid::generate();
|
||||||
|
$this->timestamp = time();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getQueue(): Queue
|
||||||
|
{
|
||||||
|
if ($this->queue === '') {
|
||||||
|
$this->queue = static::DEFAULT_QUEUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new $this->queue();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $id
|
||||||
|
*/
|
||||||
|
public function setId(string $id): void
|
||||||
|
{
|
||||||
|
$this->id = $id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getPayload(): array
|
||||||
|
{
|
||||||
|
return $this->payload;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTimestamp(): int
|
||||||
|
{
|
||||||
|
return $this->timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __serialize(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'id' => $this->id,
|
||||||
|
'payload' => $this->payload,
|
||||||
|
'timestamp' => $this->timestamp,
|
||||||
|
'queue' => $this->queue,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __unserialize(array $data): void
|
||||||
|
{
|
||||||
|
$this->id = $data['id'];
|
||||||
|
$this->payload = $data['payload'];
|
||||||
|
$this->timestamp = $data['timestamp'];
|
||||||
|
$this->queue = $data['queue'];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function serialize(): string
|
||||||
|
{
|
||||||
|
return serialize($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function unserialize(string $data): Message
|
||||||
|
{
|
||||||
|
$unserializedData = unserialize($data, ['allowed_classes' => [Message::class]]);
|
||||||
|
|
||||||
|
$this->id = $unserializedData['id'];
|
||||||
|
$this->uniqueId = $unserializedData['uniqueId'];
|
||||||
|
$this->payload = $unserializedData['payload'];
|
||||||
|
$this->timestamp = $unserializedData['timestamp'];
|
||||||
|
$this->queue = $unserializedData['queue'];
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
}
|
||||||
41
src/Async/Messages/SayHelloMessage.php
Normal file
41
src/Async/Messages/SayHelloMessage.php
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Messages;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\Facades\Broker;
|
||||||
|
|
||||||
|
class SayHelloMessage extends Message
|
||||||
|
{
|
||||||
|
public static function dispatch(...$args): void
|
||||||
|
{
|
||||||
|
$name = $args[0] ?? 'World';
|
||||||
|
$message = new self($name);
|
||||||
|
Broker::publish(
|
||||||
|
$message->getQueue(),
|
||||||
|
$message
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function dispatchLater(int $delay, ...$args): void
|
||||||
|
{
|
||||||
|
$name = $args[0] ?? 'World';
|
||||||
|
$message = new self($name);
|
||||||
|
Broker::publishLater(
|
||||||
|
$message->getQueue(),
|
||||||
|
$message,
|
||||||
|
$delay
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function __construct(
|
||||||
|
private readonly string $name
|
||||||
|
) {
|
||||||
|
parent::__construct();
|
||||||
|
|
||||||
|
$this->payload = [
|
||||||
|
'name' => $this->name,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
13
src/Async/Queues/DefaultQueue.php
Normal file
13
src/Async/Queues/DefaultQueue.php
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Queues;
|
||||||
|
|
||||||
|
readonly class DefaultQueue extends Queue
|
||||||
|
{
|
||||||
|
public function queueName(): string
|
||||||
|
{
|
||||||
|
return 'default';
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/Async/Queues/Queue.php
Normal file
28
src/Async/Queues/Queue.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Async\Queues;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
use Siteworxpro\App\Services\Facades\Broker;
|
||||||
|
|
||||||
|
readonly abstract class Queue
|
||||||
|
{
|
||||||
|
abstract public function queueName(): string;
|
||||||
|
|
||||||
|
public function push(Message $message): void
|
||||||
|
{
|
||||||
|
Broker::publish($this, $message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function later(int $delay, Message $message): void
|
||||||
|
{
|
||||||
|
Broker::publish($this, $message, $delay);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function pop(): Message | null
|
||||||
|
{
|
||||||
|
return Broker::consume($this);
|
||||||
|
}
|
||||||
|
}
|
||||||
45
src/Cli/App.php
Normal file
45
src/Cli/App.php
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Cli;
|
||||||
|
|
||||||
|
use Ahc\Cli\Application;
|
||||||
|
use Siteworxpro\App\Cli\Commands\DemoCommand;
|
||||||
|
use Siteworxpro\App\Cli\Commands\Queue\Start;
|
||||||
|
use Siteworxpro\App\Cli\Commands\Queue\TestJob;
|
||||||
|
use Siteworxpro\App\Kernel;
|
||||||
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
|
||||||
|
class App
|
||||||
|
{
|
||||||
|
private Application $app;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \ReflectionException
|
||||||
|
*/
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
Kernel::boot();
|
||||||
|
$this->app = new Application('Php-Template', Config::get('app.version') ?? 'dev-master');
|
||||||
|
|
||||||
|
$this->app->add(new DemoCommand());
|
||||||
|
$this->app->add(new Start());
|
||||||
|
$this->app->add(new TestJob());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function run(): int
|
||||||
|
{
|
||||||
|
$this->app->logo(
|
||||||
|
<<<EOF
|
||||||
|
▄▄▄▄▄ ▄ ▄ ▄▄▄▄▄ ▄▄▄▄▄▄▄ ▀▀█ ▄
|
||||||
|
█ ▀█ █ █ █ ▀█ █ ▄▄▄ ▄▄▄▄▄ ▄▄▄▄ █ ▄▄▄ ▄▄█▄▄ ▄▄▄
|
||||||
|
█▄▄▄█▀ █▄▄▄▄█ █▄▄▄█▀ █ █▀ █ █ █ █ █▀ ▀█ █ ▀ █ █ █▀ █
|
||||||
|
█ █ █ █ ▀▀▀ █ █▀▀▀▀ █ █ █ █ █ █ ▄▀▀▀█ █ █▀▀▀▀
|
||||||
|
█ █ █ █ █ ▀█▄▄▀ █ █ █ ██▄█▀ ▀▄▄ ▀▄▄▀█ ▀▄▄ ▀█▄▄▀
|
||||||
|
█
|
||||||
|
EOF
|
||||||
|
);
|
||||||
|
return $this->app->handle($_SERVER['argv']);
|
||||||
|
}
|
||||||
|
}
|
||||||
15
src/Cli/Commands/CommandInterface.php
Normal file
15
src/Cli/Commands/CommandInterface.php
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Cli\Commands;
|
||||||
|
|
||||||
|
interface CommandInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Execute the command.
|
||||||
|
*
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public function execute(): int;
|
||||||
|
}
|
||||||
47
src/Cli/Commands/DemoCommand.php
Normal file
47
src/Cli/Commands/DemoCommand.php
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Cli\Commands;
|
||||||
|
|
||||||
|
use Ahc\Cli\Input\Command;
|
||||||
|
|
||||||
|
class DemoCommand extends Command implements CommandInterface
|
||||||
|
{
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct('api:demo', 'A demo command to showcase the CLI functionality.');
|
||||||
|
|
||||||
|
$this->argument('[name]', 'Your name')
|
||||||
|
->option('-g, --greet', 'Include a greeting message');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function execute(): int
|
||||||
|
{
|
||||||
|
$pb = $this->progress(100);
|
||||||
|
|
||||||
|
for ($i = 0; $i < 100; $i += 10) {
|
||||||
|
usleep(100000); // Simulate work
|
||||||
|
$pb->advance(10);
|
||||||
|
}
|
||||||
|
|
||||||
|
$pb->finish();
|
||||||
|
|
||||||
|
$this->writer()->boldBlue("Demo Command Executed!\n");
|
||||||
|
|
||||||
|
if ($this->values()['name']) {
|
||||||
|
$name = $this->values()['name'];
|
||||||
|
$greet = $this->values()['greet'] ?? false;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($greet) {
|
||||||
|
$this->writer()->green("Hello, $name! Welcome to the CLI demo.\n");
|
||||||
|
} else {
|
||||||
|
$this->writer()->yellow("Name provided: {$name}\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/Cli/Commands/Queue/Start.php
Normal file
32
src/Cli/Commands/Queue/Start.php
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Cli\Commands\Queue;
|
||||||
|
|
||||||
|
use Ahc\Cli\Input\Command;
|
||||||
|
use Siteworxpro\App\Async\Consumer;
|
||||||
|
use Siteworxpro\App\Async\Messages\SayHelloMessage;
|
||||||
|
use Siteworxpro\App\Cli\Commands\CommandInterface;
|
||||||
|
|
||||||
|
class Start extends Command implements CommandInterface
|
||||||
|
{
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct('queue:start', 'Start the queue consumer to process messages.');
|
||||||
|
$this->argument('[queues]', 'The name of the queue to consume from. ex. "first_queue,second_queue"');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function execute(): int
|
||||||
|
{
|
||||||
|
$queues = [];
|
||||||
|
if ($this->values()['queues'] !== null) {
|
||||||
|
$queues = explode(',', $this->values()['queues']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$consumer = new Consumer($queues);
|
||||||
|
$consumer->start();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
34
src/Cli/Commands/Queue/TestJob.php
Normal file
34
src/Cli/Commands/Queue/TestJob.php
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Cli\Commands\Queue;
|
||||||
|
|
||||||
|
use Ahc\Cli\Input\Command;
|
||||||
|
use Siteworxpro\App\Async\Messages\SayHelloMessage;
|
||||||
|
use Siteworxpro\App\Cli\Commands\CommandInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class TestJob
|
||||||
|
*
|
||||||
|
* A CLI command to schedule a demo job that dispatches a SayHelloMessage.
|
||||||
|
*/
|
||||||
|
class TestJob extends Command implements CommandInterface
|
||||||
|
{
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct('queue:demo', 'Schedule a demo job.');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the command to dispatch a SayHelloMessage.
|
||||||
|
*
|
||||||
|
* @return int Exit code
|
||||||
|
*/
|
||||||
|
public function execute(): int
|
||||||
|
{
|
||||||
|
SayHelloMessage::dispatch('World from TestJob Command!');
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,6 +8,13 @@ use League\Route\Http\Exception\NotFoundException;
|
|||||||
use Nyholm\Psr7\ServerRequest;
|
use Nyholm\Psr7\ServerRequest;
|
||||||
use Psr\Http\Message\ResponseInterface;
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Controller
|
||||||
|
*
|
||||||
|
* An abstract base controller providing default implementations for HTTP methods.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Controllers
|
||||||
|
*/
|
||||||
abstract class Controller implements ControllerInterface
|
abstract class Controller implements ControllerInterface
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@@ -28,6 +35,14 @@ abstract class Controller implements ControllerInterface
|
|||||||
throw new NotFoundException("not found");
|
throw new NotFoundException("not found");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws NotFoundException
|
||||||
|
*/
|
||||||
|
public function put(ServerRequest $request): ResponseInterface
|
||||||
|
{
|
||||||
|
throw new NotFoundException("not found");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws NotFoundException
|
* @throws NotFoundException
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -7,6 +7,11 @@ namespace Siteworxpro\App\Controllers;
|
|||||||
use Nyholm\Psr7\ServerRequest;
|
use Nyholm\Psr7\ServerRequest;
|
||||||
use Psr\Http\Message\ResponseInterface;
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface ControllerInterface
|
||||||
|
*
|
||||||
|
* Defines the contract for handling HTTP requests in a controller.
|
||||||
|
*/
|
||||||
interface ControllerInterface
|
interface ControllerInterface
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@@ -25,6 +30,14 @@ interface ControllerInterface
|
|||||||
*/
|
*/
|
||||||
public function post(ServerRequest $request): ResponseInterface;
|
public function post(ServerRequest $request): ResponseInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle the request and return a response.
|
||||||
|
*
|
||||||
|
* @param ServerRequest $request The request data.
|
||||||
|
* @return ResponseInterface The response data.
|
||||||
|
*/
|
||||||
|
public function put(ServerRequest $request): ResponseInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle the request and return a response.
|
* Handle the request and return a response.
|
||||||
*
|
*
|
||||||
|
|||||||
53
src/Controllers/HealthcheckController.php
Normal file
53
src/Controllers/HealthcheckController.php
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Controllers;
|
||||||
|
|
||||||
|
use Illuminate\Database\PostgresConnection;
|
||||||
|
use Nyholm\Psr7\ServerRequest;
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
|
use Siteworxpro\App\Models\Model;
|
||||||
|
use Siteworxpro\App\Services\Facades\Redis;
|
||||||
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class HealthcheckController
|
||||||
|
*
|
||||||
|
* Handles health check requests to verify database and cache connectivity.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Controllers
|
||||||
|
*/
|
||||||
|
class HealthcheckController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @throws \JsonException
|
||||||
|
*/
|
||||||
|
public function get(ServerRequest $request): ResponseInterface
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
/** @var PostgresConnection $conn */
|
||||||
|
$conn = Model::getConnectionResolver()->connection();
|
||||||
|
$conn->getPdo()->exec('SELECT 1');
|
||||||
|
|
||||||
|
$response = Redis::ping();
|
||||||
|
if ($response->getPayload() !== 'PONG') {
|
||||||
|
throw new \Exception('Redis ping failed');
|
||||||
|
}
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return JsonResponseFactory::createJsonResponse(
|
||||||
|
[
|
||||||
|
'status_code' => CodesEnum::SERVICE_UNAVAILABLE->value,
|
||||||
|
'message' => 'Healthcheck Failed',
|
||||||
|
'error' => $e->getMessage(),
|
||||||
|
],
|
||||||
|
CodesEnum::SERVICE_UNAVAILABLE
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return JsonResponseFactory::createJsonResponse(
|
||||||
|
['status_code' => 200, 'message' => 'Healthcheck OK']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ namespace Siteworxpro\App\Controllers;
|
|||||||
|
|
||||||
use Nyholm\Psr7\ServerRequest;
|
use Nyholm\Psr7\ServerRequest;
|
||||||
use Psr\Http\Message\ResponseInterface;
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Siteworxpro\App\Annotations\Guards;
|
||||||
use Siteworxpro\App\Http\JsonResponseFactory;
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -20,8 +21,20 @@ class IndexController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws \JsonException
|
* @throws \JsonException
|
||||||
*/
|
*/
|
||||||
|
#[Guards\Jwt]
|
||||||
|
#[Guards\Scope(['get.index'])]
|
||||||
public function get(ServerRequest $request): ResponseInterface
|
public function get(ServerRequest $request): ResponseInterface
|
||||||
{
|
{
|
||||||
return JsonResponseFactory::createJsonResponse(['status_code' => 200, 'message' => 'Server is running']);
|
return JsonResponseFactory::createJsonResponse(['status_code' => 200, 'message' => 'Server is running']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \JsonException
|
||||||
|
*/
|
||||||
|
#[Guards\Jwt]
|
||||||
|
#[Guards\Scope(['post.index'])]
|
||||||
|
public function post(ServerRequest $request): ResponseInterface
|
||||||
|
{
|
||||||
|
return JsonResponseFactory::createJsonResponse(['status_code' => 200, 'message' => 'Server is running']);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
205
src/Events/Dispatcher.php
Normal file
205
src/Events/Dispatcher.php
Normal file
@@ -0,0 +1,205 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Events;
|
||||||
|
|
||||||
|
use Illuminate\Contracts\Events\Dispatcher as DispatcherContract;
|
||||||
|
use Illuminate\Contracts\Support\Arrayable;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
use Siteworxpro\App\Annotations\Events\ListensFor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Dispatcher
|
||||||
|
*
|
||||||
|
* A custom event dispatcher that automatically registers event listeners
|
||||||
|
* based on the ListensFor attribute.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Events
|
||||||
|
*/
|
||||||
|
class Dispatcher implements DispatcherContract, Arrayable
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var array $listeners Registered event listeners
|
||||||
|
*/
|
||||||
|
private array $listeners = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Collection $pushed Pushed events collection
|
||||||
|
*/
|
||||||
|
private Collection $pushed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string LISTENERS_NAMESPACE The namespace where listeners are located
|
||||||
|
*/
|
||||||
|
private const string LISTENERS_NAMESPACE = 'Siteworxpro\\App\\Events\\Listeners\\';
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
$this->pushed = new Collection();
|
||||||
|
$this->registerListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register event listeners based on the ListensFor attribute.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function registerListeners(): void
|
||||||
|
{
|
||||||
|
// traverse the Listeners directory and register all listeners
|
||||||
|
$listenersPath = __DIR__ . '/Listeners';
|
||||||
|
$iterator = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($listenersPath));
|
||||||
|
|
||||||
|
foreach ($iterator as $file) {
|
||||||
|
if ($file->isFile() && $file->getExtension() === 'php') {
|
||||||
|
$relativePath = str_replace($listenersPath . '/', '', $file->getPathname());
|
||||||
|
$className = self::LISTENERS_NAMESPACE . str_replace(['/', '.php'], ['\\', ''], $relativePath);
|
||||||
|
if (class_exists($className)) {
|
||||||
|
$reflectionClass = new \ReflectionClass($className);
|
||||||
|
$attributes = $reflectionClass->getAttributes(ListensFor::class);
|
||||||
|
foreach ($attributes as $attribute) {
|
||||||
|
$instance = $attribute->newInstance();
|
||||||
|
$eventClass = $instance->eventClass;
|
||||||
|
$this->listen($eventClass, new $className());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a listener for the given events.
|
||||||
|
*
|
||||||
|
* @param $events
|
||||||
|
* @param $listener
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function listen($events, $listener = null): void
|
||||||
|
{
|
||||||
|
$this->listeners[$events][] = $listener;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if there are listeners for the given event.
|
||||||
|
*
|
||||||
|
* @param $eventName
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function hasListeners($eventName): bool
|
||||||
|
{
|
||||||
|
return isset($this->listeners[$eventName]) && !empty($this->listeners[$eventName]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subscribe a subscriber to the dispatcher.
|
||||||
|
*
|
||||||
|
* @param Arrayable $subscriber
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function subscribe($subscriber): void
|
||||||
|
{
|
||||||
|
$this->listeners = array_merge($this->listeners, (array) $subscriber);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dispatch an event and halt on the first non-null response.
|
||||||
|
*
|
||||||
|
* @param $event
|
||||||
|
* @param array $payload
|
||||||
|
* @return array|null
|
||||||
|
*/
|
||||||
|
public function until($event, $payload = []): array|null
|
||||||
|
{
|
||||||
|
return $this->dispatch($event, $payload, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dispatch an event to its listeners.
|
||||||
|
*
|
||||||
|
* @param $event
|
||||||
|
* @param array $payload
|
||||||
|
* @param bool $halt
|
||||||
|
* @return array|null
|
||||||
|
*/
|
||||||
|
public function dispatch($event, $payload = [], $halt = false): array|null
|
||||||
|
{
|
||||||
|
if (is_object($event)) {
|
||||||
|
$eventClass = get_class($event);
|
||||||
|
} else {
|
||||||
|
$eventClass = $event;
|
||||||
|
}
|
||||||
|
|
||||||
|
$listeners = $this->listeners[$eventClass] ?? null;
|
||||||
|
|
||||||
|
if ($listeners === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$responses = [];
|
||||||
|
|
||||||
|
foreach ($listeners as $listener) {
|
||||||
|
$response = $listener($event, $payload);
|
||||||
|
$responses[] = $response;
|
||||||
|
|
||||||
|
if ($halt && $response !== null) {
|
||||||
|
return $response;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $responses;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Push an event to be dispatched later.
|
||||||
|
*
|
||||||
|
* @param $event
|
||||||
|
* @param array $payload
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function push($event, $payload = []): void
|
||||||
|
{
|
||||||
|
$this->pushed->put($event, $payload);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flush a pushed event, dispatching it if it exists.
|
||||||
|
*
|
||||||
|
* @param $event
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function flush($event): void
|
||||||
|
{
|
||||||
|
if ($this->pushed->has($event)) {
|
||||||
|
$payload = $this->pushed->get($event);
|
||||||
|
$this->dispatch($event, $payload);
|
||||||
|
$this->pushed->forget([$event]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Forget a pushed event without dispatching it.
|
||||||
|
*
|
||||||
|
* @param $event
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function forget($event): void
|
||||||
|
{
|
||||||
|
$this->pushed->forget([$event]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Forget all pushed events.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function forgetPushed(): void
|
||||||
|
{
|
||||||
|
$this->pushed = new Collection();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function toArray(): array
|
||||||
|
{
|
||||||
|
return $this->listeners;
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/Events/Listeners/Database/Connected.php
Normal file
32
src/Events/Listeners/Database/Connected.php
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Events\Listeners\Database;
|
||||||
|
|
||||||
|
use Illuminate\Database\Events\ConnectionEstablished;
|
||||||
|
use Illuminate\Database\Events\ConnectionEvent;
|
||||||
|
use Siteworxpro\App\Annotations\Events\ListensFor;
|
||||||
|
use Siteworxpro\App\Events\Listeners\Listener;
|
||||||
|
use Siteworxpro\App\Services\Facades\Logger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Connected
|
||||||
|
* @package Siteworxpro\App\Events\Listeners\Database
|
||||||
|
*/
|
||||||
|
#[ListensFor(ConnectionEstablished::class)]
|
||||||
|
class Connected extends Listener
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param ConnectionEvent $event
|
||||||
|
* @param array $payload
|
||||||
|
* @return null
|
||||||
|
*/
|
||||||
|
public function __invoke($event, array $payload = []): null
|
||||||
|
{
|
||||||
|
|
||||||
|
Logger::info("Database connection event", [get_class($event), $event->connectionName]);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
14
src/Events/Listeners/Listener.php
Normal file
14
src/Events/Listeners/Listener.php
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Events\Listeners;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Listener
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Events\Listeners
|
||||||
|
*/
|
||||||
|
abstract class Listener implements ListenerInterface
|
||||||
|
{
|
||||||
|
}
|
||||||
19
src/Events/Listeners/ListenerInterface.php
Normal file
19
src/Events/Listeners/ListenerInterface.php
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Events\Listeners;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface ListenerInterface
|
||||||
|
* @package Siteworxpro\App\Events\Listeners
|
||||||
|
*/
|
||||||
|
interface ListenerInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param mixed $event
|
||||||
|
* @param array $payload
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function __invoke(mixed $event, array $payload = []): mixed;
|
||||||
|
}
|
||||||
@@ -4,6 +4,10 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Siteworxpro\App\Helpers;
|
namespace Siteworxpro\App\Helpers;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Env
|
||||||
|
* @package Siteworxpro\App\Helpers
|
||||||
|
*/
|
||||||
abstract class Env
|
abstract class Env
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
|
|||||||
22
src/Helpers/Ulid.php
Normal file
22
src/Helpers/Ulid.php
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Helpers;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Ulid
|
||||||
|
* @package Siteworxpro\App\Helpers
|
||||||
|
*/
|
||||||
|
class Ulid
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Generate a ULID string
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function generate(): string
|
||||||
|
{
|
||||||
|
return \Ulid\Ulid::generate()->getRandomness();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace Siteworxpro\App\Http;
|
namespace Siteworxpro\App\Http;
|
||||||
|
|
||||||
use Nyholm\Psr7\Response;
|
use Nyholm\Psr7\Response;
|
||||||
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class JsonResponseFactory
|
* Class JsonResponseFactory
|
||||||
@@ -17,14 +18,14 @@ class JsonResponseFactory
|
|||||||
* Create a JSON response with the given data and status code.
|
* Create a JSON response with the given data and status code.
|
||||||
*
|
*
|
||||||
* @param array $data The data to include in the response.
|
* @param array $data The data to include in the response.
|
||||||
* @param int $statusCode The HTTP status code for the response.
|
* @param CodesEnum $statusCode The HTTP status code for the response.
|
||||||
* @return Response The JSON response.
|
* @return Response The JSON response.
|
||||||
* @throws \JsonException
|
* @throws \JsonException
|
||||||
*/
|
*/
|
||||||
public static function createJsonResponse(array $data, int $statusCode = 200): Response
|
public static function createJsonResponse(array $data, CodesEnum $statusCode = CodesEnum::OK): Response
|
||||||
{
|
{
|
||||||
return new Response(
|
return new Response(
|
||||||
status: $statusCode,
|
status: $statusCode->value,
|
||||||
headers: [
|
headers: [
|
||||||
'Content-Type' => 'application/json',
|
'Content-Type' => 'application/json',
|
||||||
],
|
],
|
||||||
|
|||||||
187
src/Http/Middleware/JwtMiddleware.php
Normal file
187
src/Http/Middleware/JwtMiddleware.php
Normal file
@@ -0,0 +1,187 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Http\Middleware;
|
||||||
|
|
||||||
|
use Carbon\Carbon;
|
||||||
|
use Carbon\WrapperClock;
|
||||||
|
use Lcobucci\JWT\JwtFacade;
|
||||||
|
use Lcobucci\JWT\Signer\Hmac\Sha256 as Hmac256;
|
||||||
|
use Lcobucci\JWT\Signer\Key\InMemory;
|
||||||
|
use Lcobucci\JWT\Signer\Rsa\Sha256;
|
||||||
|
use Lcobucci\JWT\Token\InvalidTokenStructure;
|
||||||
|
use Lcobucci\JWT\Validation\Constraint\IssuedBy;
|
||||||
|
use Lcobucci\JWT\Validation\Constraint\LooseValidAt;
|
||||||
|
use Lcobucci\JWT\Validation\Constraint\PermittedFor;
|
||||||
|
use Lcobucci\JWT\Validation\Constraint\SignedWith;
|
||||||
|
use Lcobucci\JWT\Validation\Constraint\StrictValidAt;
|
||||||
|
use Lcobucci\JWT\Validation\RequiredConstraintsViolated;
|
||||||
|
use League\Route\Dispatcher;
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Psr\Http\Message\ServerRequestInterface;
|
||||||
|
use Psr\Http\Server\RequestHandlerInterface;
|
||||||
|
use Siteworxpro\App\Annotations\Guards\Jwt;
|
||||||
|
use Siteworxpro\App\Controllers\Controller;
|
||||||
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JWT authorization middleware.
|
||||||
|
*
|
||||||
|
* Applies JWT validation to controller actions annotated with `Jwt` attribute.
|
||||||
|
* Flow:
|
||||||
|
* - Resolve the targeted controller and method for the current route.
|
||||||
|
* - If the method has `Jwt`, read the `Authorization` header and parse the Bearer token.
|
||||||
|
* - Validate signature, time constraints, issuer\(\) and audience\(\) based on attribute and config.
|
||||||
|
* - On success, attach all token claims to the request as attributes.
|
||||||
|
* - On failure, return a 401 JSON response with validation errors.
|
||||||
|
*
|
||||||
|
* Configuration:
|
||||||
|
* - `jwt.signing_key`: key material or `file://` path to key.
|
||||||
|
* - `jwt.strict_validation`: bool toggling strict vs loose time validation.
|
||||||
|
*/
|
||||||
|
class JwtMiddleware extends Middleware
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Process the incoming request.
|
||||||
|
*
|
||||||
|
* If the matched controller method is annotated with `Jwt`, validates the token and
|
||||||
|
* augments the request with claims on success. Otherwise, just delegates to the next handler.
|
||||||
|
*
|
||||||
|
* @param ServerRequestInterface $request PSR-7 request instance.
|
||||||
|
* @param RequestHandlerInterface|Dispatcher $handler Next middleware or route dispatcher.
|
||||||
|
*
|
||||||
|
* @return ResponseInterface Response produced by the next handler or a 401 JSON response.
|
||||||
|
*
|
||||||
|
* @throws \JsonException On JSON error response encoding issues.
|
||||||
|
* @throws \Exception On unexpected reflection or JWT parsing issues.
|
||||||
|
*/
|
||||||
|
public function process(
|
||||||
|
ServerRequestInterface $request,
|
||||||
|
RequestHandlerInterface|Dispatcher $handler
|
||||||
|
): ResponseInterface {
|
||||||
|
|
||||||
|
// Resolve the callable \[Controller, method] for the current route.
|
||||||
|
$callable = $this->extractRouteCallable($handler);
|
||||||
|
if ($callable === null) {
|
||||||
|
return $handler->handle($request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var Controller $class */
|
||||||
|
[$class, $method] = $callable;
|
||||||
|
|
||||||
|
if (class_exists($class::class)) {
|
||||||
|
$reflectionClass = new \ReflectionClass($class);
|
||||||
|
|
||||||
|
if ($reflectionClass->hasMethod($method)) {
|
||||||
|
$reflectionMethod = $reflectionClass->getMethod($method);
|
||||||
|
// Read `Jwt` attribute on the controller method.
|
||||||
|
$attributes = $reflectionMethod->getAttributes(Jwt::class);
|
||||||
|
|
||||||
|
// If no `Jwt` attribute, do not enforce auth here.
|
||||||
|
if (empty($attributes)) {
|
||||||
|
return $handler->handle($request);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract Bearer token from Authorization header.
|
||||||
|
$token = str_replace('Bearer ', '', $request->getHeaderLine('Authorization'));
|
||||||
|
|
||||||
|
if (empty($token)) {
|
||||||
|
return JsonResponseFactory::createJsonResponse([
|
||||||
|
'status_code' => 401,
|
||||||
|
'message' => 'Unauthorized: Missing token',
|
||||||
|
], CodesEnum::UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Aggregate required issuers and audience from attributes.
|
||||||
|
$requiredIssuers = [];
|
||||||
|
$requiredAudience = '';
|
||||||
|
|
||||||
|
foreach ($attributes as $attribute) {
|
||||||
|
/** @var Jwt $jwtInstance */
|
||||||
|
$jwtInstance = $attribute->newInstance();
|
||||||
|
|
||||||
|
if ($jwtInstance->getRequiredAudience() !== '') {
|
||||||
|
$requiredAudience = $jwtInstance->getAudience();
|
||||||
|
}
|
||||||
|
|
||||||
|
$requiredIssuers[] = $jwtInstance->getIssuer();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Parse and validate the token with signature, time, issuer and audience constraints.
|
||||||
|
$jwt = new JwtFacade()->parse(
|
||||||
|
$token,
|
||||||
|
$this->getSignedWith(),
|
||||||
|
Config::get('jwt.strict_validation') ?
|
||||||
|
new StrictValidAt(new WrapperClock(Carbon::now())) :
|
||||||
|
new LooseValidAt(new WrapperClock(Carbon::now())),
|
||||||
|
new IssuedBy(...$requiredIssuers),
|
||||||
|
new PermittedFor($requiredAudience)
|
||||||
|
);
|
||||||
|
} catch (RequiredConstraintsViolated $exception) {
|
||||||
|
// Collect human-readable violations to return to the client.
|
||||||
|
$violations = [];
|
||||||
|
foreach ($exception->violations() as $violation) {
|
||||||
|
$violations[] = $violation->getMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
return JsonResponseFactory::createJsonResponse([
|
||||||
|
'status_code' => 401,
|
||||||
|
'message' => 'Unauthorized: Invalid token',
|
||||||
|
'errors' => $violations
|
||||||
|
], CodesEnum::UNAUTHORIZED);
|
||||||
|
} catch (InvalidTokenStructure) {
|
||||||
|
// Token could not be parsed due to malformed structure.
|
||||||
|
return JsonResponseFactory::createJsonResponse([
|
||||||
|
'status_code' => 401,
|
||||||
|
'message' => 'Unauthorized: Invalid token',
|
||||||
|
], CodesEnum::UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Expose all token claims as request attributes for downstream consumers.
|
||||||
|
foreach ($jwt->claims()->all() as $item => $value) {
|
||||||
|
$request = $request->withAttribute($item, $value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $handler->handle($request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build the signature validation constraint from configured key.
|
||||||
|
*
|
||||||
|
* - If the configured key content includes the string `PUBLIC KEY`, use RSA SHA-256.
|
||||||
|
* - Otherwise assume an HMAC SHA-256 shared secret.
|
||||||
|
* - Supports raw key strings or `file://` paths.
|
||||||
|
*
|
||||||
|
* @return SignedWith Signature constraint used during JWT parsing.
|
||||||
|
*
|
||||||
|
* @throws \RuntimeException When no signing key is configured.
|
||||||
|
*/
|
||||||
|
private function getSignedWith(): SignedWith
|
||||||
|
{
|
||||||
|
$key = Config::get('jwt.signing_key');
|
||||||
|
|
||||||
|
if ($key === null) {
|
||||||
|
throw new \RuntimeException('JWT signing key is not configured.');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load key either from file or raw text.
|
||||||
|
if (str_starts_with($key, 'file://')) {
|
||||||
|
$key = InMemory::file(substr($key, 7));
|
||||||
|
} else {
|
||||||
|
$key = InMemory::plainText($key);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Heuristic: if PEM public key content is detected, use RSA; otherwise use HMAC.
|
||||||
|
if (str_contains($key->contents(), 'PUBLIC KEY')) {
|
||||||
|
return new SignedWith(new Sha256(), $key);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new SignedWith(new Hmac256(), $key);
|
||||||
|
}
|
||||||
|
}
|
||||||
71
src/Http/Middleware/Middleware.php
Normal file
71
src/Http/Middleware/Middleware.php
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Http\Middleware;
|
||||||
|
|
||||||
|
use League\Route\Dispatcher;
|
||||||
|
use League\Route\Route;
|
||||||
|
use Psr\Http\Server\MiddlewareInterface;
|
||||||
|
use Psr\Http\Server\RequestHandlerInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Base middleware helper for extracting route callables.
|
||||||
|
*
|
||||||
|
* This abstract middleware provides a utility method to inspect a League\Route
|
||||||
|
* dispatcher and obtain the underlying route callable as a [class, method] tuple.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Http\Middleware
|
||||||
|
*/
|
||||||
|
abstract class Middleware implements MiddlewareInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Extract the route callable [class, method] from a League\Route dispatcher.
|
||||||
|
*
|
||||||
|
* When the provided handler is a League\Route\Dispatcher, this inspects its
|
||||||
|
* middleware stack, looks at the last segment (the resolved Route), and
|
||||||
|
* attempts to normalize its callable into a [class, method] pair.
|
||||||
|
*
|
||||||
|
* Supported callable forms:
|
||||||
|
* - array callable: [object|class-string, method-string]
|
||||||
|
* - string callable: "ClassName::methodName"
|
||||||
|
*
|
||||||
|
* Returns null when the handler is not a Dispatcher, the stack is empty,
|
||||||
|
* or the callable cannot be parsed.
|
||||||
|
*
|
||||||
|
* @param RequestHandlerInterface|Dispatcher $handler The downstream handler or dispatcher.
|
||||||
|
*
|
||||||
|
* @return array{0: class-string|object|null, 1: string|null}|null Tuple of [class|object, method] or null.
|
||||||
|
*/
|
||||||
|
protected function extractRouteCallable(
|
||||||
|
RequestHandlerInterface|Dispatcher $handler
|
||||||
|
): array|null {
|
||||||
|
// Only proceed if this is a League\Route dispatcher.
|
||||||
|
if (!$handler instanceof Dispatcher) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var Route | null $lastSegment */
|
||||||
|
// Retrieve the last middleware in the stack, which should be the Route.
|
||||||
|
$lastSegment = array_last($handler->getMiddlewareStack());
|
||||||
|
|
||||||
|
if ($lastSegment === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Obtain the callable associated with the route.
|
||||||
|
$callable = $lastSegment->getCallable();
|
||||||
|
$class = null;
|
||||||
|
$method = null;
|
||||||
|
|
||||||
|
// Handle array callable: [object|class-string, 'method']
|
||||||
|
if (is_array($callable) && count($callable) === 2) {
|
||||||
|
[$class, $method] = $callable;
|
||||||
|
} elseif (is_string($callable)) {
|
||||||
|
// Handle string callable: 'ClassName::methodName'
|
||||||
|
[$class, $method] = explode('::', $callable);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$class, $method];
|
||||||
|
}
|
||||||
|
}
|
||||||
94
src/Http/Middleware/ScopeMiddleware.php
Normal file
94
src/Http/Middleware/ScopeMiddleware.php
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Http\Middleware;
|
||||||
|
|
||||||
|
use League\Route\Dispatcher;
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Psr\Http\Message\ServerRequestInterface;
|
||||||
|
use Psr\Http\Server\RequestHandlerInterface;
|
||||||
|
use Siteworxpro\App\Annotations\Guards\Scope;
|
||||||
|
use Siteworxpro\App\Controllers\Controller;
|
||||||
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Middleware that enforces scope-based access control on controller actions.
|
||||||
|
*
|
||||||
|
* It inspects PHP 8 attributes of type \`Scope\` applied to the resolved controller method,
|
||||||
|
* compares the required scopes with the user scopes provided on the request attribute \`scopes\`,
|
||||||
|
* and returns a 403 JSON response when any required scope is missing.
|
||||||
|
*
|
||||||
|
* If the route callable cannot be resolved, or no scope is required, the request is passed through.
|
||||||
|
*
|
||||||
|
* @see Scope
|
||||||
|
*/
|
||||||
|
class ScopeMiddleware extends Middleware
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Resolve the route callable, read any \`Scope\` attributes, and enforce required scopes.
|
||||||
|
*
|
||||||
|
* Expected user scopes are provided on the request under the attribute name \`scopes\`
|
||||||
|
* as an array of strings.
|
||||||
|
*
|
||||||
|
* @param ServerRequestInterface $request Incoming PSR-7 request (expects \`scopes\` attribute).
|
||||||
|
* @param RequestHandlerInterface|Dispatcher $handler Next handler or League\Route dispatcher.
|
||||||
|
*
|
||||||
|
* @return ResponseInterface A 403 JSON response when scopes are insufficient; otherwise the handler response.
|
||||||
|
*
|
||||||
|
* @throws \JsonException If encoding the JSON error response fails.
|
||||||
|
* @throws \ReflectionException If reflection on the controller or method fails.
|
||||||
|
*/
|
||||||
|
public function process(
|
||||||
|
ServerRequestInterface $request,
|
||||||
|
RequestHandlerInterface | Dispatcher $handler
|
||||||
|
): ResponseInterface {
|
||||||
|
// Attempt to resolve the route's callable [Controller instance, method name].
|
||||||
|
$callable = $this->extractRouteCallable($handler);
|
||||||
|
if ($callable === null) {
|
||||||
|
// If no callable is available, delegate to the next handler.
|
||||||
|
return $handler->handle($request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var Controller $class Controller instance resolved from the route. */
|
||||||
|
[$class, $method] = $callable;
|
||||||
|
|
||||||
|
// Ensure the controller exists and the method is defined before reflecting.
|
||||||
|
if (class_exists($class::class)) {
|
||||||
|
$reflectionClass = new \ReflectionClass($class);
|
||||||
|
if ($reflectionClass->hasMethod($method)) {
|
||||||
|
$reflectionMethod = $reflectionClass->getMethod($method);
|
||||||
|
|
||||||
|
// Fetch all Scope attributes declared on the method.
|
||||||
|
$attributes = $reflectionMethod->getAttributes(Scope::class);
|
||||||
|
|
||||||
|
foreach ($attributes as $attribute) {
|
||||||
|
/** @var Scope $scopeInstance Concrete Scope attribute instance. */
|
||||||
|
$scopeInstance = $attribute->newInstance();
|
||||||
|
$requiredScopes = $scopeInstance->getScopes();
|
||||||
|
|
||||||
|
// Retrieve user scopes from the request (defaults to an empty array).
|
||||||
|
$userScopes = $request->getAttribute('scopes', []);
|
||||||
|
|
||||||
|
// Deny if any required scope is missing from the user's scopes.
|
||||||
|
if (
|
||||||
|
array_any(
|
||||||
|
$requiredScopes,
|
||||||
|
fn($requiredScope) => !in_array($requiredScope, $userScopes, true)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return JsonResponseFactory::createJsonResponse([
|
||||||
|
'error' => 'insufficient_scope',
|
||||||
|
'error_description' =>
|
||||||
|
'The request requires higher privileges than provided by the access token.'
|
||||||
|
], CodesEnum::FORBIDDEN);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// All checks passed; continue down the middleware pipeline.
|
||||||
|
return $handler->handle($request);
|
||||||
|
}
|
||||||
|
}
|
||||||
95
src/Kernel.php
Normal file
95
src/Kernel.php
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Siteworxpro\App;
|
||||||
|
|
||||||
|
use Illuminate\Container\Container;
|
||||||
|
use Illuminate\Database\Capsule\Manager;
|
||||||
|
use Illuminate\Support\ServiceProvider;
|
||||||
|
use Siteworx\Config\Config as SWConfig;
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
use Siteworxpro\App\Services\Facades\Dispatcher;
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\BrokerServiceProvider;
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\DispatcherServiceProvider;
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\LoggerServiceProvider;
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\RedisServiceProvider;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Kernel
|
||||||
|
*
|
||||||
|
* The Kernel class is responsible for bootstrapping the application by
|
||||||
|
* initializing service providers and setting up the database connection.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App
|
||||||
|
*/
|
||||||
|
class Kernel
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* List of service providers to be registered during bootstrapping.
|
||||||
|
*
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
private static array $serviceProviders = [
|
||||||
|
LoggerServiceProvider::class,
|
||||||
|
RedisServiceProvider::class,
|
||||||
|
DispatcherServiceProvider::class,
|
||||||
|
BrokerServiceProvider::class
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bootstraps the server by initializing the PSR-7 worker and router.
|
||||||
|
*
|
||||||
|
* This method sets up the PSR-7 worker and router instances, and registers
|
||||||
|
* the routes for the server. It should be called in the constructor of
|
||||||
|
* subclasses to ensure proper initialization.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
* @throws \ReflectionException
|
||||||
|
*/
|
||||||
|
public static function boot(): void
|
||||||
|
{
|
||||||
|
$container = new Container();
|
||||||
|
Facade::setFacadeContainer($container);
|
||||||
|
|
||||||
|
// Bind the container to the Config facade first so that it can be used by service providers
|
||||||
|
$container->bind(SWConfig::class, function () {
|
||||||
|
return SWConfig::load(__DIR__ . '/../config.php');
|
||||||
|
});
|
||||||
|
|
||||||
|
foreach (self::$serviceProviders as $serviceProvider) {
|
||||||
|
if (class_exists($serviceProvider)) {
|
||||||
|
$provider = new $serviceProvider($container);
|
||||||
|
if ($provider instanceof ServiceProvider) {
|
||||||
|
$provider->register();
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException(sprintf(
|
||||||
|
'Service provider %s is not an instance of ServiceProvider.',
|
||||||
|
$serviceProvider
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException(sprintf('Service provider %s not found.', $serviceProvider));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self::bootModelCapsule();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bootstraps the model capsule for database connections.
|
||||||
|
*
|
||||||
|
* This method sets up the database connection using the Eloquent ORM.
|
||||||
|
* It retrieves the database configuration from the Config facade and
|
||||||
|
* initializes the Eloquent capsule manager.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private static function bootModelCapsule(): void
|
||||||
|
{
|
||||||
|
$capsule = new Manager();
|
||||||
|
$capsule->setEventDispatcher(Dispatcher::getFacadeRoot());
|
||||||
|
$capsule->addConnection(Config::get('db'));
|
||||||
|
$capsule->setAsGlobal();
|
||||||
|
$capsule->bootEloquent();
|
||||||
|
}
|
||||||
|
}
|
||||||
236
src/Log/Logger.php
Normal file
236
src/Log/Logger.php
Normal file
@@ -0,0 +1,236 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Log;
|
||||||
|
|
||||||
|
use Monolog\Formatter\JsonFormatter;
|
||||||
|
use Monolog\Handler\StreamHandler;
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Psr\Log\LogLevel;
|
||||||
|
use RoadRunner\Logger\Logger as RRLogger;
|
||||||
|
use Siteworxpro\App\Services\Facades\RoadRunnerLogger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logger implementation that conforms to PSR-3 (`Psr\Log\LoggerInterface`).
|
||||||
|
*
|
||||||
|
* Behavior:
|
||||||
|
* - If environment indicates RoadRunner RPC (`$_SERVER['RR_RPC']`), logs are forwarded
|
||||||
|
* to a RoadRunner RPC logger (`RoadRunner\Logger\Logger`) created via Goridge RPC.
|
||||||
|
* - Otherwise, logs are written to `php://stdout` using Monolog with a JSON formatter.
|
||||||
|
* - Messages below the configured threshold are ignored (level filtering).
|
||||||
|
*
|
||||||
|
* Supported PSR-3 levels are mapped to an internal numeric ordering in `$levels`.
|
||||||
|
* When using the RPC logger, levels are translated to the respective RPC methods
|
||||||
|
* (debug, info, warning, error). When using Monolog, the numeric mapping is used
|
||||||
|
* as the numeric level passed to Monolog's `log` method.
|
||||||
|
*/
|
||||||
|
class Logger implements LoggerInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* RoadRunner RPC logger instance when running under RoadRunner.
|
||||||
|
*
|
||||||
|
* @var RRLogger | LoggerInterface | null
|
||||||
|
*/
|
||||||
|
private RRLogger | LoggerInterface | null $rpcLogger = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Monolog logger used as a fallback to write JSON-formatted logs to stdout.
|
||||||
|
*
|
||||||
|
* @var \Monolog\Logger
|
||||||
|
*/
|
||||||
|
private \Monolog\Logger $monologLogger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Numeric ordering for PSR-3 log levels.
|
||||||
|
*
|
||||||
|
* Lower numbers represent higher severity. This mapping is used for filtering
|
||||||
|
* messages according to the configured minimum level and for Monolog numeric level.
|
||||||
|
*
|
||||||
|
* @var array<string,int>
|
||||||
|
*/
|
||||||
|
private array $levels = [
|
||||||
|
LogLevel::EMERGENCY => 0,
|
||||||
|
LogLevel::ALERT => 1,
|
||||||
|
LogLevel::CRITICAL => 2,
|
||||||
|
LogLevel::ERROR => 3,
|
||||||
|
LogLevel::WARNING => 4,
|
||||||
|
LogLevel::NOTICE => 5,
|
||||||
|
LogLevel::INFO => 6,
|
||||||
|
LogLevel::DEBUG => 7,
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new Logger.
|
||||||
|
*
|
||||||
|
* @param string $level Minimum level to log (PSR-3 level string). Messages with
|
||||||
|
* a higher numeric value in `$levels` will be ignored.
|
||||||
|
*
|
||||||
|
* @param resource | null $streamOutput Optional stream handler for Monolog.
|
||||||
|
*
|
||||||
|
* The default is `LogLevel::DEBUG` (log everything).
|
||||||
|
*
|
||||||
|
* If `$_SERVER['RR_RPC']` is set, an RPC connection will be attempted at
|
||||||
|
* $_SERVER['RR_RPC'] and a RoadRunner RPC logger will be used.
|
||||||
|
*
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
private readonly string $level = LogLevel::DEBUG,
|
||||||
|
$streamOutput = null,
|
||||||
|
) {
|
||||||
|
if (isset($_SERVER['RR_RPC'])) {
|
||||||
|
$this->rpcLogger = RoadRunnerLogger::getFacadeRoot();
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->monologLogger = new \Monolog\Logger('app_logger');
|
||||||
|
$formatter = new JsonFormatter();
|
||||||
|
$stream = $streamOutput ?? 'php://stdout';
|
||||||
|
$this->monologLogger->pushHandler(new StreamHandler($stream)->setFormatter($formatter));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* System is unusable.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function emergency(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::EMERGENCY, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Action must be taken immediately.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function alert(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::ALERT, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Critical conditions.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function critical(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::CRITICAL, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runtime errors that do not require immediate action but should typically be logged and monitored.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function error(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::ERROR, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exceptional occurrences that are not errors.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function warning(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::WARNING, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normal but significant events.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function notice(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::NOTICE, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interesting events.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function info(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::INFO, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detailed debug information.
|
||||||
|
*
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function debug(\Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
$this->log(LogLevel::DEBUG, $message, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs with an arbitrary level.
|
||||||
|
*
|
||||||
|
* Behavior details:
|
||||||
|
* - If the provided `$level` maps to a numeric value greater than the configured
|
||||||
|
* minimum level, the message is discarded (filtered).
|
||||||
|
* - If an RPC logger is available, the message is forwarded to the RPC logger
|
||||||
|
* using a method chosen by level (debug, info, warning, error).
|
||||||
|
* - Otherwise, the message is written to Monolog using the numeric mapping.
|
||||||
|
*
|
||||||
|
* Notes:
|
||||||
|
* - `$level` should be a PSR-3 level string (values defined in `Psr\Log\LogLevel`).
|
||||||
|
* - If an unknown level string is passed, accessing `$this->levels[$level]` may
|
||||||
|
* trigger a PHP notice or undefined index. Ensure callers use valid PSR-3 levels.
|
||||||
|
*
|
||||||
|
* @param mixed $level PSR-3 log level (string)
|
||||||
|
* @param \Stringable|string $message
|
||||||
|
* @param array $context
|
||||||
|
*/
|
||||||
|
public function log($level, \Stringable|string $message, array $context = []): void
|
||||||
|
{
|
||||||
|
if (isset($this->levels[$level]) && $this->levels[$level] > $this->levels[$this->level]) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->rpcLogger) {
|
||||||
|
switch ($level) {
|
||||||
|
case LogLevel::DEBUG:
|
||||||
|
$this->rpcLogger->debug((string)$message, $context);
|
||||||
|
break;
|
||||||
|
case LogLevel::NOTICE:
|
||||||
|
case LogLevel::INFO:
|
||||||
|
$this->rpcLogger->info((string)$message, $context);
|
||||||
|
break;
|
||||||
|
case LogLevel::WARNING:
|
||||||
|
$this->rpcLogger->warning((string)$message, $context);
|
||||||
|
break;
|
||||||
|
case LogLevel::CRITICAL:
|
||||||
|
case LogLevel::ERROR:
|
||||||
|
case LogLevel::ALERT:
|
||||||
|
case LogLevel::EMERGENCY:
|
||||||
|
$this->rpcLogger->error((string)$message, $context);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$this->rpcLogger->log($level, (string)$message, $context);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->monologLogger->log($this->levels[$level], (string)$message, $context);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,12 @@ namespace Siteworxpro\App\Models;
|
|||||||
|
|
||||||
use Illuminate\Database\Eloquent\Model as ORM;
|
use Illuminate\Database\Eloquent\Model as ORM;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Model
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Models
|
||||||
|
*/
|
||||||
abstract class Model extends ORM
|
abstract class Model extends ORM
|
||||||
{
|
{
|
||||||
|
protected $dateFormat = 'Y-m-d H:i:s';
|
||||||
}
|
}
|
||||||
|
|||||||
52
src/Models/User.php
Normal file
52
src/Models/User.php
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Models;
|
||||||
|
|
||||||
|
use Carbon\Carbon;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class User
|
||||||
|
*
|
||||||
|
* @property string $id
|
||||||
|
* @property string $first_name
|
||||||
|
* @property string $last_name
|
||||||
|
* @property string $email
|
||||||
|
* @property string $password
|
||||||
|
* @property Carbon $created_at
|
||||||
|
*
|
||||||
|
* @property-read string $full_name
|
||||||
|
* @property-read string $formatted_email
|
||||||
|
*/
|
||||||
|
class User extends Model
|
||||||
|
{
|
||||||
|
protected $casts = [
|
||||||
|
'created_at' => 'datetime',
|
||||||
|
];
|
||||||
|
|
||||||
|
protected $hidden = [
|
||||||
|
'password',
|
||||||
|
];
|
||||||
|
|
||||||
|
protected $fillable = [
|
||||||
|
'first_name',
|
||||||
|
'last_name',
|
||||||
|
'email',
|
||||||
|
'password',
|
||||||
|
];
|
||||||
|
|
||||||
|
public function getFullNameAttribute(): string
|
||||||
|
{
|
||||||
|
return "$this->first_name $this->last_name";
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getFormattedEmailAttribute(): string
|
||||||
|
{
|
||||||
|
return sprintf(
|
||||||
|
'%s <%s>',
|
||||||
|
$this->getFullNameAttribute(),
|
||||||
|
strtolower($this->email)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace Siteworxpro\App\Services;
|
namespace Siteworxpro\App\Services;
|
||||||
|
|
||||||
use Illuminate\Contracts\Container\Container;
|
use Illuminate\Contracts\Container\Container;
|
||||||
|
use Illuminate\Support\HigherOrderTapProxy;
|
||||||
use Illuminate\Support\Testing\Fakes\Fake;
|
use Illuminate\Support\Testing\Fakes\Fake;
|
||||||
use Mockery;
|
use Mockery;
|
||||||
use Mockery\Expectation;
|
use Mockery\Expectation;
|
||||||
@@ -57,9 +58,9 @@ class Facade
|
|||||||
/**
|
/**
|
||||||
* Convert the facade into a Mockery spy.
|
* Convert the facade into a Mockery spy.
|
||||||
*
|
*
|
||||||
* @return MockInterface
|
* @return HigherOrderTapProxy | MockInterface
|
||||||
*/
|
*/
|
||||||
public static function spy(): MockInterface
|
public static function spy(): HigherOrderTapProxy | MockInterface
|
||||||
{
|
{
|
||||||
if (! static::isMock()) {
|
if (! static::isMock()) {
|
||||||
$class = static::getMockableClass();
|
$class = static::getMockableClass();
|
||||||
@@ -278,10 +279,10 @@ class Facade
|
|||||||
/**
|
/**
|
||||||
* Set the application instance.
|
* Set the application instance.
|
||||||
*
|
*
|
||||||
* @param Container $container
|
* @param Container | null $container
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public static function setFacadeContainer(Container $container): void
|
public static function setFacadeContainer(Container | null $container): void
|
||||||
{
|
{
|
||||||
static::$container = $container;
|
static::$container = $container;
|
||||||
}
|
}
|
||||||
|
|||||||
29
src/Services/Facades/Broker.php
Normal file
29
src/Services/Facades/Broker.php
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Services\Facades;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Async\Messages\Message;
|
||||||
|
use Siteworxpro\App\Async\Queues\Queue;
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Broker Facade
|
||||||
|
*
|
||||||
|
* @method static void publish(Queue $queue, Message $message, int $delay = 0)
|
||||||
|
* @method static void publishLater(Queue $queue, Message $message, int $delay)
|
||||||
|
* @method static Message|null consume(Queue $queue)
|
||||||
|
*/
|
||||||
|
class Broker extends Facade
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Get the registered name of the component.
|
||||||
|
*
|
||||||
|
* @return string The name of the component.
|
||||||
|
*/
|
||||||
|
protected static function getFacadeAccessor(): string
|
||||||
|
{
|
||||||
|
return \Siteworxpro\App\Async\Brokers\Broker::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
35
src/Services/Facades/Dispatcher.php
Normal file
35
src/Services/Facades/Dispatcher.php
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Services\Facades;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Events\Dispatcher as DispatcherConcrete;
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Dispatcher
|
||||||
|
*
|
||||||
|
* A facade for the event dispatcher.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Services\Facades
|
||||||
|
*
|
||||||
|
* @method static void listen(string $event, callable|string $listener)
|
||||||
|
* @method static void dispatch(object|string $event, array $payload = [], bool $halt = false)
|
||||||
|
* @method static void push(object|string $event, array $payload = [])
|
||||||
|
* @method static array|null until(object|string $event, array $payload = [])
|
||||||
|
* @method static bool hasListeners(string $eventName)
|
||||||
|
* @method static void subscribe(mixed $subscriber)
|
||||||
|
*/
|
||||||
|
class Dispatcher extends Facade
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Get the registered name of the component.
|
||||||
|
*
|
||||||
|
* @return string The name of the component.
|
||||||
|
*/
|
||||||
|
protected static function getFacadeAccessor(): string
|
||||||
|
{
|
||||||
|
return DispatcherConcrete::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,7 +4,6 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Siteworxpro\App\Services\Facades;
|
namespace Siteworxpro\App\Services\Facades;
|
||||||
|
|
||||||
use RoadRunner\Logger\Logger as RRLogger;
|
|
||||||
use Siteworxpro\App\Services\Facade;
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -13,10 +12,13 @@ use Siteworxpro\App\Services\Facade;
|
|||||||
* This class serves as a facade for the Monolog logger.
|
* This class serves as a facade for the Monolog logger.
|
||||||
* It extends the Facade class from the Illuminate\Support\Facades namespace.
|
* It extends the Facade class from the Illuminate\Support\Facades namespace.
|
||||||
*
|
*
|
||||||
* @method static debug(string $message, array $context = []) Log an informational message.
|
* @method static debug(\Stringable|string $message, array $context = []) Log an informational message.
|
||||||
* @method static info(string $message, array $context = []) Log an informational message.
|
* @method static info(\Stringable|string $message, array $context = []) Log an informational message.
|
||||||
* @method static error(string $message, array $context = []) Log an error message.
|
* @method static error(\Stringable|string $message, array $context = []) Log an error message.
|
||||||
* @method static warning(string $message, array $context = []) Log a warning message.
|
* @method static warning(\Stringable|string $message, array $context = []) Log a warning message.
|
||||||
|
* @method static critical(\Stringable|string $message, array $context = []) Log a critical error message.
|
||||||
|
* @method static alert(\Stringable|string $message, array $context = []) Log an alert message.
|
||||||
|
* @method static emergency(\Stringable|string $message, array $context = []) Log an emergency message.
|
||||||
*
|
*
|
||||||
* @package Siteworxpro\App\Facades
|
* @package Siteworxpro\App\Facades
|
||||||
*/
|
*/
|
||||||
@@ -29,6 +31,6 @@ class Logger extends Facade
|
|||||||
*/
|
*/
|
||||||
protected static function getFacadeAccessor(): string
|
protected static function getFacadeAccessor(): string
|
||||||
{
|
{
|
||||||
return RRLogger::class;
|
return \Siteworxpro\App\Log\Logger::class;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ use Siteworxpro\App\Services\Facade;
|
|||||||
* @method static Status|null set(string $key, $value, $expireResolution = null, $expireTTL = null, $flag = null)
|
* @method static Status|null set(string $key, $value, $expireResolution = null, $expireTTL = null, $flag = null)
|
||||||
* @method static array keys(string $pattern)
|
* @method static array keys(string $pattern)
|
||||||
* @method static int del(string $key)
|
* @method static int del(string $key)
|
||||||
|
* @method static Status ping()
|
||||||
*/
|
*/
|
||||||
class Redis extends Facade
|
class Redis extends Facade
|
||||||
{
|
{
|
||||||
|
|||||||
39
src/Services/Facades/RoadRunnerLogger.php
Normal file
39
src/Services/Facades/RoadRunnerLogger.php
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Services\Facades;
|
||||||
|
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
|
use RoadRunner\Logger\Logger;
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
use Spiral\Goridge\RPC\RPC;
|
||||||
|
|
||||||
|
class RoadRunnerLogger extends Facade
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public static function getFacadeRoot(): mixed
|
||||||
|
{
|
||||||
|
$container = static::getFacadeContainer();
|
||||||
|
if ($container && $container->has(Logger::class) === false) {
|
||||||
|
$rpc = RPC::create($_SERVER['RR_RPC']);
|
||||||
|
$logger = new Logger($rpc);
|
||||||
|
$container->bind(static::getFacadeAccessor(), function () use ($logger) {
|
||||||
|
return $logger;
|
||||||
|
});
|
||||||
|
|
||||||
|
return $logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $container->get(Logger::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function getFacadeAccessor(): string
|
||||||
|
{
|
||||||
|
return Logger::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
45
src/Services/ServiceProviders/BrokerServiceProvider.php
Normal file
45
src/Services/ServiceProviders/BrokerServiceProvider.php
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Services\ServiceProviders;
|
||||||
|
|
||||||
|
use Illuminate\Support\ServiceProvider;
|
||||||
|
use Siteworxpro\App\Async\Brokers\Broker;
|
||||||
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class BrokerServiceProvider
|
||||||
|
*
|
||||||
|
* This service provider is responsible for binding the Broker implementation
|
||||||
|
* to the Laravel service container based on configuration settings.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Services\ServiceProviders
|
||||||
|
*/
|
||||||
|
class BrokerServiceProvider extends ServiceProvider
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Register services.
|
||||||
|
*
|
||||||
|
* This method binds the Broker interface to a specific implementation
|
||||||
|
* based on the configuration defined in 'queue.broker' and 'queue.broker_config'.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
* @throws \RuntimeException if the specified broker class does not exist.
|
||||||
|
*/
|
||||||
|
public function register(): void
|
||||||
|
{
|
||||||
|
$this->app->singleton(Broker::class, function (): Broker {
|
||||||
|
$configName = Config::get('queue.broker');
|
||||||
|
$brokerConfig = Config::get('queue.broker_config.' . $configName) ?? [];
|
||||||
|
|
||||||
|
$brokerClass = Broker::BROKER_TYPES[$configName] ?? null;
|
||||||
|
|
||||||
|
if ($brokerClass && class_exists($brokerClass)) {
|
||||||
|
return new $brokerClass($brokerConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new \RuntimeException("Broker class $brokerClass does not exist.");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
23
src/Services/ServiceProviders/DispatcherServiceProvider.php
Normal file
23
src/Services/ServiceProviders/DispatcherServiceProvider.php
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\App\Services\ServiceProviders;
|
||||||
|
|
||||||
|
use Illuminate\Support\ServiceProvider;
|
||||||
|
use Siteworxpro\App\Events\Dispatcher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class DispatcherServiceProvider
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Services\ServiceProviders
|
||||||
|
*/
|
||||||
|
class DispatcherServiceProvider extends ServiceProvider
|
||||||
|
{
|
||||||
|
public function register(): void
|
||||||
|
{
|
||||||
|
$this->app->singleton(Dispatcher::class, function () {
|
||||||
|
return new Dispatcher();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,16 +5,19 @@ declare(strict_types=1);
|
|||||||
namespace Siteworxpro\App\Services\ServiceProviders;
|
namespace Siteworxpro\App\Services\ServiceProviders;
|
||||||
|
|
||||||
use Illuminate\Support\ServiceProvider;
|
use Illuminate\Support\ServiceProvider;
|
||||||
use RoadRunner\Logger\Logger as RRLogger;
|
use Siteworxpro\App\Log\Logger;
|
||||||
use Spiral\Goridge\RPC\RPC;
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class LoggerServiceProvider
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Services\ServiceProviders
|
||||||
|
*/
|
||||||
class LoggerServiceProvider extends ServiceProvider
|
class LoggerServiceProvider extends ServiceProvider
|
||||||
{
|
{
|
||||||
public function register(): void
|
public function register(): void
|
||||||
{
|
{
|
||||||
$this->app->singleton(RRLogger::class, function () {
|
$this->app->singleton(Logger::class, function () {
|
||||||
$rpc = RPC::create('tcp://127.0.0.1:6001');
|
return new Logger();
|
||||||
return new RRLogger($rpc);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,13 @@ use Illuminate\Support\ServiceProvider;
|
|||||||
use Predis\Client;
|
use Predis\Client;
|
||||||
use Siteworxpro\App\Services\Facades\Config;
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class RedisServiceProvider
|
||||||
|
*
|
||||||
|
* This service provider registers a Redis client as a singleton in the application container.
|
||||||
|
*
|
||||||
|
* @package Siteworxpro\App\Services\ServiceProviders
|
||||||
|
*/
|
||||||
class RedisServiceProvider extends ServiceProvider
|
class RedisServiceProvider extends ServiceProvider
|
||||||
{
|
{
|
||||||
public function register(): void
|
public function register(): void
|
||||||
@@ -18,6 +25,7 @@ class RedisServiceProvider extends ServiceProvider
|
|||||||
'host' => Config::get('redis.host'),
|
'host' => Config::get('redis.host'),
|
||||||
'port' => Config::get('redis.port'),
|
'port' => Config::get('redis.port'),
|
||||||
'database' => Config::get('redis.database'),
|
'database' => Config::get('redis.database'),
|
||||||
|
'password' => Config::get('redis.password'),
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
16
tests/Controllers/AbstractController.php
Normal file
16
tests/Controllers/AbstractController.php
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Controllers;
|
||||||
|
|
||||||
|
use Nyholm\Psr7\ServerRequest;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
abstract class AbstractController extends Unit
|
||||||
|
{
|
||||||
|
protected function getMockRequest(string $method = 'GET', string $uri = '/'): ServerRequest
|
||||||
|
{
|
||||||
|
return new ServerRequest($method, $uri);
|
||||||
|
}
|
||||||
|
}
|
||||||
54
tests/Controllers/ControllerTest.php
Normal file
54
tests/Controllers/ControllerTest.php
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Controllers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Controllers\Controller;
|
||||||
|
|
||||||
|
class ControllerTest extends AbstractController
|
||||||
|
{
|
||||||
|
public function testNotFoundExceptions()
|
||||||
|
{
|
||||||
|
$testClass = new TestClass();
|
||||||
|
|
||||||
|
$this->expectException(\League\Route\Http\Exception\NotFoundException::class);
|
||||||
|
$testClass->get($this->getMockRequest());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testNotFoundExceptionPost()
|
||||||
|
{
|
||||||
|
$testClass = new TestClass();
|
||||||
|
|
||||||
|
$this->expectException(\League\Route\Http\Exception\NotFoundException::class);
|
||||||
|
$testClass->post($this->getMockRequest());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testNotFoundExceptionPut()
|
||||||
|
{
|
||||||
|
$testClass = new TestClass();
|
||||||
|
|
||||||
|
$this->expectException(\League\Route\Http\Exception\NotFoundException::class);
|
||||||
|
$testClass->put($this->getMockRequest());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testNotFoundExceptionDelete()
|
||||||
|
{
|
||||||
|
$testClass = new TestClass();
|
||||||
|
|
||||||
|
$this->expectException(\League\Route\Http\Exception\NotFoundException::class);
|
||||||
|
$testClass->delete($this->getMockRequest());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testNotFoundExceptionPatch()
|
||||||
|
{
|
||||||
|
$testClass = new TestClass();
|
||||||
|
|
||||||
|
$this->expectException(\League\Route\Http\Exception\NotFoundException::class);
|
||||||
|
$testClass->patch($this->getMockRequest());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class TestClass extends Controller // phpcs:ignore
|
||||||
|
{
|
||||||
|
}
|
||||||
25
tests/Controllers/IndexControllerTest.php
Normal file
25
tests/Controllers/IndexControllerTest.php
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Controllers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Controllers\IndexController;
|
||||||
|
|
||||||
|
class IndexControllerTest extends AbstractController
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @throws \JsonException
|
||||||
|
*/
|
||||||
|
public function testGet(): void
|
||||||
|
{
|
||||||
|
$this->assertTrue(true);
|
||||||
|
|
||||||
|
$controller = new IndexController();
|
||||||
|
|
||||||
|
$response = $controller->get($this->getMockRequest());
|
||||||
|
|
||||||
|
$this->assertEquals(200, $response->getStatusCode());
|
||||||
|
$this->assertEquals('{"status_code":200,"message":"Server is running"}', (string)$response->getBody());
|
||||||
|
}
|
||||||
|
}
|
||||||
32
tests/Facades/AbstractFacade.php
Normal file
32
tests/Facades/AbstractFacade.php
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Facades;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
abstract class AbstractFacade extends Unit
|
||||||
|
{
|
||||||
|
abstract protected function getFacadeClass(): string;
|
||||||
|
abstract protected function getConcrete(): string;
|
||||||
|
|
||||||
|
public function testFacadeAccessor(): void
|
||||||
|
{
|
||||||
|
/** @var Facade | string $class */
|
||||||
|
$class = $this->getFacadeClass();
|
||||||
|
|
||||||
|
$this->assertTrue(
|
||||||
|
method_exists($class, 'getFacadeAccessor'),
|
||||||
|
sprintf('The class %s must implement the method getFacadeAccessor.', $class)
|
||||||
|
);
|
||||||
|
|
||||||
|
$facade = $class::getFacadeRoot();
|
||||||
|
|
||||||
|
$this->assertNotNull(
|
||||||
|
$facade,
|
||||||
|
sprintf('The facade %s is not properly initialized.', $this->getConcrete())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
20
tests/Facades/DispatcherTest.php
Normal file
20
tests/Facades/DispatcherTest.php
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Facades;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\Facades\Dispatcher;
|
||||||
|
|
||||||
|
class DispatcherTest extends AbstractFacade
|
||||||
|
{
|
||||||
|
protected function getFacadeClass(): string
|
||||||
|
{
|
||||||
|
return Dispatcher::class;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getConcrete(): string
|
||||||
|
{
|
||||||
|
return \Siteworxpro\App\Events\Dispatcher::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
20
tests/Facades/LoggerTest.php
Normal file
20
tests/Facades/LoggerTest.php
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Facades;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\Facades\Logger;
|
||||||
|
|
||||||
|
class LoggerTest extends AbstractFacade
|
||||||
|
{
|
||||||
|
protected function getFacadeClass(): string
|
||||||
|
{
|
||||||
|
return Logger::class;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getConcrete(): string
|
||||||
|
{
|
||||||
|
return \Siteworxpro\App\Log\Logger::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
21
tests/Facades/RedisTest.php
Normal file
21
tests/Facades/RedisTest.php
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Facades;
|
||||||
|
|
||||||
|
use Predis\Client;
|
||||||
|
use Siteworxpro\App\Services\Facades\Redis;
|
||||||
|
|
||||||
|
class RedisTest extends AbstractFacade
|
||||||
|
{
|
||||||
|
protected function getFacadeClass(): string
|
||||||
|
{
|
||||||
|
return Redis::class;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getConcrete(): string
|
||||||
|
{
|
||||||
|
return Client::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
19
tests/Helpers/UlidTest.php
Normal file
19
tests/Helpers/UlidTest.php
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Helpers;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Helpers\Env;
|
||||||
|
use Siteworxpro\App\Helpers\Ulid;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
class UlidTest extends Unit
|
||||||
|
{
|
||||||
|
public function testGetString(): void
|
||||||
|
{
|
||||||
|
$ulid = Ulid::generate();
|
||||||
|
$this->assertIsString($ulid);
|
||||||
|
$this->assertEquals(16, strlen($ulid));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,29 +6,36 @@ namespace Siteworxpro\Tests\Http;
|
|||||||
|
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
use Siteworxpro\App\Http\JsonResponseFactory;
|
use Siteworxpro\App\Http\JsonResponseFactory;
|
||||||
|
use Siteworxpro\HttpStatus\CodesEnum;
|
||||||
|
|
||||||
class JsonResponseFactoryTest extends TestCase
|
class JsonResponseFactoryTest extends TestCase
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* @throws \JsonException
|
||||||
|
*/
|
||||||
public function testCreateJsonResponseReturnsValidResponse(): void
|
public function testCreateJsonResponseReturnsValidResponse(): void
|
||||||
{
|
{
|
||||||
$data = ['key' => 'value'];
|
$data = ['key' => 'value'];
|
||||||
$statusCode = 200;
|
$statusCode = CodesEnum::OK;
|
||||||
|
|
||||||
$response = JsonResponseFactory::createJsonResponse($data, $statusCode);
|
$response = JsonResponseFactory::createJsonResponse($data, $statusCode);
|
||||||
|
|
||||||
$this->assertSame($statusCode, $response->getStatusCode());
|
$this->assertSame($statusCode->value, $response->getStatusCode());
|
||||||
$this->assertSame('application/json', $response->getHeaderLine('Content-Type'));
|
$this->assertSame('application/json', $response->getHeaderLine('Content-Type'));
|
||||||
$this->assertSame(json_encode($data), (string) $response->getBody());
|
$this->assertSame(json_encode($data), (string) $response->getBody());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \JsonException
|
||||||
|
*/
|
||||||
public function testCreateJsonResponseHandlesEmptyData(): void
|
public function testCreateJsonResponseHandlesEmptyData(): void
|
||||||
{
|
{
|
||||||
$data = [];
|
$data = [];
|
||||||
$statusCode = 204;
|
$statusCode = CodesEnum::NO_CONTENT;
|
||||||
|
|
||||||
$response = JsonResponseFactory::createJsonResponse($data, $statusCode);
|
$response = JsonResponseFactory::createJsonResponse($data, $statusCode);
|
||||||
|
|
||||||
$this->assertSame($statusCode, $response->getStatusCode());
|
$this->assertSame($statusCode->value, $response->getStatusCode());
|
||||||
$this->assertSame('application/json', $response->getHeaderLine('Content-Type'));
|
$this->assertSame('application/json', $response->getHeaderLine('Content-Type'));
|
||||||
$this->assertSame(json_encode($data), (string) $response->getBody());
|
$this->assertSame(json_encode($data), (string) $response->getBody());
|
||||||
}
|
}
|
||||||
|
|||||||
167
tests/Log/LoggerRpcTest.php
Normal file
167
tests/Log/LoggerRpcTest.php
Normal file
@@ -0,0 +1,167 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Log;
|
||||||
|
|
||||||
|
use Mockery;
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Psr\Log\LogLevel;
|
||||||
|
use Siteworxpro\App\Log\Logger;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
class LoggerRpcTest extends Unit
|
||||||
|
{
|
||||||
|
protected function tearDown(): void
|
||||||
|
{
|
||||||
|
parent::tearDown();
|
||||||
|
unset($_SERVER['RR_RPC']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
* @throws \Throwable
|
||||||
|
*/
|
||||||
|
public function testLogsDebugMessageWhenLevelIsDebug(): void
|
||||||
|
{
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
|
||||||
|
$_SERVER['RR_RPC'] = 'tcp://127.0.0.1:6001';
|
||||||
|
|
||||||
|
$mock = Mockery::mock(LoggerInterface::class);
|
||||||
|
$mock->expects('debug')
|
||||||
|
->with('message', ['key' => 'value'])
|
||||||
|
->once();
|
||||||
|
|
||||||
|
\Siteworxpro\App\Services\Facades\Logger::getFacadeContainer()
|
||||||
|
->bind(\RoadRunner\Logger\Logger::class, function () use ($mock) {
|
||||||
|
return $mock;
|
||||||
|
});
|
||||||
|
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
$logger = new Logger(LogLevel::DEBUG, $inputBuffer);
|
||||||
|
$logger->debug('message', ['key' => 'value']);
|
||||||
|
|
||||||
|
$mock->shouldHaveReceived('debug');
|
||||||
|
|
||||||
|
Mockery::close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function testLogsDebugMessageWhenLevelIsInfoNotice(): void
|
||||||
|
{
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
|
||||||
|
$_SERVER['RR_RPC'] = 'tcp://127.0.0.1:6001';
|
||||||
|
|
||||||
|
$mock = Mockery::mock(LoggerInterface::class);
|
||||||
|
$mock->expects('info')
|
||||||
|
->with('message', ['key' => 'value'])
|
||||||
|
->times(2);
|
||||||
|
|
||||||
|
\Siteworxpro\App\Services\Facades\Logger::getFacadeContainer()
|
||||||
|
->bind(\RoadRunner\Logger\Logger::class, function () use ($mock) {
|
||||||
|
return $mock;
|
||||||
|
});
|
||||||
|
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
$logger = new Logger(LogLevel::DEBUG, $inputBuffer);
|
||||||
|
$logger->info('message', ['key' => 'value']);
|
||||||
|
$logger->notice('message', ['key' => 'value']);
|
||||||
|
|
||||||
|
$mock->shouldHaveReceived('info')->times(2);
|
||||||
|
Mockery::close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function testLogsDebugMessageWhenLevelIsInfoWarning(): void
|
||||||
|
{
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
|
||||||
|
$_SERVER['RR_RPC'] = 'tcp://127.0.0.1:6001';
|
||||||
|
|
||||||
|
$mock = Mockery::mock(LoggerInterface::class);
|
||||||
|
$mock->expects('warning')
|
||||||
|
->with('message', ['key' => 'value'])
|
||||||
|
->times(1);
|
||||||
|
|
||||||
|
\Siteworxpro\App\Services\Facades\Logger::getFacadeContainer()
|
||||||
|
->bind(\RoadRunner\Logger\Logger::class, function () use ($mock) {
|
||||||
|
return $mock;
|
||||||
|
});
|
||||||
|
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
$logger = new Logger(LogLevel::DEBUG, $inputBuffer);
|
||||||
|
$logger->warning('message', ['key' => 'value']);
|
||||||
|
|
||||||
|
$mock->shouldHaveReceived('warning');
|
||||||
|
Mockery::close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function testLogsDebugMessageWhenLevelIsInfoError(): void
|
||||||
|
{
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
|
||||||
|
$_SERVER['RR_RPC'] = 'tcp://127.0.0.1:6001';
|
||||||
|
|
||||||
|
$mock = Mockery::mock(LoggerInterface::class);
|
||||||
|
$mock->expects('error')
|
||||||
|
->with('message', ['key' => 'value'])
|
||||||
|
->times(4);
|
||||||
|
|
||||||
|
\Siteworxpro\App\Services\Facades\Logger::getFacadeContainer()
|
||||||
|
->bind(\RoadRunner\Logger\Logger::class, function () use ($mock) {
|
||||||
|
return $mock;
|
||||||
|
});
|
||||||
|
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
$logger = new Logger(LogLevel::DEBUG, $inputBuffer);
|
||||||
|
$logger->error('message', ['key' => 'value']);
|
||||||
|
$logger->critical('message', ['key' => 'value']);
|
||||||
|
$logger->alert('message', ['key' => 'value']);
|
||||||
|
$logger->emergency('message', ['key' => 'value']);
|
||||||
|
|
||||||
|
$mock->shouldHaveReceived('error')->times(4);
|
||||||
|
Mockery::close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function testLogsLog(): void
|
||||||
|
{
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
|
||||||
|
$_SERVER['RR_RPC'] = 'tcp://127.0.0.1:6001';
|
||||||
|
|
||||||
|
$mock = Mockery::mock(LoggerInterface::class);
|
||||||
|
$mock->expects('log')
|
||||||
|
->with('notaloglevel', 'message', ['key' => 'value']);
|
||||||
|
|
||||||
|
\Siteworxpro\App\Services\Facades\Logger::getFacadeContainer()
|
||||||
|
->bind(\RoadRunner\Logger\Logger::class, function () use ($mock) {
|
||||||
|
return $mock;
|
||||||
|
});
|
||||||
|
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
$logger = new Logger(LogLevel::DEBUG, $inputBuffer);
|
||||||
|
$logger->log('notaloglevel', 'message', ['key' => 'value']);
|
||||||
|
|
||||||
|
$mock->shouldHaveReceived('log')->times(1);
|
||||||
|
Mockery::close();
|
||||||
|
}
|
||||||
|
}
|
||||||
155
tests/Log/LoggerTest.php
Normal file
155
tests/Log/LoggerTest.php
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\Log;
|
||||||
|
|
||||||
|
use Psr\Log\LogLevel;
|
||||||
|
use Siteworxpro\App\Log\Logger;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
class LoggerTest extends Unit
|
||||||
|
{
|
||||||
|
private function getLoggerWithBuffer(string $logLevel): array
|
||||||
|
{
|
||||||
|
$inputBuffer = fopen('php://memory', 'r+');
|
||||||
|
return [new Logger($logLevel, $inputBuffer), $inputBuffer];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getContents($inputBuffer): string
|
||||||
|
{
|
||||||
|
return stream_get_contents($inputBuffer, -1, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function testLogLevel(string $level): void
|
||||||
|
{
|
||||||
|
[$logger, $inputBuffer] = $this->getLoggerWithBuffer($level);
|
||||||
|
$logger->$level('message', ['key' => 'value']);
|
||||||
|
$output = $this->getContents($inputBuffer);
|
||||||
|
|
||||||
|
$this->assertNotEmpty($output);
|
||||||
|
$decoded = json_decode($output, true);
|
||||||
|
$this->assertEquals('message', $decoded['message']);
|
||||||
|
$this->assertEquals('value', $decoded['context']['key']);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function testLogLevelEmpty(string $configLevel, string $logLevel): void
|
||||||
|
{
|
||||||
|
[$logger, $inputBuffer] = $this->getLoggerWithBuffer($configLevel);
|
||||||
|
$logger->$logLevel('message', ['key' => 'value']);
|
||||||
|
$output = $this->getContents($inputBuffer);
|
||||||
|
|
||||||
|
$this->assertEmpty($output);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsDebugMessageWhenLevelIsDebug(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::DEBUG);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsInfoMessageWhenLevelIsInfo(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::INFO);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsWarningMessageWhenLevelIsWarning(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::WARNING);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsErrorMessageWhenLevelIsError(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::ERROR);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsCriticalMessageWhenLevelIsCritical(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::CRITICAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsAlertMessageWhenLevelIsAlert(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::ALERT);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsEmergencyMessageWhenLevelIsEmergency(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::EMERGENCY);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsNoticeMessageWhenLevelIsNotice(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevel(LogLevel::NOTICE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testDoesNotLogWhenMinimumLevelIsInfo(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevelEmpty(LogLevel::INFO, LogLevel::DEBUG);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testDoesNotLogWhenMinimumLevelIsWarning(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevelEmpty(LogLevel::WARNING, LogLevel::INFO);
|
||||||
|
$this->testLogLevelEmpty(LogLevel::WARNING, LogLevel::DEBUG);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testDoesNotLogWhenMinimumLevelIsError(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevelEmpty(LogLevel::ERROR, LogLevel::DEBUG);
|
||||||
|
$this->testLogLevelEmpty(LogLevel::ERROR, LogLevel::INFO);
|
||||||
|
$this->testLogLevelEmpty(LogLevel::ERROR, LogLevel::WARNING);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testDoesNotLogWhenMinimumLevelIsNotice(): void
|
||||||
|
{
|
||||||
|
$this->testLogLevelEmpty(LogLevel::NOTICE, LogLevel::DEBUG);
|
||||||
|
$this->testLogLevelEmpty(LogLevel::NOTICE, LogLevel::INFO);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsMessageWithEmptyContext(): void
|
||||||
|
{
|
||||||
|
[$logger, $buffer] = $this->getLoggerWithBuffer(LogLevel::INFO);
|
||||||
|
|
||||||
|
$logger->info('Message without context');
|
||||||
|
$output = $this->getContents($buffer);
|
||||||
|
|
||||||
|
$this->assertNotEmpty($output);
|
||||||
|
$decoded = json_decode($output, true);
|
||||||
|
$this->assertEquals('Message without context', $decoded['message']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsMessageWithComplexContext(): void
|
||||||
|
{
|
||||||
|
[$logger, $buffer] = $this->getLoggerWithBuffer(LogLevel::INFO);
|
||||||
|
|
||||||
|
$logger->info('Complex context', [
|
||||||
|
'user_id' => 123,
|
||||||
|
'nested' => ['key' => 'value'],
|
||||||
|
'array' => [1, 2, 3]
|
||||||
|
]);
|
||||||
|
$output = $this->getContents($buffer);
|
||||||
|
|
||||||
|
$this->assertNotEmpty($output);
|
||||||
|
$decoded = json_decode($output, true);
|
||||||
|
$this->assertEquals(123, $decoded['context']['user_id']);
|
||||||
|
$this->assertEquals('value', $decoded['context']['nested']['key']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testLogsStringableMessage(): void
|
||||||
|
{
|
||||||
|
[$logger, $buffer] = $this->getLoggerWithBuffer(LogLevel::INFO);
|
||||||
|
|
||||||
|
$stringable = new class implements \Stringable {
|
||||||
|
public function __toString(): string
|
||||||
|
{
|
||||||
|
return 'Stringable message';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$logger->info($stringable);
|
||||||
|
$output = $this->getContents($buffer);
|
||||||
|
$this->assertNotEmpty($output);
|
||||||
|
$decoded = json_decode($output, true);
|
||||||
|
$this->assertEquals('Stringable message', $decoded['message']);
|
||||||
|
}
|
||||||
|
}
|
||||||
43
tests/ServiceProviders/AbstractServiceProvider.php
Normal file
43
tests/ServiceProviders/AbstractServiceProvider.php
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\ServiceProviders;
|
||||||
|
|
||||||
|
use Illuminate\Container\Container;
|
||||||
|
use Illuminate\Contracts\Container\BindingResolutionException;
|
||||||
|
use Illuminate\Support\ServiceProvider;
|
||||||
|
use Siteworxpro\Tests\Unit;
|
||||||
|
|
||||||
|
abstract class AbstractServiceProvider extends Unit
|
||||||
|
{
|
||||||
|
abstract protected function getProviderClass(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws BindingResolutionException
|
||||||
|
*/
|
||||||
|
public function testProvider(): void
|
||||||
|
{
|
||||||
|
$container = new Container();
|
||||||
|
|
||||||
|
$providerClass = $this->getProviderClass();
|
||||||
|
|
||||||
|
/** @var ServiceProvider $providerClass */
|
||||||
|
$provider = new $providerClass($container);
|
||||||
|
|
||||||
|
$this->assertInstanceOf($providerClass, $provider);
|
||||||
|
$provider->register();
|
||||||
|
|
||||||
|
$bindings = $provider->bindings;
|
||||||
|
foreach ($bindings as $abstract => $concrete) {
|
||||||
|
$this->assertTrue($container->bound($abstract), "The $abstract is not bound in the container.");
|
||||||
|
$this->assertNotNull($container->make($abstract), "The $abstract could not be resolved.");
|
||||||
|
|
||||||
|
$this->assertInstanceOf(
|
||||||
|
$concrete,
|
||||||
|
$container->make($abstract),
|
||||||
|
"The $abstract is not an instance of $concrete."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
15
tests/ServiceProviders/LoggerServiceProviderTest.php
Normal file
15
tests/ServiceProviders/LoggerServiceProviderTest.php
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\ServiceProviders;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\LoggerServiceProvider;
|
||||||
|
|
||||||
|
class LoggerServiceProviderTest extends AbstractServiceProvider
|
||||||
|
{
|
||||||
|
protected function getProviderClass(): string
|
||||||
|
{
|
||||||
|
return LoggerServiceProvider::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
15
tests/ServiceProviders/RedisServiceProviderTest.php
Normal file
15
tests/ServiceProviders/RedisServiceProviderTest.php
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Siteworxpro\Tests\ServiceProviders;
|
||||||
|
|
||||||
|
use Siteworxpro\App\Services\ServiceProviders\RedisServiceProvider;
|
||||||
|
|
||||||
|
class RedisServiceProviderTest extends AbstractServiceProvider
|
||||||
|
{
|
||||||
|
protected function getProviderClass(): string
|
||||||
|
{
|
||||||
|
return RedisServiceProvider::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,21 +5,29 @@ declare(strict_types=1);
|
|||||||
namespace Siteworxpro\Tests;
|
namespace Siteworxpro\Tests;
|
||||||
|
|
||||||
use Illuminate\Container\Container;
|
use Illuminate\Container\Container;
|
||||||
use Illuminate\Support\Facades\Facade;
|
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Siteworx\Config\Config as SWConfig;
|
||||||
|
use Siteworxpro\App\Services\Facade;
|
||||||
use Siteworxpro\App\Services\Facades\Config;
|
use Siteworxpro\App\Services\Facades\Config;
|
||||||
|
|
||||||
abstract class Unit extends TestCase
|
abstract class Unit extends TestCase
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* @throws \ReflectionException
|
||||||
|
*/
|
||||||
protected function setUp(): void
|
protected function setUp(): void
|
||||||
{
|
{
|
||||||
$container = new Container();
|
$container = new Container();
|
||||||
Facade::setFacadeApplication($container);
|
Facade::setFacadeContainer($container);
|
||||||
|
|
||||||
|
$container->bind(SWConfig::class, function () {
|
||||||
|
return SWConfig::load(__DIR__ . '/../config.php');
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function tearDown(): void
|
protected function tearDown(): void
|
||||||
{
|
{
|
||||||
Config::clearResolvedInstances();
|
Config::clearResolvedInstances();
|
||||||
Facade::setFacadeApplication(null);
|
Facade::setFacadeContainer(null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user