mirror of
				https://github.com/crater-invoice/crater.git
				synced 2025-10-30 13:11:08 -04:00 
			
		
		
		
	Compare commits
	
		
			2 Commits
		
	
	
		
			fix-pdf-ur
			...
			report-pdf
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 0c821b8ae9 | |||
| bbddd88573 | 
							
								
								
									
										161
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										161
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,161 +0,0 @@ | |||||||
| name: Build PR Image |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|     types: [opened,synchronize,reopened,closed] |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|  |  | ||||||
|   build-application: |  | ||||||
|     name: Build and Push `application` |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }} |  | ||||||
|     outputs: |  | ||||||
|       tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout git repo |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - name: Generate UUID image name |  | ||||||
|         id: uuid |  | ||||||
|         run: echo "UUID_TAG_APP=$(uuidgen)" >> $GITHUB_ENV |  | ||||||
|       - name: Docker metadata |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v3 |  | ||||||
|         with: |  | ||||||
|           images: registry.uffizzi.com/${{ env.UUID_TAG_APP }} |  | ||||||
|           tags: type=raw,value=60d |  | ||||||
|       - name: Build and Push Image to registry.uffizzi.com ephemeral registry |  | ||||||
|         uses: docker/build-push-action@v2 |  | ||||||
|         with: |  | ||||||
|           push: true |  | ||||||
|           context: ./ |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|           file: ./uffizzi/Dockerfile |  | ||||||
|  |  | ||||||
|   build-nginx: |  | ||||||
|     needs:  |  | ||||||
|       - build-application |  | ||||||
|     name: Build and Push `nginx` |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }} |  | ||||||
|     outputs: |  | ||||||
|       tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout git repo |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2         |  | ||||||
|       - name: Generate UUID image name |  | ||||||
|         id: uuid |  | ||||||
|         run: echo "UUID_TAG_NGINX=$(uuidgen)" >> $GITHUB_ENV |  | ||||||
|       - name: Docker metadata |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v3 |  | ||||||
|         with: |  | ||||||
|           images: registry.uffizzi.com/${{ env.UUID_TAG_NGINX }} |  | ||||||
|           tags: type=raw,value=60d |  | ||||||
|       - name: Build and Push Image to Uffizzi ephemeral registry |  | ||||||
|         uses: docker/build-push-action@v2 |  | ||||||
|         with: |  | ||||||
|           push: true |  | ||||||
|           context: ./ |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|           file: ./uffizzi/nginx/Dockerfile |  | ||||||
|           build-args: | |  | ||||||
|             BASE_IMAGE=${{ needs.build-application.outputs.tags }} |  | ||||||
|           cache-from: type=gha |  | ||||||
|           cache-to: type=gha,mode=max |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   build-crond: |  | ||||||
|     name: Build and Push `crond` |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }} |  | ||||||
|     outputs: |  | ||||||
|       tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout git repo |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2         |  | ||||||
|       - name: Generate UUID image name |  | ||||||
|         id: uuid |  | ||||||
|         run: echo "UUID_TAG_CROND=$(uuidgen)" >> $GITHUB_ENV |  | ||||||
|       - name: Docker metadata |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v3 |  | ||||||
|         with: |  | ||||||
|           images: registry.uffizzi.com/${{ env.UUID_TAG_CROND }} |  | ||||||
|           tags: type=raw,value=60d |  | ||||||
|       - name: Build and Push Image to registry.uffizzi.com ephemeral registry |  | ||||||
|         uses: docker/build-push-action@v2 |  | ||||||
|         with: |  | ||||||
|           push: true |  | ||||||
|           context: ./ |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|           file: ./uffizzi/crond/Dockerfile       |  | ||||||
|           cache-from: type=gha |  | ||||||
|           cache-to: type=gha,mode=max |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   render-compose-file: |  | ||||||
|     name: Render Docker Compose File |  | ||||||
|     # Pass output of this workflow to another triggered by `workflow_run` event. |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     outputs: |  | ||||||
|       compose-file-cache-key: ${{ steps.hash.outputs.hash }} |  | ||||||
|     needs:  |  | ||||||
|       - build-application |  | ||||||
|       - build-nginx |  | ||||||
|       - build-crond |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout git repo |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - name: Render Compose File |  | ||||||
|         run: | |  | ||||||
|           APP_IMAGE=$(echo ${{ needs.build-application.outputs.tags }}) |  | ||||||
|           export APP_IMAGE |  | ||||||
|           NGINX_IMAGE=$(echo ${{ needs.build-nginx.outputs.tags }}) |  | ||||||
|           export NGINX_IMAGE |  | ||||||
|           CROND_IMAGE=$(echo ${{ needs.build-crond.outputs.tags }}) |  | ||||||
|           export CROND_IMAGE |  | ||||||
|           # Render simple template from environment variables. |  | ||||||
|           envsubst < ./uffizzi/docker-compose.uffizzi.yml > docker-compose.rendered.yml |  | ||||||
|           cat docker-compose.rendered.yml |  | ||||||
|       - name: Upload Rendered Compose File as Artifact |  | ||||||
|         uses: actions/upload-artifact@v3 |  | ||||||
|         with: |  | ||||||
|           name: preview-spec |  | ||||||
|           path: docker-compose.rendered.yml |  | ||||||
|           retention-days: 2 |  | ||||||
|       - name: Serialize PR Event to File |  | ||||||
|         run:  | |  | ||||||
|           cat << EOF > event.json |  | ||||||
|           ${{ toJSON(github.event) }}  |  | ||||||
|            |  | ||||||
|           EOF |  | ||||||
|       - name: Upload PR Event as Artifact |  | ||||||
|         uses: actions/upload-artifact@v3 |  | ||||||
|         with: |  | ||||||
|           name: preview-spec |  | ||||||
|           path: event.json |  | ||||||
|           retention-days: 2 |  | ||||||
|  |  | ||||||
|   delete-preview: |  | ||||||
|     name: Call for Preview Deletion |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     if: ${{ github.event.action == 'closed' }} |  | ||||||
|     steps: |  | ||||||
|       # If this PR is closing, we will not render a compose file nor pass it to the next workflow. |  | ||||||
|       - name: Serialize PR Event to File |  | ||||||
|         run: echo '${{ toJSON(github.event) }}' > event.json |  | ||||||
|       - name: Upload PR Event as Artifact |  | ||||||
|         uses: actions/upload-artifact@v3 |  | ||||||
|         with: |  | ||||||
|           name: preview-spec |  | ||||||
|           path: event.json |  | ||||||
|           retention-days: 2 |  | ||||||
|  |  | ||||||
							
								
								
									
										84
									
								
								.github/workflows/uffizzi-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										84
									
								
								.github/workflows/uffizzi-preview.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,84 +0,0 @@ | |||||||
| name: Deploy Uffizzi Preview |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_run: |  | ||||||
|     workflows: |  | ||||||
|       - "Build PR Image" |  | ||||||
|     types: |  | ||||||
|       - completed |  | ||||||
|  |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   cache-compose-file: |  | ||||||
|     name: Cache Compose File |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     outputs: |  | ||||||
|       compose-file-cache-key: ${{ env.COMPOSE_FILE_HASH }} |  | ||||||
|       pr-number: ${{ env.PR_NUMBER }} |  | ||||||
|     steps: |  | ||||||
|       - name: 'Download artifacts' |  | ||||||
|         # Fetch output (zip archive) from the workflow run that triggered this workflow. |  | ||||||
|         uses: actions/github-script@v6 |  | ||||||
|         with: |  | ||||||
|           script: | |  | ||||||
|             let allArtifacts = await github.rest.actions.listWorkflowRunArtifacts({ |  | ||||||
|                owner: context.repo.owner, |  | ||||||
|                repo: context.repo.repo, |  | ||||||
|                run_id: context.payload.workflow_run.id, |  | ||||||
|             }); |  | ||||||
|             let matchArtifact = allArtifacts.data.artifacts.filter((artifact) => { |  | ||||||
|               return artifact.name == "preview-spec" |  | ||||||
|             })[0]; |  | ||||||
|             let download = await github.rest.actions.downloadArtifact({ |  | ||||||
|                owner: context.repo.owner, |  | ||||||
|                repo: context.repo.repo, |  | ||||||
|                artifact_id: matchArtifact.id, |  | ||||||
|                archive_format: 'zip', |  | ||||||
|             }); |  | ||||||
|             let fs = require('fs'); |  | ||||||
|             fs.writeFileSync(`${process.env.GITHUB_WORKSPACE}/preview-spec.zip`, Buffer.from(download.data)); |  | ||||||
|       - name: 'Unzip artifact' |  | ||||||
|         run: unzip preview-spec.zip |  | ||||||
|       - name: Read Event into ENV |  | ||||||
|         run: | |  | ||||||
|           echo 'EVENT_JSON<<EOF' >> $GITHUB_ENV |  | ||||||
|           cat event.json >> $GITHUB_ENV |  | ||||||
|           echo 'EOF' >> $GITHUB_ENV |  | ||||||
|       - name: Hash Rendered Compose File |  | ||||||
|         id: hash |  | ||||||
|         # If the previous workflow was triggered by a PR close event, we will not have a compose file artifact. |  | ||||||
|         if: ${{ fromJSON(env.EVENT_JSON).action != 'closed' }} |  | ||||||
|         run: echo "COMPOSE_FILE_HASH=$(md5sum docker-compose.rendered.yml | awk '{ print $1 }')" >> $GITHUB_ENV |  | ||||||
|       - name: Cache Rendered Compose File |  | ||||||
|         if: ${{ fromJSON(env.EVENT_JSON).action != 'closed' }} |  | ||||||
|         uses: actions/cache@v3 |  | ||||||
|         with: |  | ||||||
|           path: docker-compose.rendered.yml |  | ||||||
|           key: ${{ env.COMPOSE_FILE_HASH }} |  | ||||||
|  |  | ||||||
|       - name: Read PR Number From Event Object |  | ||||||
|         id: pr |  | ||||||
|         run: echo "PR_NUMBER=${{ fromJSON(env.EVENT_JSON).number }}" >> $GITHUB_ENV |  | ||||||
|  |  | ||||||
|       - name: DEBUG - Print Job Outputs |  | ||||||
|         if: ${{ runner.debug }} |  | ||||||
|         run: | |  | ||||||
|           echo "PR number: ${{ env.PR_NUMBER }}" |  | ||||||
|           echo "Compose file hash: ${{ env.COMPOSE_FILE_HASH }}" |  | ||||||
|           cat event.json |  | ||||||
|   deploy-uffizzi-preview: |  | ||||||
|     name: Use Remote Workflow to Preview on Uffizzi |  | ||||||
|     needs: |  | ||||||
|       - cache-compose-file |  | ||||||
|     uses: UffizziCloud/preview-action/.github/workflows/reusable.yaml@v2.6.1 |  | ||||||
|     with: |  | ||||||
|       # If this workflow was triggered by a PR close event, cache-key will be an empty string |  | ||||||
|       # and this reusable workflow will delete the preview deployment. |  | ||||||
|       compose-file-cache-key: ${{ needs.cache-compose-file.outputs.compose-file-cache-key }} |  | ||||||
|       compose-file-cache-path: docker-compose.rendered.yml |  | ||||||
|       server: https://app.uffizzi.com/ |  | ||||||
|       pr-number: ${{ needs.cache-compose-file.outputs.pr-number }} |  | ||||||
|     permissions: |  | ||||||
|       contents: read |  | ||||||
|       pull-requests: write |  | ||||||
|       id-token: write |  | ||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -16,5 +16,4 @@ Homestead.yaml | |||||||
| .gitkeep | .gitkeep | ||||||
| /public/docs | /public/docs | ||||||
| /.scribe | /.scribe | ||||||
| !storage/fonts/.gitkeep | !storage/fonts/.gitkeep | ||||||
| .DS_Store |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| FROM php:8.1-fpm | FROM php:7.4-fpm | ||||||
|  |  | ||||||
| # Arguments defined in docker-compose.yml | # Arguments defined in docker-compose.yml | ||||||
| ARG user | ARG user | ||||||
|  | |||||||
| @ -55,7 +55,7 @@ class CreateTemplateCommand extends Command | |||||||
|         copy(public_path("/build/img/PDF/{$type}1.png"), public_path("/build/img/PDF/{$templateName}.png")); |         copy(public_path("/build/img/PDF/{$type}1.png"), public_path("/build/img/PDF/{$templateName}.png")); | ||||||
|         copy(resource_path("/static/img/PDF/{$type}1.png"), resource_path("/static/img/PDF/{$templateName}.png")); |         copy(resource_path("/static/img/PDF/{$type}1.png"), resource_path("/static/img/PDF/{$templateName}.png")); | ||||||
|  |  | ||||||
|         $path = resource_path("views/app/pdf/{$type}/{$templateName}.blade.php"); |         $path = resource_path("app/pdf/{$type}/{$templateName}.blade.php"); | ||||||
|         $type = ucfirst($type); |         $type = ucfirst($type); | ||||||
|         $this->info("{$type} Template created successfully at ".$path); |         $this->info("{$type} Template created successfully at ".$path); | ||||||
|  |  | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ | |||||||
|     "barryvdh/laravel-ide-helper": "^2.6", |     "barryvdh/laravel-ide-helper": "^2.6", | ||||||
|     "beyondcode/laravel-dump-server": "^1.0", |     "beyondcode/laravel-dump-server": "^1.0", | ||||||
|     "facade/ignition": "^2.3.6", |     "facade/ignition": "^2.3.6", | ||||||
|     "friendsofphp/php-cs-fixer": "^3.8", |     "friendsofphp/php-cs-fixer": "^3.0", | ||||||
|     "fakerphp/faker": "^1.9.1", |     "fakerphp/faker": "^1.9.1", | ||||||
|     "mockery/mockery": "^1.3.1", |     "mockery/mockery": "^1.3.1", | ||||||
|     "nunomaduro/collision": "^5.0", |     "nunomaduro/collision": "^5.0", | ||||||
| @ -81,10 +81,7 @@ | |||||||
|   "config": { |   "config": { | ||||||
|     "optimize-autoloader": true, |     "optimize-autoloader": true, | ||||||
|     "preferred-install": "dist", |     "preferred-install": "dist", | ||||||
|     "sort-packages": true, |     "sort-packages": true | ||||||
|     "allow-plugins": { |  | ||||||
|       "pestphp/pest-plugin": true |  | ||||||
|     } |  | ||||||
|   }, |   }, | ||||||
|   "extra": { |   "extra": { | ||||||
|     "laravel": { |     "laravel": { | ||||||
|  | |||||||
							
								
								
									
										2345
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2345
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -46,32 +46,32 @@ return [ | |||||||
|         Invoice::class => [ |         Invoice::class => [ | ||||||
|             'salt' => Invoice::class.config('app.key'), |             'salt' => Invoice::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 'XKAR7m8jD2bqP9OSVeNGiYL465T10zhfWuc3', |             'alphabet' => 'XKyIAR7mgt8jD2vbqPrOSVenNGpiYLx4M61T', | ||||||
|         ], |         ], | ||||||
|         Estimate::class => [ |         Estimate::class => [ | ||||||
|             'salt' => Estimate::class.config('app.key'), |             'salt' => Estimate::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 'yJW2P79M8rCHsVq5zbn1fXl6IUt3dAekGo40', |             'alphabet' => 'yLJWP79M8rYVqbn1NXjulO6IUDdvekRQGo40', | ||||||
|         ], |         ], | ||||||
|         Payment::class => [ |         Payment::class => [ | ||||||
|             'salt' => Payment::class.config('app.key'), |             'salt' => Payment::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 'aqW3eR2Icf0jp65Gl7UVS1dhyb8Mn9XKTZ4O', |             'alphabet' => 'asqtW3eDRIxB65GYl7UVLS1dybn9XrKTZ4zO', | ||||||
|         ], |         ], | ||||||
|         Company::class => [ |         Company::class => [ | ||||||
|             'salt' => Company::class.config('app.key'), |             'salt' => Company::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 's0D7xOFYEqn2uKJm3Pr9g8Cz46A1iHLBTVW5', |             'alphabet' => 's0DxOFtEYEnuKPmP08Ch6A1iHlLmBTBVWms5', | ||||||
|         ], |         ], | ||||||
|         EmailLog::class => [ |         EmailLog::class => [ | ||||||
|             'salt' => EmailLog::class.config('app.key'), |             'salt' => EmailLog::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 'BA5tJUVNPe93fCq6DHlY2x4ZO1Kg7i8wSm0R', |             'alphabet' => 'BRAMEz5str5UVe9oCqzoYY2oKgUi8wQQSmrR', | ||||||
|         ], |         ], | ||||||
|         Transaction::class => [ |         Transaction::class => [ | ||||||
|             'salt' => Transaction::class.config('app.key'), |             'salt' => Transaction::class.config('app.key'), | ||||||
|             'length' => '20', |             'length' => '20', | ||||||
|             'alphabet' => 'ADyWE86Cg7jF23vS0bonXrZ5KLH9puIQ4M1T', |             'alphabet' => 'ADyQWE8mgt7jF2vbnPrKLJenHVpiUIq4M12T', | ||||||
|         ], |         ], | ||||||
|     ], |     ], | ||||||
| ]; | ]; | ||||||
|  | |||||||
| @ -27,7 +27,6 @@ return [ | |||||||
|             'tokenizer', |             'tokenizer', | ||||||
|             'JSON', |             'JSON', | ||||||
|             'cURL', |             'cURL', | ||||||
|             'zip', |  | ||||||
|         ], |         ], | ||||||
|         'apache' => [ |         'apache' => [ | ||||||
|             'mod_rewrite', |             'mod_rewrite', | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| #!/bin/sh | #!/bin/sh | ||||||
|  |  | ||||||
|  | docker compose exec app composer config --no-plugins allow-plugins.pestphp/pest-plugin true | ||||||
| docker-compose exec app composer install --no-interaction --prefer-dist --optimize-autoloader | docker-compose exec app composer install --no-interaction --prefer-dist --optimize-autoloader | ||||||
|  |  | ||||||
| docker-compose exec app php artisan storage:link || true | docker-compose exec app php artisan storage:link || true | ||||||
|  | |||||||
| @ -17,7 +17,18 @@ | |||||||
|             <td class="px-5 py-4 text-left align-top"> |             <td class="px-5 py-4 text-left align-top"> | ||||||
|               <div class="flex justify-start"> |               <div class="flex justify-start"> | ||||||
|                 <div |                 <div | ||||||
|                   class="flex items-center justify-center w-5 h-5 mt-2 mr-2 text-gray-300 cursor-move  handle" |                   class=" | ||||||
|  |                     flex | ||||||
|  |                     items-center | ||||||
|  |                     justify-center | ||||||
|  |                     w-5 | ||||||
|  |                     h-5 | ||||||
|  |                     mt-2 | ||||||
|  |                     text-gray-300 | ||||||
|  |                     cursor-move | ||||||
|  |                     handle | ||||||
|  |                     mr-2 | ||||||
|  |                   " | ||||||
|                 > |                 > | ||||||
|                   <DragIcon /> |                   <DragIcon /> | ||||||
|                 </div> |                 </div> | ||||||
| @ -97,7 +108,7 @@ | |||||||
|  |  | ||||||
|                           <BaseIcon |                           <BaseIcon | ||||||
|                             name="ChevronDownIcon" |                             name="ChevronDownIcon" | ||||||
|                             class="w-4 h-4 ml-1 text-gray-500" |                             class="w-4 h-4 text-gray-500 ml-1" | ||||||
|                           /> |                           /> | ||||||
|                         </span> |                         </span> | ||||||
|                       </BaseButton> |                       </BaseButton> | ||||||
| @ -144,7 +155,7 @@ | |||||||
|               <BaseContentPlaceholders v-if="loading"> |               <BaseContentPlaceholders v-if="loading"> | ||||||
|                 <BaseContentPlaceholdersText |                 <BaseContentPlaceholdersText | ||||||
|                   :lines="1" |                   :lines="1" | ||||||
|                   class="w-24 h-8 border rounded-md" |                   class="w-24 h-8 rounded-md border" | ||||||
|                 /> |                 /> | ||||||
|               </BaseContentPlaceholders> |               </BaseContentPlaceholders> | ||||||
|  |  | ||||||
| @ -164,7 +175,6 @@ | |||||||
|                 :ability="abilities.CREATE_INVOICE" |                 :ability="abilities.CREATE_INVOICE" | ||||||
|                 :store="store" |                 :store="store" | ||||||
|                 :store-prop="storeProp" |                 :store-prop="storeProp" | ||||||
|                 :discount="discount" |  | ||||||
|                 @update="updateTax" |                 @update="updateTax" | ||||||
|               /> |               /> | ||||||
|             </td> |             </td> | ||||||
|  | |||||||
| @ -30,13 +30,24 @@ | |||||||
|         <template v-if="userStore.hasAbilities(ability)" #action> |         <template v-if="userStore.hasAbilities(ability)" #action> | ||||||
|           <button |           <button | ||||||
|             type="button" |             type="button" | ||||||
|             class="flex items-center justify-center w-full px-2 py-2 bg-gray-200 border-none outline-none cursor-pointer " |             class=" | ||||||
|  |               flex | ||||||
|  |               items-center | ||||||
|  |               justify-center | ||||||
|  |               w-full | ||||||
|  |               px-2 | ||||||
|  |               cursor-pointer | ||||||
|  |               py-2 | ||||||
|  |               bg-gray-200 | ||||||
|  |               border-none | ||||||
|  |               outline-none | ||||||
|  |             " | ||||||
|             @click="openTaxModal" |             @click="openTaxModal" | ||||||
|           > |           > | ||||||
|             <BaseIcon name="CheckCircleIcon" class="h-5 text-primary-400" /> |             <BaseIcon name="CheckCircleIcon" class="h-5 text-primary-400" /> | ||||||
|  |  | ||||||
|             <label |             <label | ||||||
|               class="ml-2 text-sm leading-none cursor-pointer text-primary-400" |               class="ml-2 text-sm leading-none text-primary-400 cursor-pointer" | ||||||
|               >{{ $t('invoices.add_new_tax') }}</label |               >{{ $t('invoices.add_new_tax') }}</label | ||||||
|             > |             > | ||||||
|           </button> |           </button> | ||||||
| @ -104,10 +115,6 @@ const props = defineProps({ | |||||||
|     type: Number, |     type: Number, | ||||||
|     default: 0, |     default: 0, | ||||||
|   }, |   }, | ||||||
|   discountedTotal: { |  | ||||||
|     type: Number, |  | ||||||
|     default: 0, |  | ||||||
|   }, |  | ||||||
|   currency: { |   currency: { | ||||||
|     type: [Object, String], |     type: [Object, String], | ||||||
|     required: true, |     required: true, | ||||||
| @ -146,19 +153,19 @@ const filteredTypes = computed(() => { | |||||||
| }) | }) | ||||||
|  |  | ||||||
| const taxAmount = computed(() => { | const taxAmount = computed(() => { | ||||||
|   if (localTax.compound_tax && props.discountedTotal) { |   if (localTax.compound_tax && props.total) { | ||||||
|     return ((props.discountedTotal + props.totalTax) * localTax.percent) / 100 |     return ((props.total + props.totalTax) * localTax.percent) / 100 | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (props.discountedTotal && localTax.percent) { |   if (props.total && localTax.percent) { | ||||||
|     return (props.discountedTotal * localTax.percent) / 100 |     return (props.total * localTax.percent) / 100 | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return 0 |   return 0 | ||||||
| }) | }) | ||||||
|  |  | ||||||
| watch( | watch( | ||||||
|   () => props.discountedTotal, |   () => props.total, | ||||||
|   () => { |   () => { | ||||||
|     updateRowTax() |     updateRowTax() | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -29,7 +29,14 @@ | |||||||
|  |  | ||||||
|       <label |       <label | ||||||
|         v-else |         v-else | ||||||
|         class="flex items-center justify-center m-0 text-lg text-black uppercase " |         class=" | ||||||
|  |           flex | ||||||
|  |           items-center | ||||||
|  |           justify-center | ||||||
|  |           m-0 | ||||||
|  |           text-lg text-black | ||||||
|  |           uppercase | ||||||
|  |         " | ||||||
|       > |       > | ||||||
|         <BaseFormatMoney |         <BaseFormatMoney | ||||||
|           :amount="store.getSubTotal" |           :amount="store.getSubTotal" | ||||||
| @ -59,7 +66,14 @@ | |||||||
|  |  | ||||||
|       <label |       <label | ||||||
|         v-else-if="store[storeProp].tax_per_item === 'YES'" |         v-else-if="store[storeProp].tax_per_item === 'YES'" | ||||||
|         class="flex items-center justify-center m-0 text-lg text-black uppercase " |         class=" | ||||||
|  |           flex | ||||||
|  |           items-center | ||||||
|  |           justify-center | ||||||
|  |           m-0 | ||||||
|  |           text-lg text-black | ||||||
|  |           uppercase | ||||||
|  |         " | ||||||
|       > |       > | ||||||
|         <BaseFormatMoney :amount="tax.amount" :currency="defaultCurrency" /> |         <BaseFormatMoney :amount="tax.amount" :currency="defaultCurrency" /> | ||||||
|       </label> |       </label> | ||||||
| @ -84,7 +98,7 @@ | |||||||
|       <BaseContentPlaceholders v-if="isLoading"> |       <BaseContentPlaceholders v-if="isLoading"> | ||||||
|         <BaseContentPlaceholdersText |         <BaseContentPlaceholdersText | ||||||
|           :lines="1" |           :lines="1" | ||||||
|           class="w-24 h-8 border rounded-md" |           class="w-24 h-8 rounded-md border" | ||||||
|         /> |         /> | ||||||
|       </BaseContentPlaceholders> |       </BaseContentPlaceholders> | ||||||
|       <div v-else class="flex" style="width: 140px" role="group"> |       <div v-else class="flex" style="width: 140px" role="group"> | ||||||
| @ -100,7 +114,7 @@ | |||||||
|         <BaseDropdown position="bottom-end"> |         <BaseDropdown position="bottom-end"> | ||||||
|           <template #activator> |           <template #activator> | ||||||
|             <BaseButton |             <BaseButton | ||||||
|               class="p-2 rounded-none rounded-tr-md rounded-br-md" |               class="rounded-tr-md rounded-br-md p-2 rounded-none" | ||||||
|               type="button" |               type="button" | ||||||
|               variant="white" |               variant="white" | ||||||
|             > |             > | ||||||
| @ -113,7 +127,7 @@ | |||||||
|  |  | ||||||
|                 <BaseIcon |                 <BaseIcon | ||||||
|                   name="ChevronDownIcon" |                   name="ChevronDownIcon" | ||||||
|                   class="w-4 h-4 ml-1 text-gray-500" |                   class="w-4 h-4 text-gray-500 ml-1" | ||||||
|                 /> |                 /> | ||||||
|               </span> |               </span> | ||||||
|             </BaseButton> |             </BaseButton> | ||||||
| @ -166,7 +180,15 @@ | |||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div |     <div | ||||||
|       class="flex items-center justify-between w-full pt-2 mt-5 border-t border-gray-200 border-solid " |       class=" | ||||||
|  |         flex | ||||||
|  |         items-center | ||||||
|  |         justify-between | ||||||
|  |         w-full | ||||||
|  |         pt-2 | ||||||
|  |         mt-5 | ||||||
|  |         border-t border-gray-200 border-solid | ||||||
|  |       " | ||||||
|     > |     > | ||||||
|       <BaseContentPlaceholders v-if="isLoading"> |       <BaseContentPlaceholders v-if="isLoading"> | ||||||
|         <BaseContentPlaceholdersText :lines="1" class="w-16 h-5" /> |         <BaseContentPlaceholdersText :lines="1" class="w-16 h-5" /> | ||||||
| @ -182,7 +204,14 @@ | |||||||
|       </BaseContentPlaceholders> |       </BaseContentPlaceholders> | ||||||
|       <label |       <label | ||||||
|         v-else |         v-else | ||||||
|         class="flex items-center justify-center text-lg uppercase  text-primary-400" |         class=" | ||||||
|  |           flex | ||||||
|  |           items-center | ||||||
|  |           justify-center | ||||||
|  |           text-lg | ||||||
|  |           uppercase | ||||||
|  |           text-primary-400 | ||||||
|  |         " | ||||||
|       > |       > | ||||||
|         <BaseFormatMoney :amount="store.getTotal" :currency="defaultCurrency" /> |         <BaseFormatMoney :amount="store.getTotal" :currency="defaultCurrency" /> | ||||||
|       </label> |       </label> | ||||||
| @ -305,7 +334,6 @@ function selectPercentage() { | |||||||
|  |  | ||||||
| function onSelectTax(selectedTax) { | function onSelectTax(selectedTax) { | ||||||
|   let amount = 0 |   let amount = 0 | ||||||
|  |  | ||||||
|   if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) { |   if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) { | ||||||
|     amount = Math.round( |     amount = Math.round( | ||||||
|       ((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) * |       ((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) * | ||||||
|  | |||||||
| @ -143,7 +143,7 @@ | |||||||
|           <template #activator> |           <template #activator> | ||||||
|             <img |             <img | ||||||
|               :src="previewAvatar" |               :src="previewAvatar" | ||||||
|               class="block w-8 h-8 rounded md:h-9 md:w-9 object-cover" |               class="block w-8 h-8 rounded md:h-9 md:w-9" | ||||||
|             /> |             /> | ||||||
|           </template> |           </template> | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,40 +0,0 @@ | |||||||
| APP_ENV=production |  | ||||||
| APP_KEY=base64:kgk/4DW1vEVy7aEvet5FPp5un6PIGe/so8H0mvoUtW0= |  | ||||||
| APP_DEBUG=true |  | ||||||
| APP_LOG_LEVEL=debug |  | ||||||
| APP_URL=http://crater.test |  | ||||||
|  |  | ||||||
| DB_CONNECTION=mysql |  | ||||||
| DB_HOST=127.0.0.1 |  | ||||||
| DB_PORT=3306 |  | ||||||
| DB_DATABASE=crater |  | ||||||
| DB_USERNAME=crater |  | ||||||
| DB_PASSWORD=crater |  | ||||||
|  |  | ||||||
| BROADCAST_DRIVER=log |  | ||||||
| CACHE_DRIVER=file |  | ||||||
| QUEUE_DRIVER=sync |  | ||||||
| SESSION_DRIVER=cookie |  | ||||||
| SESSION_LIFETIME=1440 |  | ||||||
|  |  | ||||||
| REDIS_HOST=127.0.0.1 |  | ||||||
| REDIS_PASSWORD=null |  | ||||||
| REDIS_PORT=6379 |  | ||||||
|  |  | ||||||
| MAIL_DRIVER=smtp |  | ||||||
| MAIL_HOST= |  | ||||||
| MAIL_PORT= |  | ||||||
| MAIL_USERNAME= |  | ||||||
| MAIL_PASSWORD= |  | ||||||
| MAIL_ENCRYPTION= |  | ||||||
|  |  | ||||||
| PUSHER_APP_ID= |  | ||||||
| PUSHER_KEY= |  | ||||||
| PUSHER_SECRET= |  | ||||||
|  |  | ||||||
| SANCTUM_STATEFUL_DOMAINS=crater.test |  | ||||||
| SESSION_DOMAIN=crater.test |  | ||||||
|  |  | ||||||
| TRUSTED_PROXIES="*" |  | ||||||
|  |  | ||||||
| CRON_JOB_AUTH_TOKEN="" |  | ||||||
| @ -1,64 +0,0 @@ | |||||||
| FROM php:8.1-fpm |  | ||||||
|  |  | ||||||
| # Install system dependencies |  | ||||||
| RUN apt-get update && apt-get install -y \ |  | ||||||
|     git \ |  | ||||||
|     curl \ |  | ||||||
|     libpng-dev \ |  | ||||||
|     libonig-dev \ |  | ||||||
|     libxml2-dev \ |  | ||||||
|     zip \ |  | ||||||
|     unzip \ |  | ||||||
|     libzip-dev \ |  | ||||||
|     libmagickwand-dev \ |  | ||||||
|     mariadb-client \ |  | ||||||
|     npm |  | ||||||
|  |  | ||||||
| # Clear cache |  | ||||||
| RUN apt-get clean && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| RUN pecl install imagick \ |  | ||||||
|     && docker-php-ext-enable imagick |  | ||||||
|  |  | ||||||
| # Install PHP extensions |  | ||||||
| RUN docker-php-ext-install pdo_mysql mbstring zip exif pcntl bcmath gd |  | ||||||
|  |  | ||||||
| # Get latest Composer |  | ||||||
| COPY --from=composer:latest /usr/bin/composer /usr/bin/composer |  | ||||||
|  |  | ||||||
| # Create system user to run Composer and Artisan Commands |  | ||||||
| RUN useradd -G www-data,root -u 1000 -d /home/crater-user crater-user |  | ||||||
| RUN mkdir -p /home/crater-user/.composer && \ |  | ||||||
|     chown -R crater-user:crater-user /home/crater-user |  | ||||||
|  |  | ||||||
| # Mounted volumes |  | ||||||
| COPY ./ /var/www |  | ||||||
| COPY ./docker-compose/php/uploads.ini /usr/local/etc/php/conf.d/uploads.ini |  | ||||||
| COPY ./uffizzi/.env.example /var/www/.env |  | ||||||
|  |  | ||||||
| # Set working directory |  | ||||||
| WORKDIR /var/www |  | ||||||
|  |  | ||||||
| RUN chown -R crater-user:crater-user ./ |  | ||||||
| RUN chmod -R 775 composer.json composer.lock \  |  | ||||||
|         composer.lock storage/framework/ \ |  | ||||||
|         storage/logs/ bootstrap/cache/ /home/crater-user/.composer |  | ||||||
| RUN chown -R $(whoami):$(whoami) /var/log/ |  | ||||||
| RUN chmod -R 775 /var/log |  | ||||||
|  |  | ||||||
| # Cleanup manually generated build files |  | ||||||
| RUN rm -rf /var/www/public/build |  | ||||||
| RUN npm config set user 0 |  | ||||||
| RUN npm config set unsafe-perm true |  | ||||||
| # Frontend bulding |  | ||||||
| RUN sed -i 's/DB_CONNECTION=mysql/DB_CONNECTION=sqlite/g' /var/www/.env |  | ||||||
| RUN sed -i 's/DB_DATABASE=crater/DB_DATABASE=\/tmp\/crater.sqlite/g' /var/www/.env |  | ||||||
| RUN touch /tmp/crater.sqlite |  | ||||||
| RUN composer install --no-interaction --prefer-dist |  | ||||||
| RUN npm i -f |  | ||||||
| RUN npm install --save-dev sass |  | ||||||
| RUN export NODE_OPTIONS="--max-old-space-size=4096" && /usr/bin/npx vite build --target=es2020 |  | ||||||
| RUN sed -i 's/DB_CONNECTION=sqlite/DB_CONNECTION=mysql/g' /var/www/.env |  | ||||||
| RUN sed -i 's/DB_DATABASE=\/tmp\/crater.sqlite/DB_DATABASE=crater/g' /var/www/.env |  | ||||||
|  |  | ||||||
| USER crater-user |  | ||||||
| @ -1,68 +0,0 @@ | |||||||
| FROM php:8.1-fpm as build |  | ||||||
|  |  | ||||||
| # Install system dependencies |  | ||||||
| RUN apt-get update && apt-get install -y \ |  | ||||||
|     git \ |  | ||||||
|     curl \ |  | ||||||
|     libpng-dev \ |  | ||||||
|     libonig-dev \ |  | ||||||
|     libxml2-dev \ |  | ||||||
|     zip \ |  | ||||||
|     unzip \ |  | ||||||
|     libzip-dev \ |  | ||||||
|     libmagickwand-dev \ |  | ||||||
|     mariadb-client |  | ||||||
|  |  | ||||||
| # Clear cache |  | ||||||
| RUN apt-get clean && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| RUN pecl install imagick \ |  | ||||||
|     && docker-php-ext-enable imagick |  | ||||||
|  |  | ||||||
| # Install PHP extensions |  | ||||||
| RUN docker-php-ext-install pdo_mysql mbstring zip exif pcntl bcmath gd |  | ||||||
|  |  | ||||||
| # Get latest Composer |  | ||||||
| COPY --from=composer:latest /usr/bin/composer /usr/bin/composer |  | ||||||
|  |  | ||||||
| # Create system user to run Composer and Artisan Commands |  | ||||||
| RUN useradd -G www-data,root -u 1000 -d /home/crater-user crater-user |  | ||||||
| RUN mkdir -p /home/crater-user/.composer && \ |  | ||||||
|     chown -R crater-user:crater-user /home/crater-user |  | ||||||
|  |  | ||||||
| # Mounted volumes |  | ||||||
| COPY ./ /var/www |  | ||||||
| COPY ./docker-compose/php/uploads.ini /usr/local/etc/php/conf.d/uploads.ini |  | ||||||
| COPY ./uffizzi/.env.example /var/www/.env |  | ||||||
|  |  | ||||||
| # Set working directory |  | ||||||
| WORKDIR /var/www |  | ||||||
|  |  | ||||||
| RUN chown -R crater-user:crater-user ./ |  | ||||||
| RUN chmod -R 775 composer.json composer.lock \  |  | ||||||
|         composer.lock storage/framework/ \ |  | ||||||
|         storage/logs/ bootstrap/cache/ /home/crater-user/.composer |  | ||||||
|  |  | ||||||
| RUN composer config --no-plugins allow-plugins.pestphp/pest-plugin true && \ |  | ||||||
|     composer install --no-interaction --prefer-dist --optimize-autoloader && \ |  | ||||||
|     php artisan storage:link || true && \ |  | ||||||
|     php artisan key:generate |  | ||||||
|  |  | ||||||
| FROM php:8.0-fpm-alpine |  | ||||||
|  |  | ||||||
| RUN apk add --no-cache \ |  | ||||||
|     php8-bcmath |  | ||||||
|  |  | ||||||
| RUN docker-php-ext-install pdo pdo_mysql bcmath |  | ||||||
|  |  | ||||||
| COPY docker-compose/crontab /etc/crontabs/root |  | ||||||
|  |  | ||||||
| # Mounted volumes |  | ||||||
| COPY --from=build /var/www /var/www |  | ||||||
|  |  | ||||||
| RUN chown -R $(whoami):$(whoami) /var/www/ |  | ||||||
| RUN chmod -R 775 /var/www/ |  | ||||||
| RUN chown -R $(whoami):$(whoami) /var/log/ |  | ||||||
| RUN chmod -R 775 /var/log/ |  | ||||||
|  |  | ||||||
| CMD ["crond", "-f"] |  | ||||||
| @ -1,58 +0,0 @@ | |||||||
| version: '3' |  | ||||||
|  |  | ||||||
| x-uffizzi: |  | ||||||
|   ingress: |  | ||||||
|     service: nginx |  | ||||||
|     port: 80 |  | ||||||
|  |  | ||||||
| services: |  | ||||||
|   app: |  | ||||||
|     image: "${APP_IMAGE}" |  | ||||||
|     restart: unless-stopped |  | ||||||
|     working_dir: /var/www/ |  | ||||||
|     command: ["-c"," |  | ||||||
|             composer config --no-plugins allow-plugins.pestphp/pest-plugin true &&  |  | ||||||
|             composer install --no-interaction --prefer-dist --optimize-autoloader &&  |  | ||||||
|             php artisan storage:link || true &&  |  | ||||||
|             php artisan key:generate --force &&  |  | ||||||
|             php-fpm", |  | ||||||
|             ] |  | ||||||
|     entrypoint: /bin/sh |  | ||||||
|     depends_on: |  | ||||||
|       - db |  | ||||||
|     deploy: |  | ||||||
|       resources: |  | ||||||
|         limits: |  | ||||||
|           memory: 1000m |  | ||||||
|  |  | ||||||
|   db: |  | ||||||
|     image: mariadb |  | ||||||
|     restart: always |  | ||||||
|     environment: |  | ||||||
|       MYSQL_USER: crater |  | ||||||
|       MYSQL_PASSWORD: crater |  | ||||||
|       MYSQL_DATABASE: crater |  | ||||||
|       MYSQL_ROOT_PASSWORD: crater |  | ||||||
|     ports: |  | ||||||
|       - '33006:3306' |  | ||||||
|     deploy: |  | ||||||
|       resources: |  | ||||||
|         limits: |  | ||||||
|           memory: 500m |  | ||||||
|  |  | ||||||
|   nginx: |  | ||||||
|     image: "${NGINX_IMAGE}" |  | ||||||
|     restart: unless-stopped |  | ||||||
|     ports: |  | ||||||
|       - 80:80 |  | ||||||
|     depends_on: |  | ||||||
|       - app |  | ||||||
|     resources: |  | ||||||
|       limits: |  | ||||||
|         memory: 500m |  | ||||||
|  |  | ||||||
|   cron: |  | ||||||
|     image: "${CROND_IMAGE}" |  | ||||||
|     restart: always |  | ||||||
|      |  | ||||||
|  |  | ||||||
| @ -1,9 +0,0 @@ | |||||||
| ARG BASE_IMAGE |  | ||||||
|  |  | ||||||
| FROM $BASE_IMAGE as build |  | ||||||
| FROM nginx:1.17-alpine |  | ||||||
|  |  | ||||||
| RUN rm /etc/nginx/conf.d/default.conf |  | ||||||
|  |  | ||||||
| COPY --from=build /var/www /var/www |  | ||||||
| COPY ./uffizzi/nginx/nginx /etc/nginx/conf.d/ |  | ||||||
| @ -1,22 +0,0 @@ | |||||||
| server { |  | ||||||
|     client_max_body_size 64M; |  | ||||||
|     listen 80; |  | ||||||
|     index index.php index.html; |  | ||||||
|     error_log  /var/log/nginx/error.log; |  | ||||||
|     access_log /var/log/nginx/access.log; |  | ||||||
|     root /var/www/public; |  | ||||||
|     location ~ \.php$ { |  | ||||||
|         try_files $uri =404; |  | ||||||
|         fastcgi_split_path_info ^(.+\.php)(/.+)$; |  | ||||||
|         fastcgi_pass localhost:9000; |  | ||||||
|         fastcgi_index index.php; |  | ||||||
|         include fastcgi_params; |  | ||||||
|         fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; |  | ||||||
|         fastcgi_param PATH_INFO $fastcgi_path_info; |  | ||||||
|         fastcgi_read_timeout 300; |  | ||||||
|     } |  | ||||||
|     location / { |  | ||||||
|         try_files $uri $uri/ /index.php?$query_string; |  | ||||||
|         gzip_static on; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
		Reference in New Issue
	
	Block a user
	