From ef9f8500acfa58446c6740a69af873c8287232a5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fr=C3=A9d=C3=A9ric=20Cilia?= Date: Sun, 7 Jan 2024 15:04:23 +0100 Subject: [PATCH 1/2] [APP] Invoice Ninja --- apps/invoice-ninja/config.json | 46 +++++ apps/invoice-ninja/data/init/init.sh | 4 + .../data/nginx/invoice-ninja.conf | 36 ++++ apps/invoice-ninja/data/php/php-cli.ini | 18 ++ apps/invoice-ninja/data/php/php.ini | 21 +++ apps/invoice-ninja/docker-compose.yml | 102 +++++++++++ apps/invoice-ninja/metadata/description.md | 160 ++++++++++++++++++ apps/invoice-ninja/metadata/logo.jpg | Bin 0 -> 23254 bytes 8 files changed, 387 insertions(+) create mode 100644 apps/invoice-ninja/config.json create mode 100644 apps/invoice-ninja/data/init/init.sh create mode 100644 apps/invoice-ninja/data/nginx/invoice-ninja.conf create mode 100644 apps/invoice-ninja/data/php/php-cli.ini create mode 100644 apps/invoice-ninja/data/php/php.ini create mode 100644 apps/invoice-ninja/docker-compose.yml create mode 100644 apps/invoice-ninja/metadata/description.md create mode 100644 apps/invoice-ninja/metadata/logo.jpg diff --git a/apps/invoice-ninja/config.json b/apps/invoice-ninja/config.json new file mode 100644 index 00000000..c5ae910c --- /dev/null +++ b/apps/invoice-ninja/config.json @@ -0,0 +1,46 @@ +{ + "$schema": "../schema.json", + "name": "Invoice Ninja", + "port": 8881, + "available": true, + "exposable": true, + "id": "invoice-ninja", + "tipi_version": 1, + "version": "1.25", + "categories": [ + "finance" + ], + "description": "Invoice Ninja is an invoicing application which makes sending invoices and receiving payments simple and easy. Our latest version is a clean slate rewrite of our popular invoicing application which builds on the existing feature set and adds a wide range of features and enhancements the community has asked for.", + "short_desc": "Invoices, Expenses and Tasks built with Laravel, Flutter and React.", + "author": "https://www.invoiceninja.org/", + "source": "https://github.com/invoiceninja/invoiceninja", + "form_fields": [ + { + "type": "email", + "label": "Invoice Ninja User Mail", + "required": true, + "env_variable": "INVOICE_NINJA_USER_MAIL" + }, + { + "type": "password", + "label": "Invoice Ninja User Password", + "required": true, + "env_variable": "INVOICE_NINJA_USER_PASSWORD" + }, + { + "type": "random", + "label": "MySQL Password", + "min": 32, + "env_variable": "INVOICE_NINJA_BDD_PASSWORD" + }, + { + "type": "text", + "label": "Invoice Ninja Application Key (Must be generated with the following command `docker run --rm -it invoiceninja/invoiceninja php artisan key:generate --show`)", + "required": true, + "env_variable": "INVOICE_NINJA_APP_KEY" + } + ], + "supported_architectures": [ + "amd64" + ] +} diff --git a/apps/invoice-ninja/data/init/init.sh b/apps/invoice-ninja/data/init/init.sh new file mode 100644 index 00000000..6812ba44 --- /dev/null +++ b/apps/invoice-ninja/data/init/init.sh @@ -0,0 +1,4 @@ +#!/usr/bin/env bash +mkdir -p /tmp/data/storage /tmp/data/public +chmod 755 /tmp/data/public +chown -R 1500:1500 /tmp/data/php /tmp/data/public /tmp/data/storage diff --git a/apps/invoice-ninja/data/nginx/invoice-ninja.conf b/apps/invoice-ninja/data/nginx/invoice-ninja.conf new file mode 100644 index 00000000..d464ed55 --- /dev/null +++ b/apps/invoice-ninja/data/nginx/invoice-ninja.conf @@ -0,0 +1,36 @@ +server { + listen 80 default_server; + + server_tokens off; + + client_max_body_size 100M; + + root /var/www/app/public/; + index index.php; + + location / { + try_files $uri $uri/ /index.php?$query_string; + } + + location /healthcheck { + return 200; + } + + location = /favicon.ico { access_log off; log_not_found off; } + location = /robots.txt { access_log off; log_not_found off; } + + location ~* /storage/.*\.php$ { + return 503; + } + + location ~ \.php$ { + fastcgi_split_path_info ^(.+\.php)(/.+)$; + fastcgi_pass invoice-ninja-server:9000; + fastcgi_index index.php; + include fastcgi_params; + fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; + fastcgi_intercept_errors off; + fastcgi_buffer_size 16k; + fastcgi_buffers 4 16k; + } +} diff --git a/apps/invoice-ninja/data/php/php-cli.ini b/apps/invoice-ninja/data/php/php-cli.ini new file mode 100644 index 00000000..438b8392 --- /dev/null +++ b/apps/invoice-ninja/data/php/php-cli.ini @@ -0,0 +1,18 @@ +session.auto_start = Off +short_open_tag = Off + +error_reporting = E_ALL & ~E_NOTICE & ~E_WARNING & ~E_STRICT & ~E_DEPRECATED + +; opcache.enable_cli=1 +; opcache.fast_shutdown=1 +; opcache.memory_consumption=256 +; opcache.interned_strings_buffer=8 +; opcache.max_accelerated_files=4000 +; opcache.revalidate_freq=60 +; # http://symfony.com/doc/current/performance.html +; realpath_cache_size = 4096K +; realpath_cache_ttl = 600 + +memory_limit = 2G +post_max_size = 60M +upload_max_filesize = 50M diff --git a/apps/invoice-ninja/data/php/php.ini b/apps/invoice-ninja/data/php/php.ini new file mode 100644 index 00000000..588baeb8 --- /dev/null +++ b/apps/invoice-ninja/data/php/php.ini @@ -0,0 +1,21 @@ +session.auto_start = Off +short_open_tag = Off + +error_reporting = E_ALL & ~E_NOTICE & ~E_WARNING & ~E_STRICT & ~E_DEPRECATED + +; opcache.enable=1 +; opcache.preload=/srv/www/invoiceninja/current/preload.php +; opcache.preload_user=www-data + +; ; The OPcache shared memory storage size. +; opcache.max_accelerated_files=300000 +; opcache.validate_timestamps=1 +; opcache.revalidate_freq=30 +; opcache.jit_buffer_size=256M +; opcache.jit=1205 +; opcache.memory_consumption=1024M + + +post_max_size = 60M +upload_max_filesize = 50M +memory_limit=512M diff --git a/apps/invoice-ninja/docker-compose.yml b/apps/invoice-ninja/docker-compose.yml new file mode 100644 index 00000000..d8d558a5 --- /dev/null +++ b/apps/invoice-ninja/docker-compose.yml @@ -0,0 +1,102 @@ +version: "3.9" + +services: + invoice-ninja: + image: nginx:1.25 + container_name: invoice-ninja + restart: unless-stopped + volumes: + - ${APP_DATA_DIR}/data/nginx/invoice-ninja.conf:/etc/nginx/conf.d/default.conf:ro + - ${APP_DATA_DIR}/data/public:/var/www/app/public:ro + depends_on: + invoice-ninja-server: + condition: service_started + ports: + - ${APP_PORT}:80 + networks: + - tipi_main_network + labels: + # Main + traefik.enable: true + traefik.http.middlewares.invoice-ninja-web-redirect.redirectscheme.scheme: https + traefik.http.services.invoice-ninja.loadbalancer.server.port: 80 + # Web + traefik.http.routers.invoice-ninja-insecure.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.invoice-ninja-insecure.entrypoints: web + traefik.http.routers.invoice-ninja-insecure.service: invoice-ninja + traefik.http.routers.invoice-ninja-insecure.middlewares: invoice-ninja-web-redirect + # Websecure + traefik.http.routers.invoice-ninja.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.invoice-ninja.entrypoints: websecure + traefik.http.routers.invoice-ninja.service: invoice-ninja + traefik.http.routers.invoice-ninja.tls.certresolver: myresolver + # Local domain + traefik.http.routers.invoice-ninja-local-insecure.rule: Host(`invoice-ninja.${LOCAL_DOMAIN}`) + traefik.http.routers.invoice-ninja-local-insecure.entrypoints: web + traefik.http.routers.invoice-ninja-local-insecure.service: invoice-ninja + traefik.http.routers.invoice-ninja-local-insecure.middlewares: invoice-ninja-web-redirect + # Local domain secure + traefik.http.routers.invoice-ninja-local.rule: Host(`invoice-ninja.${LOCAL_DOMAIN}`) + traefik.http.routers.invoice-ninja-local.entrypoints: websecure + traefik.http.routers.invoice-ninja-local.service: invoice-ninja + traefik.http.routers.invoice-ninja-local.tls: true + + invoice-ninja-server: + image: invoiceninja/invoiceninja:5.8.2 + container_name: invoice-ninja-server + restart: unless-stopped + user: 1500:1500 + environment: + - IN_USER_EMAIL=${INVOICE_NINJA_USER_MAIL} + - IN_PASSWORD=${INVOICE_NINJA_USER_PASSWORD} + - APP_URL=http://invoice-ninja + - APP_KEY=${INVOICE_NINJA_APP_KEY} + - APP_CIPHER=AES-256-CBC + - DB_HOST=invoice-ninja-db + - DB_PORT=3306 + - DB_DATABASE=ninja + - DB_USERNAME=ninja + - DB_PASSWORD=ninja + - REQUIRE_HTTPS=false + - QUEUE_CONNECTION=database + - IS_DOCKER=TRUE + volumes: + - ${APP_DATA_DIR}/data/public:/var/www/app/public:rw,delegated + - ${APP_DATA_DIR}/data/storage:/var/www/app/storage:rw,delegated + - ${APP_DATA_DIR}/data/php/php.ini:/usr/local/etc/php/php.ini:ro + - ${APP_DATA_DIR}/data/php/php-cli.ini:/usr/local/etc/php/php-cli.ini:ro + depends_on: + invoice-ninja-db: + condition: service_healthy + networks: + - tipi_main_network + + invoice-ninja-db: + image: mariadb:10.4 + container_name: invoice-ninja-db + restart: unless-stopped + environment: + - MARIADB_ROOT_PASSWORD=${INVOICE_NINJA_BDD_PASSWORD} + - MARIADB_USER=ninja + - MARIADB_PASSWORD=ninja + - MARIADB_DATABASE=ninja + volumes: + - ${APP_DATA_DIR}/data/mysql:/var/lib/mysql:rw,delegated + networks: + - tipi_main_network + depends_on: + invoice-ninja-init: + condition: service_completed_successfully + healthcheck: + test: [ "CMD", "healthcheck.sh", "--su-mysql", "--connect", "--innodb_initialized" ] + interval: 10s + timeout: 5s + retries: 5 + start_period: 10s + + invoice-ninja-init: + image: bash:5.2.21 + container_name: invoice-ninja-init + volumes: + - ${APP_DATA_DIR}/data:/tmp/data + command: bash -c "chmod +x /tmp/data/init/init.sh && /tmp/data/init/init.sh " diff --git a/apps/invoice-ninja/metadata/description.md b/apps/invoice-ninja/metadata/description.md new file mode 100644 index 00000000..f22a5dba --- /dev/null +++ b/apps/invoice-ninja/metadata/description.md @@ -0,0 +1,160 @@ +

+ Sublime's custom image +

+ +![v5-develop phpunit](https://github.com/invoiceninja/invoiceninja/workflows/phpunit/badge.svg?branch=v5-develop) +[![Codacy Badge](https://app.codacy.com/project/badge/Grade/d16c78aad8574466bf83232b513ef4fb)](https://www.codacy.com/gh/turbo124/invoiceninja/dashboard?utm_source=github.com&utm_medium=referral&utm_content=turbo124/invoiceninja&utm_campaign=Badge_Grade) +CLA assistant + +# Invoice Ninja 5 + +## [Hosted](https://www.invoiceninja.com) | [Self-Hosted](https://www.invoiceninja.org) + +Join us on [Slack](http://slack.invoiceninja.com), [Discord](https://discord.gg/ZwEdtfCwXA), [Support Forum](https://forum.invoiceninja.com) + +## Introduction + +Version 5 of Invoice Ninja is here! +We took the best parts of version 4 and add the most requested features +to produce a invoicing application like no other. + +All Pro and Enterprise features from the hosted app are included in the open code. +We offer a $30 per year white-label license to remove the Invoice Ninja branding from client facing parts of the app. + +* [Videos](https://www.youtube.com/@appinvoiceninja) +* [API Documentation](https://api-docs.invoicing.co/) +* [APP Documentation](https://invoiceninja.github.io/) +* [Support Forum](https://forum.invoiceninja.com) + +## Setup + +### Mobile Apps +* [iPhone](https://apps.apple.com/app/id1503970375?platform=iphone) +* [Android](https://play.google.com/store/apps/details?id=com.invoiceninja.app) +* [F-Droid](https://f-droid.org/en/packages/com.invoiceninja.app) + +### Desktop Apps +* [macOS](https://apps.apple.com/app/id1503970375?platform=mac) +* [Windows](https://microsoft.com/en-us/p/invoice-ninja/9n3f2bbcfdr6) +* [Linux](https://snapcraft.io/invoiceninja) + +### Installation Options +* [Docker File](https://hub.docker.com/r/invoiceninja/invoiceninja/) +* [Cloudron](https://cloudron.io/store/com.invoiceninja.cloudronapp.html) +* [Softaculous](https://www.softaculous.com/apps/ecommerce/Invoice_Ninja) + +### Recommended Providers +* [Stripe](https://stripe.com/) +* [Postmark](https://postmarkapp.com/) + +## Quick Hosting Setup + +```sh +git clone --single-branch --branch v5-stable https://github.com/invoiceninja/invoiceninja.git +cp .env.example .env +composer i -o --no-dev +php artisan key:generate +``` + +Please Note: +Your APP_KEY in the .env file is used to encrypt data, if you lose this you will not be able to run the application. + +Run if you want to load sample data, remember to configure .env +```sh +php artisan migrate:fresh --seed && php artisan db:seed && php artisan ninja:create-test-data +``` + +To run the web server +```sh +php artisan serve +``` + +Navigate to (replace localhost with the appropriate domain) +``` +http://localhost:8000/setup - To setup your configuration if you did not load sample data. +http://localhost:8000/ - For Administrator Logon + +user: small@example.com +pass: password + +http://localhost:8000/client/login - For Client Portal + +user: user@example.com +pass: password +``` +## Developers Guide + + +### App Design + +The API and client portal have been developed using [Laravel](https://laravel.com) if you wish to contribute to this project familiarity with Laravel is essential. + +When inspecting functionality of the API, the best place to start would be in the routes/api.php file which describes all of the availabe API endpoints. The controller methods then describe all the entry points into each domain of the application, ie InvoiceController / QuoteController + +The average API request follows this path into the application. + +* Middleware processes the request initially inspecting the domain being requested + provides the authentication layer. +* The request then passes into a Form Request (Type hinted in the controller methods) which is used to provide authorization and also validation of the request. If successful, the request is then passed into the controller method where it is digested, here is an example: + +```php +public function store(StoreInvoiceRequest $request) +{ + + $invoice = $this->invoice_repo->save($request->all(), InvoiceFactory::create(auth()->user()->company()->id, auth()->user()->id)); + + $invoice = $invoice->service() + ->fillDefaults() + ->triggeredActions($request) + ->adjustInventory() + ->save(); + + event(new InvoiceWasCreated($invoice, $invoice->company, Ninja::eventVars(auth()->user() ? auth()->user()->id : null))); + + return $this->itemResponse($invoice); + +} +``` + +Here for example we are storing a new invoice, we pass the validated request along with a factory into the invoice repository where it is processed and saved. + +The returned invoice then passes through its service class (app/Services/Invoice) where various actions are performed. + +A event is then fired which notifies listeners in the application (app/Providers/EventServiceProvider) which perform non blocking sub tasks + +Finally the invoice is transformed (app/Transformers/) and returned as a response via Fractal. + +### Developer environment + +Using the Quick Hosting Setup describe above you can quickly get started building out your development environment. Instead of using + +``` +composer i -o --no-dev +``` + +use + +``` +composer i -o +``` + +This provides the developer tools including phpunit which allows the test suite to be run. + +If you are considering contributing back to the main repository, please add in any tests for new functionality / modifications. This will greatly increase the chances of your PR being accepted + +Also, if you plan any additions for the main repository, you may want to discuss this with us first on Slack where we can assist with any technical information and provide advice. + +## Credits +* [Hillel Coren](https://hillelcoren.com/) +* [David Bomba](https://github.com/turbo124) +* [Benjamin Beganović](https://github.com/beganovich) +* [All Contributors](https://github.com/invoiceninja/invoiceninja/graphs/contributors) + +## Security + +If you find a security issue with this application, please send an email to contact@invoiceninja.com. +Please follow responsible disclosure procedures if you detect an issue. +For further information on responsible disclosure please read [here](https://cheatsheetseries.owasp.org/cheatsheets/Vulnerability_Disclosure_Cheat_Sheet.html). + +## License +Invoice Ninja is released under the Elastic License. +See [LICENSE](LICENSE) for details. diff --git a/apps/invoice-ninja/metadata/logo.jpg b/apps/invoice-ninja/metadata/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..d16d5841fd786ab23acee1895f0896e831f6d3ad GIT binary patch literal 23254 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�_$Co@-^@`9`uF_F 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 Date: Wed, 10 Jan 2024 17:38:03 +0100 Subject: [PATCH 2/2] [APP] Invoice Ninja - Add installation notice in description.md --- apps/invoice-ninja/config.json | 2 +- apps/invoice-ninja/metadata/description.md | 10 ++++++++++ 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/apps/invoice-ninja/config.json b/apps/invoice-ninja/config.json index c5ae910c..34a20732 100644 --- a/apps/invoice-ninja/config.json +++ b/apps/invoice-ninja/config.json @@ -35,7 +35,7 @@ }, { "type": "text", - "label": "Invoice Ninja Application Key (Must be generated with the following command `docker run --rm -it invoiceninja/invoiceninja php artisan key:generate --show`)", + "label": "Invoice Ninja Application Key (cf. Installation Notice in Readme)", "required": true, "env_variable": "INVOICE_NINJA_APP_KEY" } diff --git a/apps/invoice-ninja/metadata/description.md b/apps/invoice-ninja/metadata/description.md index f22a5dba..991a555c 100644 --- a/apps/invoice-ninja/metadata/description.md +++ b/apps/invoice-ninja/metadata/description.md @@ -1,3 +1,13 @@ +## Installation Notice + +## Initial Setup +1. Generate an API Key using the following command: + ``` + docker run --rm -it invoiceninja/invoiceninja php artisan key:generate --show + ``` +2. Paste the generated key to `Invoice Ninja Application Key` in the Tipi UI. + +---

Sublime's custom image