diff --git a/404.html b/404.html index e22e35da..4fab2dc4 100644 --- a/404.html +++ b/404.html @@ -18,7 +18,7 @@ - + diff --git a/assets/js/0058b4c6.3377be63.js b/assets/js/0058b4c6.3377be63.js deleted file mode 100644 index 3a5a10d9..00000000 --- a/assets/js/0058b4c6.3377be63.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[849],{6164:e=>{e.exports=JSON.parse('{"version":{"pluginId":"default","version":"current","label":"Next","banner":null,"badge":false,"noIndex":false,"className":"docs-version-current","isLast":true,"docsSidebars":{"tutorialSidebar":[{"type":"category","label":"Getting Started","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Installation","href":"/docs/getting-started/installation","docId":"getting-started/installation","unlisted":false},{"type":"link","label":"AthennaRC File","href":"/docs/getting-started/athennarc-file","docId":"getting-started/athennarc-file","unlisted":false},{"type":"link","label":"Configuration","href":"/docs/getting-started/configuration","docId":"getting-started/configuration","unlisted":false},{"type":"link","label":"Directory Structure","href":"/docs/getting-started/directory-structure","docId":"getting-started/directory-structure","unlisted":false}]},{"type":"category","label":"Architecture Concepts","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Application Lifecycle","href":"/docs/architecture-concepts/application-lifecycle","docId":"architecture-concepts/application-lifecycle","unlisted":false},{"type":"link","label":"Service Container","href":"/docs/architecture-concepts/service-container","docId":"architecture-concepts/service-container","unlisted":false},{"type":"link","label":"Service Providers","href":"/docs/architecture-concepts/service-providers","docId":"architecture-concepts/service-providers","unlisted":false},{"type":"link","label":"Facades","href":"/docs/architecture-concepts/facades","docId":"architecture-concepts/facades","unlisted":false}]},{"type":"category","label":"The Basics","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Logging","href":"/docs/the-basics/logging","docId":"the-basics/logging","unlisted":false},{"type":"link","label":"Helpers","href":"/docs/the-basics/helpers","docId":"the-basics/helpers","unlisted":false},{"type":"link","label":"Compilation","href":"/docs/the-basics/compilation","docId":"the-basics/compilation","unlisted":false},{"type":"link","label":"Deployment","href":"/docs/the-basics/deployment","docId":"the-basics/deployment","unlisted":false}]},{"type":"category","label":"Digging Deeper","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"REPL","href":"/docs/digging-deeper/repl","docId":"digging-deeper/repl","unlisted":false},{"type":"link","label":"Graceful Shutdown","href":"/docs/digging-deeper/graceful-shutdown","docId":"digging-deeper/graceful-shutdown","unlisted":false},{"type":"link","label":"Collections","href":"/docs/digging-deeper/collections","docId":"digging-deeper/collections","unlisted":false},{"type":"link","label":"Mail","href":"/docs/digging-deeper/mail","docId":"digging-deeper/mail","unlisted":false},{"type":"link","label":"Library Development","href":"/docs/digging-deeper/library-development","docId":"digging-deeper/library-development","unlisted":false}]},{"type":"category","label":"REST API Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Routing","href":"/docs/rest-api-application/routing","docId":"rest-api-application/routing","unlisted":false},{"type":"link","label":"Middlewares","href":"/docs/rest-api-application/middlewares","docId":"rest-api-application/middlewares","unlisted":false},{"type":"link","label":"Controllers","href":"/docs/rest-api-application/controllers","docId":"rest-api-application/controllers","unlisted":false},{"type":"link","label":"Request Context","href":"/docs/rest-api-application/request-context","docId":"rest-api-application/request-context","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/rest-api-application/error-handling","docId":"rest-api-application/error-handling","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/rest-api-application/annotations","docId":"rest-api-application/annotations","unlisted":false},{"type":"link","label":"Rate Limiting","href":"/docs/rest-api-application/rate-limiting","docId":"rest-api-application/rate-limiting","unlisted":false},{"type":"link","label":"Tracing Requests","href":"/docs/rest-api-application/tracing-requests","docId":"rest-api-application/tracing-requests","unlisted":false},{"type":"link","label":"Security with Helmet","href":"/docs/rest-api-application/security-with-helmet","docId":"rest-api-application/security-with-helmet","unlisted":false},{"type":"link","label":"Swagger Documentation","href":"/docs/rest-api-application/swagger-documentation","docId":"rest-api-application/swagger-documentation","unlisted":false},{"type":"link","label":"Views","href":"/docs/rest-api-application/views","docId":"rest-api-application/views","unlisted":false},{"type":"link","label":"Static Files","href":"/docs/rest-api-application/static-files","docId":"rest-api-application/static-files","unlisted":false}]},{"type":"category","label":"CLI Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Commands","href":"/docs/cli-application/commands","docId":"cli-application/commands","unlisted":false},{"type":"link","label":"Running","href":"/docs/cli-application/running","docId":"cli-application/running","unlisted":false},{"type":"link","label":"Publishing","href":"/docs/cli-application/publishing","docId":"cli-application/publishing","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/cli-application/error-handling","docId":"cli-application/error-handling","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/cli-application/annotations","docId":"cli-application/annotations","unlisted":false}]},{"type":"category","label":"CRON Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Schedulers","href":"/docs/cron-application/schedulers","docId":"cron-application/schedulers","unlisted":false},{"type":"link","label":"CRON Context","href":"/docs/cron-application/cron-context","docId":"cron-application/cron-context","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/cron-application/error-handling","docId":"cron-application/error-handling","unlisted":false},{"type":"link","label":"Tracing Executions","href":"/docs/cron-application/tracing-executions","docId":"cron-application/tracing-executions","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/cron-application/annotations","docId":"cron-application/annotations","unlisted":false}]},{"type":"category","label":"Database","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/database/getting-started","docId":"database/getting-started","unlisted":false},{"type":"link","label":"Query Builder","href":"/docs/database/query-builder","docId":"database/query-builder","unlisted":false},{"type":"link","label":"Migrations","href":"/docs/database/migrations","docId":"database/migrations","unlisted":false},{"type":"link","label":"Seeding","href":"/docs/database/seeding","docId":"database/seeding","unlisted":false}]},{"type":"category","label":"ORM","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/orm/getting-started","docId":"orm/getting-started","unlisted":false},{"type":"link","label":"Query Builder","href":"/docs/orm/query-builder","docId":"orm/query-builder","unlisted":false},{"type":"link","label":"Relationships","href":"/docs/orm/relationships","docId":"orm/relationships","unlisted":false},{"type":"link","label":"Extending Models","href":"/docs/orm/extending-models","docId":"orm/extending-models","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/orm/annotations","docId":"orm/annotations","unlisted":false},{"type":"link","label":"Factories","href":"/docs/orm/factories","docId":"orm/factories","unlisted":false}]},{"type":"category","label":"Testing","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/testing/getting-started","docId":"testing/getting-started","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/testing/annotations","docId":"testing/annotations","unlisted":false},{"type":"link","label":"REST API Testing","href":"/docs/testing/rest-api-testing","docId":"testing/rest-api-testing","unlisted":false},{"type":"link","label":"CLI Testing","href":"/docs/testing/cli-tests","docId":"testing/cli-tests","unlisted":false},{"type":"link","label":"Mocking","href":"/docs/testing/mocking","docId":"testing/mocking","unlisted":false}]}]},"docs":{"architecture-concepts/application-lifecycle":{"id":"architecture-concepts/application-lifecycle","title":"Application Lifecycle","description":"Understand each one of the Athenna applications lifecycles.","sidebar":"tutorialSidebar"},"architecture-concepts/facades":{"id":"architecture-concepts/facades","title":"Facades","description":"Understand the purpose and how to use the Athenna facades.","sidebar":"tutorialSidebar"},"architecture-concepts/service-container":{"id":"architecture-concepts/service-container","title":"Service Container","description":"Understand the purpose and how to use the Athenna service container.","sidebar":"tutorialSidebar"},"architecture-concepts/service-providers":{"id":"architecture-concepts/service-providers","title":"Service Providers","description":"Understand the purpose and how to use the Athenna service providers.","sidebar":"tutorialSidebar"},"cli-application/annotations":{"id":"cli-application/annotations","title":"Annotations","description":"Check all available Artisan annotations and it options.","sidebar":"tutorialSidebar"},"cli-application/commands":{"id":"cli-application/commands","title":"Commands","description":"See how to create and configure your CLI commands.","sidebar":"tutorialSidebar"},"cli-application/error-handling":{"id":"cli-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the CLI Application.","sidebar":"tutorialSidebar"},"cli-application/publishing":{"id":"cli-application/publishing","title":"Publishing","description":"Check how you can publish a global CLI application.","sidebar":"tutorialSidebar"},"cli-application/running":{"id":"cli-application/running","title":"Running","description":"Check how to run your CLI commands in different scenarios.","sidebar":"tutorialSidebar"},"cron-application/annotations":{"id":"cron-application/annotations","title":"Annotations","description":"Check all available CRON annotations and it options.","sidebar":"tutorialSidebar"},"cron-application/cron-context":{"id":"cron-application/cron-context","title":"CRON Context","description":"Understand the purpose of the CRON context object.","sidebar":"tutorialSidebar"},"cron-application/error-handling":{"id":"cron-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the CRON Application.","sidebar":"tutorialSidebar"},"cron-application/schedulers":{"id":"cron-application/schedulers","title":"Schedulers","description":"See how to create and configure your CRON job schedulers.","sidebar":"tutorialSidebar"},"cron-application/tracing-executions":{"id":"cron-application/tracing-executions","title":"Tracing Executions","description":"Understand how to trace executions of your CRON application.","sidebar":"tutorialSidebar"},"database/getting-started":{"id":"database/getting-started","title":"Getting Started","description":"See how to create database connections in Athenna Framework.","sidebar":"tutorialSidebar"},"database/migrations":{"id":"database/migrations","title":"Migrations","description":"See how to create and run database migrations in Athenna Framework.","sidebar":"tutorialSidebar"},"database/query-builder":{"id":"database/query-builder","title":"Query Builder","description":"See how to use the Athenna database query builder.","sidebar":"tutorialSidebar"},"database/seeding":{"id":"database/seeding","title":"Seeding","description":"See how to create and run database seeders.","sidebar":"tutorialSidebar"},"digging-deeper/collections":{"id":"digging-deeper/collections","title":"Collections","description":"Understand the Collections API of the Athenna Framework.","sidebar":"tutorialSidebar"},"digging-deeper/graceful-shutdown":{"id":"digging-deeper/graceful-shutdown","title":"Graceful Shutdown","description":"See how to graceful shutdown any kind of Athenna application.","sidebar":"tutorialSidebar"},"digging-deeper/library-development":{"id":"digging-deeper/library-development","title":"Library Development","description":"See how you can create your own library and integrate with Athenna.","sidebar":"tutorialSidebar"},"digging-deeper/mail":{"id":"digging-deeper/mail","title":"Mail","description":"See how to send emails in Athenna.","sidebar":"tutorialSidebar"},"digging-deeper/repl":{"id":"digging-deeper/repl","title":"REPL","description":"See how to create a REPL session with Athenna Framework ecosystem.","sidebar":"tutorialSidebar"},"getting-started/athennarc-file":{"id":"getting-started/athennarc-file","title":"AthennaRC File","description":"Understand what is the purpose of the .athennarc.json file.","sidebar":"tutorialSidebar"},"getting-started/configuration":{"id":"getting-started/configuration","title":"Configuration","description":"Understand the initial configurations of your project.","sidebar":"tutorialSidebar"},"getting-started/directory-structure":{"id":"getting-started/directory-structure","title":"Directory Structure","description":"Understand the directory structure of your project.","sidebar":"tutorialSidebar"},"getting-started/installation":{"id":"getting-started/installation","title":"Installation","description":"How to install and set up your first Athenna project.","sidebar":"tutorialSidebar"},"orm/annotations":{"id":"orm/annotations","title":"Annotations","description":"Check all available ORM annotations and it options.","sidebar":"tutorialSidebar"},"orm/extending-models":{"id":"orm/extending-models","title":"Extending Models","description":"See how to extend models implementations in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/factories":{"id":"orm/factories","title":"Factories","description":"See how to factory fake models in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/getting-started":{"id":"orm/getting-started","title":"Getting Started","description":"See how to create models in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/query-builder":{"id":"orm/query-builder","title":"Query Builder","description":"See how to retrieve, insert, update and delete models using the query builder in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/relationships":{"id":"orm/relationships","title":"Relationships","description":"See how to create relations between models in Athenna Framework.","sidebar":"tutorialSidebar"},"rest-api-application/annotations":{"id":"rest-api-application/annotations","title":"Annotations","description":"Check all available REST API annotations and it options.","sidebar":"tutorialSidebar"},"rest-api-application/controllers":{"id":"rest-api-application/controllers","title":"Controllers","description":"Understand how you can set up controllers in your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/error-handling":{"id":"rest-api-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/middlewares":{"id":"rest-api-application/middlewares","title":"Middlewares","description":"Understand how you can set up middlewares in your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/rate-limiting":{"id":"rest-api-application/rate-limiting","title":"Rate Limiting","description":"See how to create rate limiting rules for Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/request-context":{"id":"rest-api-application/request-context","title":"Request Context","description":"Understand the purpose of the request context object.","sidebar":"tutorialSidebar"},"rest-api-application/routing":{"id":"rest-api-application/routing","title":"Routing","description":"Understand how you can set up routes for your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/security-with-helmet":{"id":"rest-api-application/security-with-helmet","title":"Security with Helmet","description":"See how to improve the security of your REST API with Helmet in Athenna.","sidebar":"tutorialSidebar"},"rest-api-application/static-files":{"id":"rest-api-application/static-files","title":"Static Files","description":"See how to serve static files in Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/swagger-documentation":{"id":"rest-api-application/swagger-documentation","title":"Swagger Documentation","description":"See how to create the Swagger documentation for Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/tracing-requests":{"id":"rest-api-application/tracing-requests","title":"Tracing Requests","description":"Understand how to trace requests in your REST API application of Athenna.","sidebar":"tutorialSidebar"},"rest-api-application/views":{"id":"rest-api-application/views","title":"Views","description":"Understand how you can use the Athenna view API for rendering HTML pages.","sidebar":"tutorialSidebar"},"testing/annotations":{"id":"testing/annotations","title":"Annotations","description":"Check all available testing annotations and it options.","sidebar":"tutorialSidebar"},"testing/cli-tests":{"id":"testing/cli-tests","title":"CLI Testing","description":"See how to create tests for CLI applications in Athenna.","sidebar":"tutorialSidebar"},"testing/getting-started":{"id":"testing/getting-started","title":"Getting Started","description":"See how to create tests in an Athenna application.","sidebar":"tutorialSidebar"},"testing/mocking":{"id":"testing/mocking","title":"Mocking","description":"Understand how to mock dependencies and functions in Athenna.","sidebar":"tutorialSidebar"},"testing/rest-api-testing":{"id":"testing/rest-api-testing","title":"REST API Testing","description":"See how to create tests for REST APIs applications in Athenna.","sidebar":"tutorialSidebar"},"the-basics/compilation":{"id":"the-basics/compilation","title":"Compilation","description":"Understand the TypeScript compilation process of Athenna.","sidebar":"tutorialSidebar"},"the-basics/deployment":{"id":"the-basics/deployment","title":"Deployment","description":"See how to deploy any kind of Athenna application.","sidebar":"tutorialSidebar"},"the-basics/helpers":{"id":"the-basics/helpers","title":"Helpers","description":"Understand how to use all the Athenna Helpers from @athenna/common and other packages.","sidebar":"tutorialSidebar"},"the-basics/logging":{"id":"the-basics/logging","title":"Logging","description":"Understand how you can use the Athenna logging API.","sidebar":"tutorialSidebar"}}}}')}}]); \ No newline at end of file diff --git a/assets/js/0058b4c6.e7eea27f.js b/assets/js/0058b4c6.e7eea27f.js new file mode 100644 index 00000000..0332483d --- /dev/null +++ b/assets/js/0058b4c6.e7eea27f.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[849],{6164:e=>{e.exports=JSON.parse('{"version":{"pluginId":"default","version":"current","label":"Next","banner":null,"badge":false,"noIndex":false,"className":"docs-version-current","isLast":true,"docsSidebars":{"tutorialSidebar":[{"type":"category","label":"Getting Started","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Installation","href":"/docs/getting-started/installation","docId":"getting-started/installation","unlisted":false},{"type":"link","label":"Athenna RC File","href":"/docs/getting-started/athennarc-file","docId":"getting-started/athennarc-file","unlisted":false},{"type":"link","label":"Configuration","href":"/docs/getting-started/configuration","docId":"getting-started/configuration","unlisted":false},{"type":"link","label":"Directory Structure","href":"/docs/getting-started/directory-structure","docId":"getting-started/directory-structure","unlisted":false}]},{"type":"category","label":"Architecture Concepts","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Application Lifecycle","href":"/docs/architecture-concepts/application-lifecycle","docId":"architecture-concepts/application-lifecycle","unlisted":false},{"type":"link","label":"Service Container","href":"/docs/architecture-concepts/service-container","docId":"architecture-concepts/service-container","unlisted":false},{"type":"link","label":"Service Providers","href":"/docs/architecture-concepts/service-providers","docId":"architecture-concepts/service-providers","unlisted":false},{"type":"link","label":"Facades","href":"/docs/architecture-concepts/facades","docId":"architecture-concepts/facades","unlisted":false}]},{"type":"category","label":"The Basics","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Logging","href":"/docs/the-basics/logging","docId":"the-basics/logging","unlisted":false},{"type":"link","label":"Helpers","href":"/docs/the-basics/helpers","docId":"the-basics/helpers","unlisted":false},{"type":"link","label":"Compilation","href":"/docs/the-basics/compilation","docId":"the-basics/compilation","unlisted":false},{"type":"link","label":"Deployment","href":"/docs/the-basics/deployment","docId":"the-basics/deployment","unlisted":false}]},{"type":"category","label":"Digging Deeper","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"REPL","href":"/docs/digging-deeper/repl","docId":"digging-deeper/repl","unlisted":false},{"type":"link","label":"Graceful Shutdown","href":"/docs/digging-deeper/graceful-shutdown","docId":"digging-deeper/graceful-shutdown","unlisted":false},{"type":"link","label":"Collections","href":"/docs/digging-deeper/collections","docId":"digging-deeper/collections","unlisted":false},{"type":"link","label":"Mail","href":"/docs/digging-deeper/mail","docId":"digging-deeper/mail","unlisted":false},{"type":"link","label":"Library Development","href":"/docs/digging-deeper/library-development","docId":"digging-deeper/library-development","unlisted":false}]},{"type":"category","label":"REST API Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Routing","href":"/docs/rest-api-application/routing","docId":"rest-api-application/routing","unlisted":false},{"type":"link","label":"Middlewares","href":"/docs/rest-api-application/middlewares","docId":"rest-api-application/middlewares","unlisted":false},{"type":"link","label":"Controllers","href":"/docs/rest-api-application/controllers","docId":"rest-api-application/controllers","unlisted":false},{"type":"link","label":"Request Context","href":"/docs/rest-api-application/request-context","docId":"rest-api-application/request-context","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/rest-api-application/error-handling","docId":"rest-api-application/error-handling","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/rest-api-application/annotations","docId":"rest-api-application/annotations","unlisted":false},{"type":"link","label":"Rate Limiting","href":"/docs/rest-api-application/rate-limiting","docId":"rest-api-application/rate-limiting","unlisted":false},{"type":"link","label":"Tracing Requests","href":"/docs/rest-api-application/tracing-requests","docId":"rest-api-application/tracing-requests","unlisted":false},{"type":"link","label":"Security with Helmet","href":"/docs/rest-api-application/security-with-helmet","docId":"rest-api-application/security-with-helmet","unlisted":false},{"type":"link","label":"Swagger Documentation","href":"/docs/rest-api-application/swagger-documentation","docId":"rest-api-application/swagger-documentation","unlisted":false},{"type":"link","label":"Views","href":"/docs/rest-api-application/views","docId":"rest-api-application/views","unlisted":false},{"type":"link","label":"Static Files","href":"/docs/rest-api-application/static-files","docId":"rest-api-application/static-files","unlisted":false}]},{"type":"category","label":"CLI Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Commands","href":"/docs/cli-application/commands","docId":"cli-application/commands","unlisted":false},{"type":"link","label":"Running","href":"/docs/cli-application/running","docId":"cli-application/running","unlisted":false},{"type":"link","label":"Publishing","href":"/docs/cli-application/publishing","docId":"cli-application/publishing","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/cli-application/error-handling","docId":"cli-application/error-handling","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/cli-application/annotations","docId":"cli-application/annotations","unlisted":false}]},{"type":"category","label":"CRON Application","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Schedulers","href":"/docs/cron-application/schedulers","docId":"cron-application/schedulers","unlisted":false},{"type":"link","label":"CRON Context","href":"/docs/cron-application/cron-context","docId":"cron-application/cron-context","unlisted":false},{"type":"link","label":"Error Handling","href":"/docs/cron-application/error-handling","docId":"cron-application/error-handling","unlisted":false},{"type":"link","label":"Tracing Executions","href":"/docs/cron-application/tracing-executions","docId":"cron-application/tracing-executions","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/cron-application/annotations","docId":"cron-application/annotations","unlisted":false}]},{"type":"category","label":"Database","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/database/getting-started","docId":"database/getting-started","unlisted":false},{"type":"link","label":"Query Builder","href":"/docs/database/query-builder","docId":"database/query-builder","unlisted":false},{"type":"link","label":"Migrations","href":"/docs/database/migrations","docId":"database/migrations","unlisted":false},{"type":"link","label":"Seeding","href":"/docs/database/seeding","docId":"database/seeding","unlisted":false}]},{"type":"category","label":"ORM","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/orm/getting-started","docId":"orm/getting-started","unlisted":false},{"type":"link","label":"Query Builder","href":"/docs/orm/query-builder","docId":"orm/query-builder","unlisted":false},{"type":"link","label":"Relationships","href":"/docs/orm/relationships","docId":"orm/relationships","unlisted":false},{"type":"link","label":"Extending Models","href":"/docs/orm/extending-models","docId":"orm/extending-models","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/orm/annotations","docId":"orm/annotations","unlisted":false},{"type":"link","label":"Factories","href":"/docs/orm/factories","docId":"orm/factories","unlisted":false}]},{"type":"category","label":"Testing","collapsible":true,"collapsed":true,"items":[{"type":"link","label":"Getting Started","href":"/docs/testing/getting-started","docId":"testing/getting-started","unlisted":false},{"type":"link","label":"Annotations","href":"/docs/testing/annotations","docId":"testing/annotations","unlisted":false},{"type":"link","label":"REST API Testing","href":"/docs/testing/rest-api-testing","docId":"testing/rest-api-testing","unlisted":false},{"type":"link","label":"CLI Testing","href":"/docs/testing/cli-tests","docId":"testing/cli-tests","unlisted":false},{"type":"link","label":"Mocking","href":"/docs/testing/mocking","docId":"testing/mocking","unlisted":false}]}]},"docs":{"architecture-concepts/application-lifecycle":{"id":"architecture-concepts/application-lifecycle","title":"Application Lifecycle","description":"Understand each one of the Athenna applications lifecycles.","sidebar":"tutorialSidebar"},"architecture-concepts/facades":{"id":"architecture-concepts/facades","title":"Facades","description":"Understand the purpose and how to use the Athenna facades.","sidebar":"tutorialSidebar"},"architecture-concepts/service-container":{"id":"architecture-concepts/service-container","title":"Service Container","description":"Understand the purpose and how to use the Athenna service container.","sidebar":"tutorialSidebar"},"architecture-concepts/service-providers":{"id":"architecture-concepts/service-providers","title":"Service Providers","description":"Understand the purpose and how to use the Athenna service providers.","sidebar":"tutorialSidebar"},"cli-application/annotations":{"id":"cli-application/annotations","title":"Annotations","description":"Check all available Artisan annotations and it options.","sidebar":"tutorialSidebar"},"cli-application/commands":{"id":"cli-application/commands","title":"Commands","description":"See how to create and configure your CLI commands.","sidebar":"tutorialSidebar"},"cli-application/error-handling":{"id":"cli-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the CLI Application.","sidebar":"tutorialSidebar"},"cli-application/publishing":{"id":"cli-application/publishing","title":"Publishing","description":"Check how you can publish a global CLI application.","sidebar":"tutorialSidebar"},"cli-application/running":{"id":"cli-application/running","title":"Running","description":"Check how to run your CLI commands in different scenarios.","sidebar":"tutorialSidebar"},"cron-application/annotations":{"id":"cron-application/annotations","title":"Annotations","description":"Check all available CRON annotations and it options.","sidebar":"tutorialSidebar"},"cron-application/cron-context":{"id":"cron-application/cron-context","title":"CRON Context","description":"Understand the purpose of the CRON context object.","sidebar":"tutorialSidebar"},"cron-application/error-handling":{"id":"cron-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the CRON Application.","sidebar":"tutorialSidebar"},"cron-application/schedulers":{"id":"cron-application/schedulers","title":"Schedulers","description":"See how to create and configure your CRON job schedulers.","sidebar":"tutorialSidebar"},"cron-application/tracing-executions":{"id":"cron-application/tracing-executions","title":"Tracing Executions","description":"Understand how to trace executions of your CRON application.","sidebar":"tutorialSidebar"},"database/getting-started":{"id":"database/getting-started","title":"Getting Started","description":"See how to create database connections in Athenna Framework.","sidebar":"tutorialSidebar"},"database/migrations":{"id":"database/migrations","title":"Migrations","description":"See how to create and run database migrations in Athenna Framework.","sidebar":"tutorialSidebar"},"database/query-builder":{"id":"database/query-builder","title":"Query Builder","description":"See how to use the Athenna database query builder.","sidebar":"tutorialSidebar"},"database/seeding":{"id":"database/seeding","title":"Seeding","description":"See how to create and run database seeders.","sidebar":"tutorialSidebar"},"digging-deeper/collections":{"id":"digging-deeper/collections","title":"Collections","description":"Understand the Collections API of the Athenna Framework.","sidebar":"tutorialSidebar"},"digging-deeper/graceful-shutdown":{"id":"digging-deeper/graceful-shutdown","title":"Graceful Shutdown","description":"See how to graceful shutdown any kind of Athenna application.","sidebar":"tutorialSidebar"},"digging-deeper/library-development":{"id":"digging-deeper/library-development","title":"Library Development","description":"See how you can create your own library and integrate with Athenna.","sidebar":"tutorialSidebar"},"digging-deeper/mail":{"id":"digging-deeper/mail","title":"Mail","description":"See how to send emails in Athenna.","sidebar":"tutorialSidebar"},"digging-deeper/repl":{"id":"digging-deeper/repl","title":"REPL","description":"See how to create a REPL session with Athenna Framework ecosystem.","sidebar":"tutorialSidebar"},"getting-started/athennarc-file":{"id":"getting-started/athennarc-file","title":"Athenna RC File","description":"Understand what is the purpose of the .athennarc.json file.","sidebar":"tutorialSidebar"},"getting-started/configuration":{"id":"getting-started/configuration","title":"Configuration","description":"Understand the initial configurations of your project.","sidebar":"tutorialSidebar"},"getting-started/directory-structure":{"id":"getting-started/directory-structure","title":"Directory Structure","description":"Understand the directory structure of your project.","sidebar":"tutorialSidebar"},"getting-started/installation":{"id":"getting-started/installation","title":"Installation","description":"How to install and set up your first Athenna project.","sidebar":"tutorialSidebar"},"orm/annotations":{"id":"orm/annotations","title":"Annotations","description":"Check all available ORM annotations and it options.","sidebar":"tutorialSidebar"},"orm/extending-models":{"id":"orm/extending-models","title":"Extending Models","description":"See how to extend models implementations in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/factories":{"id":"orm/factories","title":"Factories","description":"See how to factory fake models in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/getting-started":{"id":"orm/getting-started","title":"Getting Started","description":"See how to create models in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/query-builder":{"id":"orm/query-builder","title":"Query Builder","description":"See how to retrieve, insert, update and delete models using the query builder in Athenna Framework.","sidebar":"tutorialSidebar"},"orm/relationships":{"id":"orm/relationships","title":"Relationships","description":"See how to create relations between models in Athenna Framework.","sidebar":"tutorialSidebar"},"rest-api-application/annotations":{"id":"rest-api-application/annotations","title":"Annotations","description":"Check all available REST API annotations and it options.","sidebar":"tutorialSidebar"},"rest-api-application/controllers":{"id":"rest-api-application/controllers","title":"Controllers","description":"Understand how you can set up controllers in your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/error-handling":{"id":"rest-api-application/error-handling","title":"Error Handling","description":"Understand how you can handle the errors of the REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/middlewares":{"id":"rest-api-application/middlewares","title":"Middlewares","description":"Understand how you can set up middlewares in your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/rate-limiting":{"id":"rest-api-application/rate-limiting","title":"Rate Limiting","description":"See how to create rate limiting rules for Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/request-context":{"id":"rest-api-application/request-context","title":"Request Context","description":"Understand the purpose of the request context object.","sidebar":"tutorialSidebar"},"rest-api-application/routing":{"id":"rest-api-application/routing","title":"Routing","description":"Understand how you can set up routes for your REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/security-with-helmet":{"id":"rest-api-application/security-with-helmet","title":"Security with Helmet","description":"See how to improve the security of your REST API with Helmet in Athenna.","sidebar":"tutorialSidebar"},"rest-api-application/static-files":{"id":"rest-api-application/static-files","title":"Static Files","description":"See how to serve static files in Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/swagger-documentation":{"id":"rest-api-application/swagger-documentation","title":"Swagger Documentation","description":"See how to create the Swagger documentation for Athenna REST API application.","sidebar":"tutorialSidebar"},"rest-api-application/tracing-requests":{"id":"rest-api-application/tracing-requests","title":"Tracing Requests","description":"Understand how to trace requests in your REST API application of Athenna.","sidebar":"tutorialSidebar"},"rest-api-application/views":{"id":"rest-api-application/views","title":"Views","description":"Understand how you can use the Athenna view API for rendering HTML pages.","sidebar":"tutorialSidebar"},"testing/annotations":{"id":"testing/annotations","title":"Annotations","description":"Check all available testing annotations and it options.","sidebar":"tutorialSidebar"},"testing/cli-tests":{"id":"testing/cli-tests","title":"CLI Testing","description":"See how to create tests for CLI applications in Athenna.","sidebar":"tutorialSidebar"},"testing/getting-started":{"id":"testing/getting-started","title":"Getting Started","description":"See how to create tests in an Athenna application.","sidebar":"tutorialSidebar"},"testing/mocking":{"id":"testing/mocking","title":"Mocking","description":"Understand how to mock dependencies and functions in Athenna.","sidebar":"tutorialSidebar"},"testing/rest-api-testing":{"id":"testing/rest-api-testing","title":"REST API Testing","description":"See how to create tests for REST APIs applications in Athenna.","sidebar":"tutorialSidebar"},"the-basics/compilation":{"id":"the-basics/compilation","title":"Compilation","description":"Understand the TypeScript compilation process of Athenna.","sidebar":"tutorialSidebar"},"the-basics/deployment":{"id":"the-basics/deployment","title":"Deployment","description":"See how to deploy any kind of Athenna application.","sidebar":"tutorialSidebar"},"the-basics/helpers":{"id":"the-basics/helpers","title":"Helpers","description":"Understand how to use all the Athenna Helpers from @athenna/common and other packages.","sidebar":"tutorialSidebar"},"the-basics/logging":{"id":"the-basics/logging","title":"Logging","description":"Understand how you can use the Athenna logging API.","sidebar":"tutorialSidebar"}}}}')}}]); \ No newline at end of file diff --git a/assets/js/30d27832.8b98abf6.js b/assets/js/30d27832.8b98abf6.js new file mode 100644 index 00000000..233cfba0 --- /dev/null +++ b/assets/js/30d27832.8b98abf6.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[4890],{9604:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>a,default:()=>u,frontMatter:()=>r,metadata:()=>c,toc:()=>d});var i=t(4848),o=t(8453),s=t(7049);const r={title:"Configuration",sidebar_position:4,description:"Understand the initial configurations of your project."},a="Configuration",c={id:"getting-started/configuration",title:"Configuration",description:"Understand the initial configurations of your project.",source:"@site/docs/getting-started/configuration.mdx",sourceDirName:"getting-started",slug:"/getting-started/configuration",permalink:"/docs/getting-started/configuration",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/configuration.mdx",tags:[],version:"current",sidebarPosition:4,frontMatter:{title:"Configuration",sidebar_position:4,description:"Understand the initial configurations of your project."},sidebar:"tutorialSidebar",previous:{title:"Athenna RC File",permalink:"/docs/getting-started/athennarc-file"},next:{title:"Directory Structure",permalink:"/docs/getting-started/directory-structure"}},l={},d=[{value:"Environment configuration",id:"environment-configuration",level:2},{value:"Determining the current environment",id:"determining-the-current-environment",level:3},{value:"Get an environment variable value",id:"get-an-environment-variable-value",level:3},{value:"Custom environment file path",id:"custom-environment-file-path",level:2},{value:"Configuration files",id:"configuration-files",level:2},{value:"Manipulating configuration values",id:"manipulating-configuration-values",level:3},{value:"Config.get()",id:"configget",level:4},{value:"Config.set()",id:"configset",level:4},{value:"Config.safeSet()",id:"configsafeset",level:4},{value:"Config.delete()",id:"configdelete",level:4},{value:"Config.rewrite()",id:"configrewrite",level:4},{value:"Config.is()",id:"configis",level:4},{value:"Config.existsAll()",id:"configexistsall",level:4},{value:"Config.clear()",id:"configclear",level:4},{value:"Config.load()",id:"configload",level:4},{value:"Config.safeLoad()",id:"configsafeload",level:4},{value:"Config.loadAll()",id:"configloadall",level:4},{value:"Get configuration with @Value() decorator",id:"get-configuration-with-value-decorator",level:3},{value:"Define my own configuration path",id:"define-my-own-configuration-path",level:3},{value:"Safe loading configuration files",id:"safe-loading-configuration-files",level:2},{value:"Debug mode",id:"debug-mode",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.header,{children:(0,i.jsx)(n.h1,{id:"configuration",children:"Configuration"})}),"\n",(0,i.jsx)(n.p,{children:"Understand the initial configurations of your project."}),"\n",(0,i.jsx)(n.h2,{id:"environment-configuration",children:"Environment configuration"}),"\n",(0,i.jsx)(n.p,{children:"It is often helpful to have different configuration values\nbased on the environment where the application is running.\nFor example, you may wish to use a different storage driver\nlocally than you do on your production server."}),"\n",(0,i.jsxs)(n.p,{children:["To make this a cinch, Athenna utilizes the ",(0,i.jsx)(n.a,{href:"https://www.npmjs.com/package/dotenv",children:"dotenv"}),"\nNode.js library. In a fresh Athenna installation, the root\ndirectory of your application will contain a ",(0,i.jsx)(n.code,{children:".env.example"})," file\nthat defines many common environment variables. During the\nAthenna installation process, this file will automatically\nbe copied to ",(0,i.jsx)(n.code,{children:".env"})," and ",(0,i.jsx)(n.code,{children:".env.test"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Athenna default ",(0,i.jsx)(n.code,{children:".env"})," file contains some common configuration\nvalues that may differ based on whether your application is\nrunning locally or on a production. These values are then\nretrieved from various Athenna configuration files within the\n",(0,i.jsx)(n.code,{children:"config"})," directory using Athenna ",(0,i.jsx)(n.code,{children:"Env()"})," function."]}),"\n",(0,i.jsxs)(n.p,{children:["If you are developing with a team, you may wish to continue\nincluding a ",(0,i.jsx)(n.code,{children:".env.example"})," file with your application. By\nputting placeholder values in the example configuration file,\nother developers on your team can clearly see which environment\nvariables are needed to run your application."]}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["Any variable in your ",(0,i.jsx)(n.code,{children:".env"})," file can be overridden by\nexternal environment variables such as server-level or\nsystem-level environment variables. But off course you\ncan turn off this behavior setting the ",(0,i.jsx)(n.code,{children:"OVERRIDE_ENV=true"}),"\nvariable before running your application, if this variable\nis set to true, all environment variables set in ",(0,i.jsx)(n.code,{children:".env"}),"\nwill override the externals."]})}),"\n",(0,i.jsx)(n.admonition,{type:"caution",children:(0,i.jsxs)(n.p,{children:["Your ",(0,i.jsx)(n.code,{children:".env"})," file should not be committed to your application\nsource control, since each developer/server using your\napplication could require a different environment configuration.\nFurthermore, this would be a security risk in the event an\nintruder gains access to your source control repository,\nsince any sensitive credentials would get exposed."]})}),"\n",(0,i.jsx)(n.h3,{id:"determining-the-current-environment",children:"Determining the current environment"}),"\n",(0,i.jsxs)(n.p,{children:["Before loading your application's environment variables,\nAthenna determines if either the ",(0,i.jsx)(n.code,{children:"--env"})," Artisan flag has\nbeen specified or if the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," environment variable has\nbeen externally provided."]}),"\n",(0,i.jsxs)(n.p,{children:["If so, Athenna will attempt to load the ",(0,i.jsx)(n.code,{children:".env.${env}"})," or\n",(0,i.jsx)(n.code,{children:".env.${APP_ENV}"})," file if it exists. If the file cannot be\nfound, Athenna will try to find the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," variable inside\nthe ",(0,i.jsx)(n.code,{children:".env"})," file and again try to find the ",(0,i.jsx)(n.code,{children:".env.${APP_ENV}"}),"\nfile to load. Finally, if the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," is not present in ",(0,i.jsx)(n.code,{children:".env"})," file,\nAthenna will end up loading the ",(0,i.jsx)(n.code,{children:".env"})," file by default."]}),"\n",(0,i.jsxs)(n.p,{children:["Running providing ",(0,i.jsx)(n.code,{children:"--env"})," flag in Artisan:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-bash",children:"node artisan serve --env=local\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Running providing ",(0,i.jsx)(n.code,{children:"APP_ENV"})," externally:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-bash",children:"APP_ENV=local node artisan serve\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Let's check some practical examples. This is the default\n",(0,i.jsx)(n.code,{children:".env"})," file that comes in Athenna project:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-dotenv",children:"HOST=localhost\nPORT=3000\n\nAPP_NAME=Athenna\nAPP_ENV=local\nAPP_DEBUG=true\nAPP_URL=http://${HOST}:${PORT}\nAPP_DOMAIN=${HOST}\nAPP_DOCUMENTATION=${APP_URL}\nAPP_SOURCE=https://github.com/AthennaIO\n\nLOG_HTTP=true\nLOG_CHANNEL=application\n"})}),"\n",(0,i.jsxs)(n.p,{children:["In the ",(0,i.jsx)(n.code,{children:".env"})," file above, you can see that we have the\n",(0,i.jsx)(n.code,{children:"APP_ENV=local"}),". This means that if you create a new\n",(0,i.jsx)(n.code,{children:".env.local"})," file in your project root path, Athenna\nwill load it instead of ",(0,i.jsx)(n.code,{children:".env"})," if running your application\nwithout a predefined environment."]}),"\n",(0,i.jsx)(n.h3,{id:"get-an-environment-variable-value",children:"Get an environment variable value"}),"\n",(0,i.jsxs)(n.p,{children:["You can get environment variables using the ",(0,i.jsx)(n.code,{children:"Env()"})," function:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Env } from '@athenna/config'\n\nconst defaultValue = 'Athenna'\n\nconst appName = Env('APP_NAME', defaultValue)\n"})}),"\n",(0,i.jsxs)(n.p,{children:["All environment variables in your .env file and inside\n",(0,i.jsx)(n.code,{children:"process.env"})," object are always interpreted as strings.\nBut when using the ",(0,i.jsx)(n.code,{children:"Env()"})," function, it will auto cast the\nvalue for you. Check the comparison:"]}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{style:{textAlign:"left"},children:"Value in .env"}),(0,i.jsx)(n.th,{style:{textAlign:"right"},children:"Value returned by Env function"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"true" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"10" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"10.090909" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['{"name":"Paulo"} ',(0,i.jsx)(n.strong,{children:"(json)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:['{ name: "Paulo" } ',(0,i.jsx)(n.strong,{children:"(Object)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"{"name":"Paulo"}" ',(0,i.jsx)(n.strong,{children:"(json string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:['{ name: "Paulo" } ',(0,i.jsx)(n.strong,{children:"(Object)"})]})]})]})]}),"\n",(0,i.jsx)(n.p,{children:"Let's see a more practical example of it:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"process.env.PORT = '3000'\nprocess.env.APP_DEBUG = 'true'\nprocess.env.APP_JSON = '{\"name\":\"Paulo\"}'\n\nconsole.log(Env('PORT')) // 3000 <- number\nconsole.log(Env('APP_DEBUG')) // true <- boolean\nconsole.log(Env('APP_JSON')) // { name: \"Paulo\" } <- object\n"})}),"\n",(0,i.jsxs)(n.p,{children:["There will certainly have scenarios in your business rule\nwhere you explicitly need an environment variable with value\n",(0,i.jsx)(n.code,{children:"true"}),", ",(0,i.jsx)(n.code,{children:"10"})," or ",(0,i.jsx)(n.code,{children:'{"name":"Paulo"}'})," to be a string. To solve\nthis, you can turn off the auto cast when using the ",(0,i.jsx)(n.code,{children:"Env()"}),"\nfunction:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"process.env.PORT = '3000'\nprocess.env.APP_DEBUG = 'true'\nprocess.env.APP_JSON = '{\"name\":\"Paulo\"}'\n\nconst autoCast = false\nconst defaultValue = undefined\n\nconsole.log(Env('PORT', defaultValue, autoCast)) // 3000 <- string\nconsole.log(Env('APP_DEBUG', defaultValue, autoCast)) // true <- string\nconsole.log(Env('APP_JSON', defaultValue, autoCast)) // {\"name\":\"Paulo\"} <- string\n"})}),"\n",(0,i.jsx)(n.p,{children:"Environment variables can parse other environment variables\ntoo. See the example above:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-dotenv",children:"HOST=localhost\nPORT=3000\n\nAPP_URL=http://${HOST}:${PORT}\n"})}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"console.log(Env('APP_URL')) // \"http://localhost:3000\"\n"})}),"\n",(0,i.jsx)(n.h2,{id:"custom-environment-file-path",children:"Custom environment file path"}),"\n",(0,i.jsxs)(n.p,{children:["You can also change the name and the path of your ",(0,i.jsx)(n.code,{children:".env"})," file.\nTo do that you need to set the new path to ",(0,i.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('dev.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n envPath: './bootstrap/.env.dev' \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsx)(n.p,{children:"Always remember that when using relative paths to set something\nin Athenna, you need to use your project root path as reference,\njust like in the example above."})}),"\n",(0,i.jsx)(n.h2,{id:"configuration-files",children:"Configuration files"}),"\n",(0,i.jsxs)(n.p,{children:["All the configuration files for the Athenna framework are\nstored in the ",(0,i.jsx)(n.code,{children:"src/config"})," directory. Each option is documented, so\nfeel free to look through the files and get familiar with\nthe options available to you."]}),"\n",(0,i.jsx)(n.p,{children:"Athenna needs almost no additional configuration out of\nthe box. You are free to get started developing! Each\noption is documented, so feel free to look through the\nfiles and get familiar with the options available to you.\nIt contains several options such as locale that you may\nwish to change, according to your application."}),"\n",(0,i.jsx)(n.h3,{id:"manipulating-configuration-values",children:"Manipulating configuration values"}),"\n",(0,i.jsxs)(n.p,{children:["You may easily access your configuration values using the\nglobal ",(0,i.jsx)(n.code,{children:"Config"})," helper class. The configuration values may\nbe accessed using ",(0,i.jsx)(n.code,{children:'"dot (.)"'})," syntax, which includes the\nname of the file and option you wish to access. Let's cover\nsome methods bellow:"]}),"\n",(0,i.jsx)(n.h4,{id:"configget",children:(0,i.jsx)(n.code,{children:"Config.get()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"get()"})," method will return the value of your configuration.\nYou can also set a default value as second parameter that will\nbe returned if the configuration option does not exist:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nconst defaultValue = 'Athenna'\nconst name = Config.get('app.name', defaultValue)\n\nconsole.log(name) // MyAppName\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"tip",children:[(0,i.jsxs)(n.p,{children:["You can get all the configuration values using ",(0,i.jsx)(n.code,{children:"get()"})," method\nwithout any key:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"console.log(Config.get()) // { app: {...}, http: {...}, ... }\n"})})]}),"\n",(0,i.jsx)(n.h4,{id:"configset",children:(0,i.jsx)(n.code,{children:"Config.set()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"set()"})," method is very useful to set or change the value\nof some configuration in runtime:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.set('app.name', 'Athenna Framework')\n\nconsole.log(Config.get('app.name')) // Athenna Framework\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"Config.set()"})," method does not change the values in\nthe configuration file, only in runtime. To do that, you\nwill need to use the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/configuration#configrewrite",children:"Config.rewrite()"}),"\nmethod."]})}),"\n",(0,i.jsx)(n.h4,{id:"configsafeset",children:(0,i.jsx)(n.code,{children:"Config.safeSet()"})}),"\n",(0,i.jsxs)(n.p,{children:["If you are not sure if some configuration value is already\nset of not, you can use the ",(0,i.jsx)(n.code,{children:"safeSet()"})," method instead to\nnot overwrite something that was already defined:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nconsole.log(Config.get('app.name')) // MyAppName\n\nConfig.safeSet('app.name', 'Athenna Framework')\n\nconsole.log(Config.get('app.name')) // MyAppName\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configdelete",children:(0,i.jsx)(n.code,{children:"Config.delete()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"delete()"})," method could be used to delete some configuration\nvalue:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.delete('app.name')\n\nconsole.log(Config.get('app.name')) // undefined\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsxs)(n.p,{children:["Just like ",(0,i.jsx)(n.code,{children:"Config.set()"})," method, ",(0,i.jsx)(n.code,{children:"Config.delete()"})," does\nnot change the values in the configuration file, only\nin runtime. To do that you will need to use the\n",(0,i.jsx)(n.code,{children:"Config.rewrite()"})," method."]})}),"\n",(0,i.jsx)(n.h4,{id:"configrewrite",children:(0,i.jsx)(n.code,{children:"Config.rewrite()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"rewrite()"})," method is very useful for rewriting the\nconfiguration file. Very useful when you want to\nprogrammatically modify the configuration file source code.\nThis method uses the ",(0,i.jsx)(n.a,{href:"https://github.com/unjs/magicast",children:"magicast"}),"\nlibrary under the hood to do that:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.set('app.name', 'Athenna Framework')\n\nawait Config.rewrite('app')\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"warning",children:[(0,i.jsxs)(n.p,{children:["Let's suppose that you want to set a function as a value,\nyou can use builders.functionCall function of ",(0,i.jsx)(n.a,{href:"https://github.com/unjs/magicast",children:"magicast"}),"\nlibrary to do that:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { builders } from 'magicast'\nimport { Config } from '@athenna/config'\n\nConfig.set('app.name', builders.functionCall('Env', ['MY_APP_NAME']))\n\nawait Config.rewrite('app')\n"})}),(0,i.jsxs)(n.p,{children:["The example above will produce the following code in ",(0,i.jsx)(s.A,{father:"config",child:"app.ts"}),":"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"export default {\n name: Env('MY_APP_NAME')\n ...\n}\n"})})]}),"\n",(0,i.jsx)(n.h4,{id:"configis",children:(0,i.jsx)(n.code,{children:"Config.is()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"is()"})," method could be used to validate if your\nconfiguration value matches some other value:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.is('app.name', 'Athenna')) {\n // do something\n}\n"})}),"\n",(0,i.jsxs)(n.p,{children:["You can set an array as second parameter to ",(0,i.jsx)(n.code,{children:"is()"})," method.\nIf any value in the array matches the configuration value,\nthe method will return true:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.is('app.name', ['Athenna', 'MyAppName'])) {\n // do something\n}\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["You can use the ",(0,i.jsx)(n.code,{children:"isNot()"})," method to do the negated validation."]})}),"\n",(0,i.jsx)(n.h4,{id:"configexistsall",children:(0,i.jsx)(n.code,{children:"Config.existsAll()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"existsAll()"})," method could be used to validate if an\narray of configuration keys exists or not:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.existsAll(['app.name', 'app.version'])) {\n // do something\n}\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["You can use the ",(0,i.jsx)(n.code,{children:"notExistsAll()"})," method to do the negate\nvalidation."]})}),"\n",(0,i.jsx)(n.h4,{id:"configclear",children:(0,i.jsx)(n.code,{children:"Config.clear()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"clear()"})," method could be used to clear all the configuration\nvalues:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.clear()\n\nconsole.log(Config.get()) // {}\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configload",children:(0,i.jsx)(n.code,{children:"Config.load()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"load()"})," method could be used to load some configuration\nfile path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst testConfig = Path.stubs('config/test.ts') // /path/to/your/project/tests/stubs/config/test.ts\nawait Config.load(testConfig)\n\nconsole.log(Config.get('test')) // { ... }\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configsafeload",children:(0,i.jsx)(n.code,{children:"Config.safeLoad()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"safeLoad()"})," method will only load the file path if it\nis not defined:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst appConfig = Path.stubs('config/app.js') // /path/to/your/project/tests/stubs/config/app.js\nawait Config.safeLoad(appConfig)\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configloadall",children:(0,i.jsx)(n.code,{children:"Config.loadAll()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"loadAll"})," method will load all files found inside some\nconfiguration path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst config = Path.stubs('config') // /path/to/your/project/tests/stubs/config\nawait Config.loadAll(config)\n"})}),"\n",(0,i.jsxs)(n.h3,{id:"get-configuration-with-value-decorator",children:["Get configuration with ",(0,i.jsx)(n.code,{children:"@Value()"})," decorator"]}),"\n",(0,i.jsxs)(n.p,{children:["Instead of using the ",(0,i.jsx)(n.code,{children:"Config.get()"})," method to get a configuration\nvalue, you can use the ",(0,i.jsx)(n.code,{children:"@Value()"})," annotation in your classes to\nautomatically add it value to a class property:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Value } from '@athenna/config'\n\nexport class UserService {\n @Value('api.users') \ud83d\udc48\n public api: string\n}\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"tip",children:[(0,i.jsxs)(n.p,{children:["Just like ",(0,i.jsx)(n.code,{children:"Config.get()"})," method, you can set a default value\nwhen using the ",(0,i.jsx)(n.code,{children:"@Value()"})," annotation:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"@Value('api.users', 'http://localhost:3000/users') \ud83d\udc48\n"})})]}),"\n",(0,i.jsx)(n.h3,{id:"define-my-own-configuration-path",children:"Define my own configuration path"}),"\n",(0,i.jsxs)(n.p,{children:["If you are building your own project structure you might want to\nchange the configurations directory from ",(0,i.jsx)(n.code,{children:"src/config"})," to something\nelse. For this case you can specify to Athenna a different path to\nwhat ",(0,i.jsx)(s.A,{fatherPath:"config"})," will resolve."]}),"\n",(0,i.jsxs)(n.p,{children:["To specify your application directories to Athenna, you can\nopen the ",(0,i.jsx)(n.code,{children:".athennarc.json"})," file and add the ",(0,i.jsx)(n.code,{children:"directories"}),"\nproperty to it. The ",(0,i.jsx)(n.code,{children:"directories"})," property is an object that maps the directory\nbase path that the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nhelper will use to resolve your application paths:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\n\nconsole.log(Path.config()) // /path/to/your/project/src/config\n"})}),"\n",(0,i.jsxs)(n.p,{children:["All the ",(0,i.jsx)(n.code,{children:"directories"})," key names follow the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nclass methods names. This means that if you want to change\nwhat is returned by the ",(0,i.jsx)(s.A,{fatherPath:"config"})," method, you will\nneed to add the ",(0,i.jsx)(n.code,{children:"config"})," key to the ",(0,i.jsx)(n.code,{children:"directories"})," object:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-json",children:'{\n "directories": {\n "config": "src/app/config"\n }\n}\n'})}),"\n",(0,i.jsxs)(n.p,{children:["Now when calling the ",(0,i.jsx)(s.A,{father:"config"})," method, it will return\na different path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\n\nconsole.log(Path.config()) // /path/to/your/project/src/app/config \ud83d\udc48\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Athenna always rely on ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nclass methods to find files and directories that are used\ninternally by the framework, like configuration file, route\nfiles, entry points and many others."]}),"\n",(0,i.jsxs)(n.p,{children:["Check ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/athennarc-file#the-directories-property",children:"the directories property documentation section"}),"\nfor more information about the ",(0,i.jsx)(n.code,{children:"directories"})," property. And\ncheck ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/directory-structure#do-your-own-structure",children:"the do your own structure documentation section"}),"\nfor more information about how to create your own project\nstructure."]}),"\n",(0,i.jsx)(n.h2,{id:"safe-loading-configuration-files",children:"Safe loading configuration files"}),"\n",(0,i.jsxs)(n.p,{children:["Athenna got multiple types of applications, while using the framework\nyou will notice that some times you could end up igniting your application\ntwice. Let's suppose you are using ",(0,i.jsx)(n.code,{children:"node artisan serve"})," command to start\nyour application, this command will first ignite your application by Artisan\nand them by the HTTP server."]}),"\n",(0,i.jsxs)(n.p,{children:["This is usually not a problem at all, but depending on how you have created\nyour environment it could become one. To avoid reloading configuration\nfiles in these situations, you can set the ",(0,i.jsx)(n.code,{children:"loadConfigSafe"})," option as ",(0,i.jsx)(n.code,{children:"true"}),"\nin ",(0,i.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('main.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n loadConfigSafe: true, \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,i.jsx)(n.h2,{id:"debug-mode",children:"Debug mode"}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"debug"})," option in your ",(0,i.jsx)(s.A,{path:"config",child:"app.ts"})," configuration\nfile determines how much information about your application\nis actually displayed to you and for who is going to consume\nyour application. By default, this option is set to respect\nthe value of the ",(0,i.jsx)(n.code,{children:"APP_DEBUG"})," environment variable, which is\nstored in your ",(0,i.jsx)(n.code,{children:".env"})," file."]}),"\n",(0,i.jsxs)(n.p,{children:["For local development, you should set the ",(0,i.jsx)(n.code,{children:"APP_DEBUG"}),"\nenvironment variable to ",(0,i.jsx)(n.code,{children:"true"}),". In your production environment,\nthis value should always be ",(0,i.jsx)(n.code,{children:"false"}),", if the variable is\nset to ",(0,i.jsx)(n.code,{children:"true"})," in production, you risk exposing sensitive\nconfiguration values to your application's end users."]})]})}function u(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},7049:(e,n,t)=>{t.d(n,{A:()=>s});t(6540);const i={hoverCardContainer:"hoverCardContainer_jqUQ",hoverCardLink:"hoverCardLink_oDZU",hoverCard:"hoverCard_qTDS"};var o=t(4848);function s(e){let n=e.father;switch(e.father){case"storage":n="src/storage";break;case"logs":n="src/storage/logs";break;case"views":n="src/resources/views";break;case"locales":n="src/resources/locales";break;case"static":n="src/resources/static";break;case"config":n="src/config";break;case"database":n="src/database";break;case"seeders":n="src/database/seeders";break;case"migrations":n="src/database/migrations";break;case"console":n="src/console";break;case"commands":n="src/console/commands";break;case"cron":n="src/cron";break;case"schedulers":n="src/cron/schedulers";break;case"models":n="src/models";break;case"services":n="src/services";break;case"repositories":n="src/repositories";break;case"http":n="src/http";break;case"controllers":n="src/http/controllers";break;case"middlewares":n="src/http/middlewares";break;case"interceptors":n="src/http/interceptors";break;case"terminators":n="src/http/terminators";break;case"stubs":n="tests/stubs";break;case"fixtures":n="tests/fixtures";break;case"providers":n="src/providers";break;case"facades":n="src/facades";break;case"routes":n="src/routes"}return(0,o.jsxs)("div",{className:i.hoverCardContainer,children:[(0,o.jsx)("a",{className:i.hoverCardLink,href:`/docs/the-basics/helpers#path${e.father}`,children:(0,o.jsxs)("code",{children:["Path.",e.father,"(",e.child?`'${e.child}'`:"",")"]})}),(0,o.jsx)("div",{className:i.hoverCard,children:(0,o.jsxs)("p",{style:{margin:0},children:["./",n,e.child?`/${e.child}`:""]})})]})}},8453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>a});var i=t(6540);const o={},s=i.createContext(o);function r(e){const n=i.useContext(s);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:r(e.components),i.createElement(s.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/30d27832.8c09266b.js b/assets/js/30d27832.8c09266b.js deleted file mode 100644 index 5be81c7b..00000000 --- a/assets/js/30d27832.8c09266b.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[4890],{9604:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>a,default:()=>u,frontMatter:()=>r,metadata:()=>c,toc:()=>d});var i=t(4848),o=t(8453),s=t(7049);const r={title:"Configuration",sidebar_position:4,description:"Understand the initial configurations of your project."},a="Configuration",c={id:"getting-started/configuration",title:"Configuration",description:"Understand the initial configurations of your project.",source:"@site/docs/getting-started/configuration.mdx",sourceDirName:"getting-started",slug:"/getting-started/configuration",permalink:"/docs/getting-started/configuration",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/configuration.mdx",tags:[],version:"current",sidebarPosition:4,frontMatter:{title:"Configuration",sidebar_position:4,description:"Understand the initial configurations of your project."},sidebar:"tutorialSidebar",previous:{title:"AthennaRC File",permalink:"/docs/getting-started/athennarc-file"},next:{title:"Directory Structure",permalink:"/docs/getting-started/directory-structure"}},l={},d=[{value:"Environment configuration",id:"environment-configuration",level:2},{value:"Determining the current environment",id:"determining-the-current-environment",level:3},{value:"Get an environment variable value",id:"get-an-environment-variable-value",level:3},{value:"Custom environment file path",id:"custom-environment-file-path",level:2},{value:"Configuration files",id:"configuration-files",level:2},{value:"Manipulating configuration values",id:"manipulating-configuration-values",level:3},{value:"Config.get()",id:"configget",level:4},{value:"Config.set()",id:"configset",level:4},{value:"Config.safeSet()",id:"configsafeset",level:4},{value:"Config.delete()",id:"configdelete",level:4},{value:"Config.rewrite()",id:"configrewrite",level:4},{value:"Config.is()",id:"configis",level:4},{value:"Config.existsAll()",id:"configexistsall",level:4},{value:"Config.clear()",id:"configclear",level:4},{value:"Config.load()",id:"configload",level:4},{value:"Config.safeLoad()",id:"configsafeload",level:4},{value:"Config.loadAll()",id:"configloadall",level:4},{value:"Get configuration with @Value() decorator",id:"get-configuration-with-value-decorator",level:3},{value:"Define my own configuration path",id:"define-my-own-configuration-path",level:3},{value:"Safe loading configuration files",id:"safe-loading-configuration-files",level:2},{value:"Debug mode",id:"debug-mode",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.header,{children:(0,i.jsx)(n.h1,{id:"configuration",children:"Configuration"})}),"\n",(0,i.jsx)(n.p,{children:"Understand the initial configurations of your project."}),"\n",(0,i.jsx)(n.h2,{id:"environment-configuration",children:"Environment configuration"}),"\n",(0,i.jsx)(n.p,{children:"It is often helpful to have different configuration values\nbased on the environment where the application is running.\nFor example, you may wish to use a different storage driver\nlocally than you do on your production server."}),"\n",(0,i.jsxs)(n.p,{children:["To make this a cinch, Athenna utilizes the ",(0,i.jsx)(n.a,{href:"https://www.npmjs.com/package/dotenv",children:"dotenv"}),"\nNode.js library. In a fresh Athenna installation, the root\ndirectory of your application will contain a ",(0,i.jsx)(n.code,{children:".env.example"})," file\nthat defines many common environment variables. During the\nAthenna installation process, this file will automatically\nbe copied to ",(0,i.jsx)(n.code,{children:".env"})," and ",(0,i.jsx)(n.code,{children:".env.test"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Athenna default ",(0,i.jsx)(n.code,{children:".env"})," file contains some common configuration\nvalues that may differ based on whether your application is\nrunning locally or on a production. These values are then\nretrieved from various Athenna configuration files within the\n",(0,i.jsx)(n.code,{children:"config"})," directory using Athenna ",(0,i.jsx)(n.code,{children:"Env()"})," function."]}),"\n",(0,i.jsxs)(n.p,{children:["If you are developing with a team, you may wish to continue\nincluding a ",(0,i.jsx)(n.code,{children:".env.example"})," file with your application. By\nputting placeholder values in the example configuration file,\nother developers on your team can clearly see which environment\nvariables are needed to run your application."]}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["Any variable in your ",(0,i.jsx)(n.code,{children:".env"})," file can be overridden by\nexternal environment variables such as server-level or\nsystem-level environment variables. But off course you\ncan turn off this behavior setting the ",(0,i.jsx)(n.code,{children:"OVERRIDE_ENV=true"}),"\nvariable before running your application, if this variable\nis set to true, all environment variables set in ",(0,i.jsx)(n.code,{children:".env"}),"\nwill override the externals."]})}),"\n",(0,i.jsx)(n.admonition,{type:"caution",children:(0,i.jsxs)(n.p,{children:["Your ",(0,i.jsx)(n.code,{children:".env"})," file should not be committed to your application\nsource control, since each developer/server using your\napplication could require a different environment configuration.\nFurthermore, this would be a security risk in the event an\nintruder gains access to your source control repository,\nsince any sensitive credentials would get exposed."]})}),"\n",(0,i.jsx)(n.h3,{id:"determining-the-current-environment",children:"Determining the current environment"}),"\n",(0,i.jsxs)(n.p,{children:["Before loading your application's environment variables,\nAthenna determines if either the ",(0,i.jsx)(n.code,{children:"--env"})," Artisan flag has\nbeen specified or if the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," environment variable has\nbeen externally provided."]}),"\n",(0,i.jsxs)(n.p,{children:["If so, Athenna will attempt to load the ",(0,i.jsx)(n.code,{children:".env.${env}"})," or\n",(0,i.jsx)(n.code,{children:".env.${APP_ENV}"})," file if it exists. If the file cannot be\nfound, Athenna will try to find the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," variable inside\nthe ",(0,i.jsx)(n.code,{children:".env"})," file and again try to find the ",(0,i.jsx)(n.code,{children:".env.${APP_ENV}"}),"\nfile to load. Finally, if the ",(0,i.jsx)(n.code,{children:"APP_ENV"})," is not present in ",(0,i.jsx)(n.code,{children:".env"})," file,\nAthenna will end up loading the ",(0,i.jsx)(n.code,{children:".env"})," file by default."]}),"\n",(0,i.jsxs)(n.p,{children:["Running providing ",(0,i.jsx)(n.code,{children:"--env"})," flag in Artisan:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-bash",children:"node artisan serve --env=local\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Running providing ",(0,i.jsx)(n.code,{children:"APP_ENV"})," externally:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-bash",children:"APP_ENV=local node artisan serve\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Let's check some practical examples. This is the default\n",(0,i.jsx)(n.code,{children:".env"})," file that comes in Athenna project:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-dotenv",children:"HOST=localhost\nPORT=3000\n\nAPP_NAME=Athenna\nAPP_ENV=local\nAPP_DEBUG=true\nAPP_URL=http://${HOST}:${PORT}\nAPP_DOMAIN=${HOST}\nAPP_DOCUMENTATION=${APP_URL}\nAPP_SOURCE=https://github.com/AthennaIO\n\nLOG_HTTP=true\nLOG_CHANNEL=application\n"})}),"\n",(0,i.jsxs)(n.p,{children:["In the ",(0,i.jsx)(n.code,{children:".env"})," file above, you can see that we have the\n",(0,i.jsx)(n.code,{children:"APP_ENV=local"}),". This means that if you create a new\n",(0,i.jsx)(n.code,{children:".env.local"})," file in your project root path, Athenna\nwill load it instead of ",(0,i.jsx)(n.code,{children:".env"})," if running your application\nwithout a predefined environment."]}),"\n",(0,i.jsx)(n.h3,{id:"get-an-environment-variable-value",children:"Get an environment variable value"}),"\n",(0,i.jsxs)(n.p,{children:["You can get environment variables using the ",(0,i.jsx)(n.code,{children:"Env()"})," function:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Env } from '@athenna/config'\n\nconst defaultValue = 'Athenna'\n\nconst appName = Env('APP_NAME', defaultValue)\n"})}),"\n",(0,i.jsxs)(n.p,{children:["All environment variables in your .env file and inside\n",(0,i.jsx)(n.code,{children:"process.env"})," object are always interpreted as strings.\nBut when using the ",(0,i.jsx)(n.code,{children:"Env()"})," function, it will auto cast the\nvalue for you. Check the comparison:"]}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{style:{textAlign:"left"},children:"Value in .env"}),(0,i.jsx)(n.th,{style:{textAlign:"right"},children:"Value returned by Env function"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"true" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["true ",(0,i.jsx)(n.strong,{children:"(boolean)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"10" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10 ",(0,i.jsx)(n.strong,{children:"(number)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"10.090909" ',(0,i.jsx)(n.strong,{children:"(string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:["10.090909 ",(0,i.jsx)(n.strong,{children:"(float)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['{"name":"Paulo"} ',(0,i.jsx)(n.strong,{children:"(json)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:['{ name: "Paulo" } ',(0,i.jsx)(n.strong,{children:"(Object)"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsxs)(n.td,{style:{textAlign:"left"},children:['"{"name":"Paulo"}" ',(0,i.jsx)(n.strong,{children:"(json string)"})]}),(0,i.jsxs)(n.td,{style:{textAlign:"right"},children:['{ name: "Paulo" } ',(0,i.jsx)(n.strong,{children:"(Object)"})]})]})]})]}),"\n",(0,i.jsx)(n.p,{children:"Let's see a more practical example of it:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"process.env.PORT = '3000'\nprocess.env.APP_DEBUG = 'true'\nprocess.env.APP_JSON = '{\"name\":\"Paulo\"}'\n\nconsole.log(Env('PORT')) // 3000 <- number\nconsole.log(Env('APP_DEBUG')) // true <- boolean\nconsole.log(Env('APP_JSON')) // { name: \"Paulo\" } <- object\n"})}),"\n",(0,i.jsxs)(n.p,{children:["There will certainly have scenarios in your business rule\nwhere you explicitly need an environment variable with value\n",(0,i.jsx)(n.code,{children:"true"}),", ",(0,i.jsx)(n.code,{children:"10"})," or ",(0,i.jsx)(n.code,{children:'{"name":"Paulo"}'})," to be a string. To solve\nthis, you can turn off the auto cast when using the ",(0,i.jsx)(n.code,{children:"Env()"}),"\nfunction:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"process.env.PORT = '3000'\nprocess.env.APP_DEBUG = 'true'\nprocess.env.APP_JSON = '{\"name\":\"Paulo\"}'\n\nconst autoCast = false\nconst defaultValue = undefined\n\nconsole.log(Env('PORT', defaultValue, autoCast)) // 3000 <- string\nconsole.log(Env('APP_DEBUG', defaultValue, autoCast)) // true <- string\nconsole.log(Env('APP_JSON', defaultValue, autoCast)) // {\"name\":\"Paulo\"} <- string\n"})}),"\n",(0,i.jsx)(n.p,{children:"Environment variables can parse other environment variables\ntoo. See the example above:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-dotenv",children:"HOST=localhost\nPORT=3000\n\nAPP_URL=http://${HOST}:${PORT}\n"})}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"console.log(Env('APP_URL')) // \"http://localhost:3000\"\n"})}),"\n",(0,i.jsx)(n.h2,{id:"custom-environment-file-path",children:"Custom environment file path"}),"\n",(0,i.jsxs)(n.p,{children:["You can also change the name and the path of your ",(0,i.jsx)(n.code,{children:".env"})," file.\nTo do that you need to set the new path to ",(0,i.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('dev.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n envPath: './bootstrap/.env.dev' \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsx)(n.p,{children:"Always remember that when using relative paths to set something\nin Athenna, you need to use your project root path as reference,\njust like in the example above."})}),"\n",(0,i.jsx)(n.h2,{id:"configuration-files",children:"Configuration files"}),"\n",(0,i.jsxs)(n.p,{children:["All the configuration files for the Athenna framework are\nstored in the ",(0,i.jsx)(n.code,{children:"src/config"})," directory. Each option is documented, so\nfeel free to look through the files and get familiar with\nthe options available to you."]}),"\n",(0,i.jsx)(n.p,{children:"Athenna needs almost no additional configuration out of\nthe box. You are free to get started developing! Each\noption is documented, so feel free to look through the\nfiles and get familiar with the options available to you.\nIt contains several options such as locale that you may\nwish to change, according to your application."}),"\n",(0,i.jsx)(n.h3,{id:"manipulating-configuration-values",children:"Manipulating configuration values"}),"\n",(0,i.jsxs)(n.p,{children:["You may easily access your configuration values using the\nglobal ",(0,i.jsx)(n.code,{children:"Config"})," helper class. The configuration values may\nbe accessed using ",(0,i.jsx)(n.code,{children:'"dot (.)"'})," syntax, which includes the\nname of the file and option you wish to access. Let's cover\nsome methods bellow:"]}),"\n",(0,i.jsx)(n.h4,{id:"configget",children:(0,i.jsx)(n.code,{children:"Config.get()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"get()"})," method will return the value of your configuration.\nYou can also set a default value as second parameter that will\nbe returned if the configuration option does not exist:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nconst defaultValue = 'Athenna'\nconst name = Config.get('app.name', defaultValue)\n\nconsole.log(name) // MyAppName\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"tip",children:[(0,i.jsxs)(n.p,{children:["You can get all the configuration values using ",(0,i.jsx)(n.code,{children:"get()"})," method\nwithout any key:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"console.log(Config.get()) // { app: {...}, http: {...}, ... }\n"})})]}),"\n",(0,i.jsx)(n.h4,{id:"configset",children:(0,i.jsx)(n.code,{children:"Config.set()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"set()"})," method is very useful to set or change the value\nof some configuration in runtime:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.set('app.name', 'Athenna Framework')\n\nconsole.log(Config.get('app.name')) // Athenna Framework\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"Config.set()"})," method does not change the values in\nthe configuration file, only in runtime. To do that, you\nwill need to use the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/configuration#configrewrite",children:"Config.rewrite()"}),"\nmethod."]})}),"\n",(0,i.jsx)(n.h4,{id:"configsafeset",children:(0,i.jsx)(n.code,{children:"Config.safeSet()"})}),"\n",(0,i.jsxs)(n.p,{children:["If you are not sure if some configuration value is already\nset of not, you can use the ",(0,i.jsx)(n.code,{children:"safeSet()"})," method instead to\nnot overwrite something that was already defined:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nconsole.log(Config.get('app.name')) // MyAppName\n\nConfig.safeSet('app.name', 'Athenna Framework')\n\nconsole.log(Config.get('app.name')) // MyAppName\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configdelete",children:(0,i.jsx)(n.code,{children:"Config.delete()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"delete()"})," method could be used to delete some configuration\nvalue:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.delete('app.name')\n\nconsole.log(Config.get('app.name')) // undefined\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsxs)(n.p,{children:["Just like ",(0,i.jsx)(n.code,{children:"Config.set()"})," method, ",(0,i.jsx)(n.code,{children:"Config.delete()"})," does\nnot change the values in the configuration file, only\nin runtime. To do that you will need to use the\n",(0,i.jsx)(n.code,{children:"Config.rewrite()"})," method."]})}),"\n",(0,i.jsx)(n.h4,{id:"configrewrite",children:(0,i.jsx)(n.code,{children:"Config.rewrite()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"rewrite()"})," method is very useful for rewriting the\nconfiguration file. Very useful when you want to\nprogrammatically modify the configuration file source code.\nThis method uses the ",(0,i.jsx)(n.a,{href:"https://github.com/unjs/magicast",children:"magicast"}),"\nlibrary under the hood to do that:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.set('app.name', 'Athenna Framework')\n\nawait Config.rewrite('app')\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"warning",children:[(0,i.jsxs)(n.p,{children:["Let's suppose that you want to set a function as a value,\nyou can use builders.functionCall function of ",(0,i.jsx)(n.a,{href:"https://github.com/unjs/magicast",children:"magicast"}),"\nlibrary to do that:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { builders } from 'magicast'\nimport { Config } from '@athenna/config'\n\nConfig.set('app.name', builders.functionCall('Env', ['MY_APP_NAME']))\n\nawait Config.rewrite('app')\n"})}),(0,i.jsxs)(n.p,{children:["The example above will produce the following code in ",(0,i.jsx)(s.A,{father:"config",child:"app.ts"}),":"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"export default {\n name: Env('MY_APP_NAME')\n ...\n}\n"})})]}),"\n",(0,i.jsx)(n.h4,{id:"configis",children:(0,i.jsx)(n.code,{children:"Config.is()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"is()"})," method could be used to validate if your\nconfiguration value matches some other value:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.is('app.name', 'Athenna')) {\n // do something\n}\n"})}),"\n",(0,i.jsxs)(n.p,{children:["You can set an array as second parameter to ",(0,i.jsx)(n.code,{children:"is()"})," method.\nIf any value in the array matches the configuration value,\nthe method will return true:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.is('app.name', ['Athenna', 'MyAppName'])) {\n // do something\n}\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["You can use the ",(0,i.jsx)(n.code,{children:"isNot()"})," method to do the negated validation."]})}),"\n",(0,i.jsx)(n.h4,{id:"configexistsall",children:(0,i.jsx)(n.code,{children:"Config.existsAll()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"existsAll()"})," method could be used to validate if an\narray of configuration keys exists or not:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nif (Config.existsAll(['app.name', 'app.version'])) {\n // do something\n}\n"})}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsxs)(n.p,{children:["You can use the ",(0,i.jsx)(n.code,{children:"notExistsAll()"})," method to do the negate\nvalidation."]})}),"\n",(0,i.jsx)(n.h4,{id:"configclear",children:(0,i.jsx)(n.code,{children:"Config.clear()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"clear()"})," method could be used to clear all the configuration\nvalues:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Config } from '@athenna/config'\n\nConfig.clear()\n\nconsole.log(Config.get()) // {}\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configload",children:(0,i.jsx)(n.code,{children:"Config.load()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"load()"})," method could be used to load some configuration\nfile path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst testConfig = Path.stubs('config/test.ts') // /path/to/your/project/tests/stubs/config/test.ts\nawait Config.load(testConfig)\n\nconsole.log(Config.get('test')) // { ... }\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configsafeload",children:(0,i.jsx)(n.code,{children:"Config.safeLoad()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"safeLoad()"})," method will only load the file path if it\nis not defined:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst appConfig = Path.stubs('config/app.js') // /path/to/your/project/tests/stubs/config/app.js\nawait Config.safeLoad(appConfig)\n"})}),"\n",(0,i.jsx)(n.h4,{id:"configloadall",children:(0,i.jsx)(n.code,{children:"Config.loadAll()"})}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"loadAll"})," method will load all files found inside some\nconfiguration path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\nimport { Config } from '@athenna/config'\n\nconst config = Path.stubs('config') // /path/to/your/project/tests/stubs/config\nawait Config.loadAll(config)\n"})}),"\n",(0,i.jsxs)(n.h3,{id:"get-configuration-with-value-decorator",children:["Get configuration with ",(0,i.jsx)(n.code,{children:"@Value()"})," decorator"]}),"\n",(0,i.jsxs)(n.p,{children:["Instead of using the ",(0,i.jsx)(n.code,{children:"Config.get()"})," method to get a configuration\nvalue, you can use the ",(0,i.jsx)(n.code,{children:"@Value()"})," annotation in your classes to\nautomatically add it value to a class property:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Value } from '@athenna/config'\n\nexport class UserService {\n @Value('api.users') \ud83d\udc48\n public api: string\n}\n"})}),"\n",(0,i.jsxs)(n.admonition,{type:"tip",children:[(0,i.jsxs)(n.p,{children:["Just like ",(0,i.jsx)(n.code,{children:"Config.get()"})," method, you can set a default value\nwhen using the ",(0,i.jsx)(n.code,{children:"@Value()"})," annotation:"]}),(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"@Value('api.users', 'http://localhost:3000/users') \ud83d\udc48\n"})})]}),"\n",(0,i.jsx)(n.h3,{id:"define-my-own-configuration-path",children:"Define my own configuration path"}),"\n",(0,i.jsxs)(n.p,{children:["If you are building your own project structure you might want to\nchange the configurations directory from ",(0,i.jsx)(n.code,{children:"src/config"})," to something\nelse. For this case you can specify to Athenna a different path to\nwhat ",(0,i.jsx)(s.A,{fatherPath:"config"})," will resolve."]}),"\n",(0,i.jsxs)(n.p,{children:["To specify your application directories to Athenna, you can\nopen the ",(0,i.jsx)(n.code,{children:".athennarc.json"})," file and add the ",(0,i.jsx)(n.code,{children:"directories"}),"\nproperty to it. The ",(0,i.jsx)(n.code,{children:"directories"})," property is an object that maps the directory\nbase path that the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nhelper will use to resolve your application paths:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\n\nconsole.log(Path.config()) // /path/to/your/project/src/config\n"})}),"\n",(0,i.jsxs)(n.p,{children:["All the ",(0,i.jsx)(n.code,{children:"directories"})," key names follow the ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nclass methods names. This means that if you want to change\nwhat is returned by the ",(0,i.jsx)(s.A,{fatherPath:"config"})," method, you will\nneed to add the ",(0,i.jsx)(n.code,{children:"config"})," key to the ",(0,i.jsx)(n.code,{children:"directories"})," object:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-json",children:'{\n "directories": {\n "config": "src/app/config"\n }\n}\n'})}),"\n",(0,i.jsxs)(n.p,{children:["Now when calling the ",(0,i.jsx)(s.A,{father:"config"})," method, it will return\na different path:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",children:"import { Path } from '@athenna/common'\n\nconsole.log(Path.config()) // /path/to/your/project/src/app/config \ud83d\udc48\n"})}),"\n",(0,i.jsxs)(n.p,{children:["Athenna always rely on ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,i.jsx)(n.code,{children:"Path"})}),"\nclass methods to find files and directories that are used\ninternally by the framework, like configuration file, route\nfiles, entry points and many others."]}),"\n",(0,i.jsxs)(n.p,{children:["Check ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/athennarc-file#the-directories-property",children:"the directories property documentation section"}),"\nfor more information about the ",(0,i.jsx)(n.code,{children:"directories"})," property. And\ncheck ",(0,i.jsx)(n.a,{href:"https://athenna.io/docs/getting-started/directory-structure#do-your-own-structure",children:"the do your own structure documentation section"}),"\nfor more information about how to create your own project\nstructure."]}),"\n",(0,i.jsx)(n.h2,{id:"safe-loading-configuration-files",children:"Safe loading configuration files"}),"\n",(0,i.jsxs)(n.p,{children:["Athenna got multiple types of applications, while using the framework\nyou will notice that some times you could end up igniting your application\ntwice. Let's suppose you are using ",(0,i.jsx)(n.code,{children:"node artisan serve"})," command to start\nyour application, this command will first ignite your application by Artisan\nand them by the HTTP server."]}),"\n",(0,i.jsxs)(n.p,{children:["This is usually not a problem at all, but depending on how you have created\nyour environment it could become one. To avoid reloading configuration\nfiles in these situations, you can set the ",(0,i.jsx)(n.code,{children:"loadConfigSafe"})," option as ",(0,i.jsx)(n.code,{children:"true"}),"\nin ",(0,i.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('main.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n loadConfigSafe: true, \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,i.jsx)(n.h2,{id:"debug-mode",children:"Debug mode"}),"\n",(0,i.jsxs)(n.p,{children:["The ",(0,i.jsx)(n.code,{children:"debug"})," option in your ",(0,i.jsx)(s.A,{path:"config",child:"app.ts"})," configuration\nfile determines how much information about your application\nis actually displayed to you and for who is going to consume\nyour application. By default, this option is set to respect\nthe value of the ",(0,i.jsx)(n.code,{children:"APP_DEBUG"})," environment variable, which is\nstored in your ",(0,i.jsx)(n.code,{children:".env"})," file."]}),"\n",(0,i.jsxs)(n.p,{children:["For local development, you should set the ",(0,i.jsx)(n.code,{children:"APP_DEBUG"}),"\nenvironment variable to ",(0,i.jsx)(n.code,{children:"true"}),". In your production environment,\nthis value should always be ",(0,i.jsx)(n.code,{children:"false"}),", if the variable is\nset to ",(0,i.jsx)(n.code,{children:"true"})," in production, you risk exposing sensitive\nconfiguration values to your application's end users."]})]})}function u(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},7049:(e,n,t)=>{t.d(n,{A:()=>s});t(6540);const i={hoverCardContainer:"hoverCardContainer_jqUQ",hoverCardLink:"hoverCardLink_oDZU",hoverCard:"hoverCard_qTDS"};var o=t(4848);function s(e){let n=e.father;switch(e.father){case"storage":n="src/storage";break;case"logs":n="src/storage/logs";break;case"views":n="src/resources/views";break;case"locales":n="src/resources/locales";break;case"static":n="src/resources/static";break;case"config":n="src/config";break;case"database":n="src/database";break;case"seeders":n="src/database/seeders";break;case"migrations":n="src/database/migrations";break;case"console":n="src/console";break;case"commands":n="src/console/commands";break;case"cron":n="src/cron";break;case"schedulers":n="src/cron/schedulers";break;case"models":n="src/models";break;case"services":n="src/services";break;case"repositories":n="src/repositories";break;case"http":n="src/http";break;case"controllers":n="src/http/controllers";break;case"middlewares":n="src/http/middlewares";break;case"interceptors":n="src/http/interceptors";break;case"terminators":n="src/http/terminators";break;case"stubs":n="tests/stubs";break;case"fixtures":n="tests/fixtures";break;case"providers":n="src/providers";break;case"facades":n="src/facades";break;case"routes":n="src/routes"}return(0,o.jsxs)("div",{className:i.hoverCardContainer,children:[(0,o.jsx)("a",{className:i.hoverCardLink,href:`/docs/the-basics/helpers#path${e.father}`,children:(0,o.jsxs)("code",{children:["Path.",e.father,"(",e.child?`'${e.child}'`:"",")"]})}),(0,o.jsx)("div",{className:i.hoverCard,children:(0,o.jsxs)("p",{style:{margin:0},children:["./",n,e.child?`/${e.child}`:""]})})]})}},8453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>a});var i=t(6540);const o={},s=i.createContext(o);function r(e){const n=i.useContext(s);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:r(e.components),i.createElement(s.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/6459b84b.41354f80.js b/assets/js/6459b84b.41354f80.js deleted file mode 100644 index 487eb16e..00000000 --- a/assets/js/6459b84b.41354f80.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[6459],{7112:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>s,default:()=>u,frontMatter:()=>r,metadata:()=>c,toc:()=>h});var a=t(4848),i=t(8453),o=t(7049);const r={title:"Installation",sidebar_position:1,description:"How to install and set up your first Athenna project."},s="Installation",c={id:"getting-started/installation",title:"Installation",description:"How to install and set up your first Athenna project.",source:"@site/docs/getting-started/installation.mdx",sourceDirName:"getting-started",slug:"/getting-started/installation",permalink:"/docs/getting-started/installation",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/installation.mdx",tags:[],version:"current",sidebarPosition:1,frontMatter:{title:"Installation",sidebar_position:1,description:"How to install and set up your first Athenna project."},sidebar:"tutorialSidebar",next:{title:"AthennaRC File",permalink:"/docs/getting-started/athennarc-file"}},l={},h=[{value:"Meet Athenna",id:"meet-athenna",level:2},{value:"Why Athenna?",id:"why-athenna",level:2},{value:"A progressive framework",id:"a-progressive-framework",level:3},{value:"A scalable framework",id:"a-scalable-framework",level:3},{value:"An agnostic framework",id:"an-agnostic-framework",level:3},{value:"A community framework",id:"a-community-framework",level:3},{value:"Your first Athenna project",id:"your-first-athenna-project",level:2},{value:"Prerequisites",id:"prerequisites",level:3},{value:"Installing via package manager",id:"installing-via-package-manager",level:3},{value:"Application type",id:"application-type",level:4},{value:"Running your application",id:"running-your-application",level:2},{value:"What the \ud83e\udd2c is Path.bin('main.ts')?",id:"what-the--is-pathbinmaints",level:3},{value:"Initial configuration",id:"initial-configuration",level:2},{value:"Environment based configuration",id:"environment-based-configuration",level:2},{value:"Workspace and runtime configurations",id:"workspace-and-runtime-configurations",level:2},{value:"Next steps",id:"next-steps",level:2},{value:"REST API application",id:"rest-api-application",level:3},{value:"Command Line Interface (CLI) application",id:"command-line-interface-cli-application",level:3},{value:"CRON Job application",id:"cron-job-application",level:3}];function d(e){const n={a:"a",admonition:"admonition",blockquote:"blockquote",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",li:"li",ol:"ol",p:"p",pre:"pre",ul:"ul",...(0,i.R)(),...e.components};return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(n.header,{children:(0,a.jsx)(n.h1,{id:"installation",children:"Installation"})}),"\n",(0,a.jsx)(n.p,{children:"How to install and set up your first Athenna project."}),"\n",(0,a.jsx)(n.h2,{id:"meet-athenna",children:"Meet Athenna"}),"\n",(0,a.jsx)(n.p,{children:"Athenna is a framework written specially for the Node.js runtime\nwith expressive and elegant syntax. The framework provides a reliable\ncode foundation and starting point for creating your application,\nallowing you to focus on creating something amazing while we sweat the\ndetails."}),"\n",(0,a.jsx)(n.p,{children:"Athenna strives to provide an amazing developer experience while\nproviding powerful features such as thorough dependency injection,\nan expressive database abstraction layer, unit and e2e testing,\nand more."}),"\n",(0,a.jsxs)(n.p,{children:["Whether you are new to Node.js ecosystem or have years of experience,\nAthenna is a framework that can grow with you. We'll help you take\nyour first steps as a developer or give you a boost as you take your\nexpertise to the next level. We can't wait to see what you build, make\nsure to share it with us in our ",(0,a.jsx)(n.a,{href:"https://discord.gg/JdEbBAKw6X",children:"Discord community"}),"."]}),"\n",(0,a.jsx)(n.admonition,{type:"tip",children:(0,a.jsxs)(n.p,{children:["New to Athenna? Check out the ",(0,a.jsx)(n.a,{href:"https://www.youtube.com/@athennaio",children:"Athenna YouTube channel"}),"\nfor an infinity of hands-on videos!"]})}),"\n",(0,a.jsx)(n.h2,{id:"why-athenna",children:"Why Athenna?"}),"\n",(0,a.jsx)(n.p,{children:"There are a variety of tools and frameworks available to you when building\na software using Node.js. However, we believe Athenna is the best choice for\nbuilding modern microservices."}),"\n",(0,a.jsx)(n.h3,{id:"a-progressive-framework",children:"A progressive framework"}),"\n",(0,a.jsx)(n.p,{children:"We like to call Athenna a \"progressive\" framework. By that, we mean that\nAthenna grows with you. If you're just taking your first steps into backend\ndevelopment, Athenna's vast library of documentation, guides, and video\ntutorials will help you learn the ropes without becoming overwhelmed."}),"\n",(0,a.jsx)(n.p,{children:"If you're a senior developer, Athenna gives you robust tools for dependency\ninjection, unit and e2e testing, database abstracted layer and more.\nAthenna is fine-tuned for building professional applications and ready to\nhandle enterprise work loads."}),"\n",(0,a.jsx)(n.h3,{id:"a-scalable-framework",children:"A scalable framework"}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is incredibly scalable. Thanks to the scaling-friendly nature of\nNode.js and great libraries like ",(0,a.jsx)(n.a,{href:"https://fastify.dev/",children:"Fastify"}),", horizontal\nscaling with Athenna is a breeze. In fact, Athenna REST API's have been\neasily scaled to handle hundreds of millions of requests per month."]}),"\n",(0,a.jsx)(n.h3,{id:"an-agnostic-framework",children:"An agnostic framework"}),"\n",(0,a.jsx)(n.p,{children:"Athenna is perpect for building modern software using microservices architecture.\nNo matter what type of application you are creating, be it a REST API, a CLI, a CRON Job,\nAthenna has a reliable solution foundation for each occasion, the only thing you will\nneed to take care is how your application will communicate with the external world."}),"\n",(0,a.jsx)(n.h3,{id:"a-community-framework",children:"A community framework"}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is not reinventing the whell, the framework combines the best packages\nin the Node.js ecosystem to offer the most robust and developer friendly\nframework. We are always seeking for talented developers like you to\nhelp us by ",(0,a.jsx)(n.a,{href:"https://github.com/athennaio",children:"contributing to the framework"})," and\nmake it even better."]}),"\n",(0,a.jsx)(n.h2,{id:"your-first-athenna-project",children:"Your first Athenna project"}),"\n",(0,a.jsx)(n.h3,{id:"prerequisites",children:"Prerequisites"}),"\n",(0,a.jsxs)(n.p,{children:["First, you need to install ",(0,a.jsx)(n.a,{href:"https://nodejs.org/",children:"Node.js"}),".\nWe recommend using ",(0,a.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm",children:"nvm"})," to do that."]}),"\n",(0,a.jsxs)(n.blockquote,{children:["\n",(0,a.jsx)(n.p,{children:(0,a.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm#installing-and-updating",children:"Click here to install nvm and get npm and Node.js\nrunning on your machine."})}),"\n"]}),"\n",(0,a.jsxs)(n.p,{children:["After you install ",(0,a.jsx)(n.code,{children:"nvm"}),", we recommend you to install\nNode.js v20.x or above, but you can still use\nNode.js v16.x and above. Install Node.js v20.x by running:"]}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"nvm install 20\n"})}),"\n",(0,a.jsxs)(n.p,{children:["We recommend setting Node.js v20.x as the default version, to do\nso with ",(0,a.jsx)(n.code,{children:"nvm"})," run:"]}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"nvm alias default 20.8.1\n"})}),"\n",(0,a.jsx)(n.h3,{id:"installing-via-package-manager",children:"Installing via package manager"}),"\n",(0,a.jsx)(n.p,{children:"We want it to be as easy as possible to get started with\nAthenna. With that in mind, we developed a CLI to assist\nin the creation of a new project."}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"npm install -g @athenna/cli\n"})}),"\n",(0,a.jsx)(n.p,{children:"Then you can run this command to generate your project:"}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"athenna new my-project-name\n"})}),"\n",(0,a.jsx)(n.p,{children:"The installation process prompts for the following selections:"}),"\n",(0,a.jsx)(n.h4,{id:"application-type",children:"Application type"}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"REST Api"})," application is ideal for creating a Http server\nusing REST architecture."]}),"\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"CLI"})," application is ideal for creating global CLI's to publish\nin some registry like npm."]}),"\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"CRON"})," application is ideal for creating schedulers that will run\nperiodic jobs, such as for maintenance or calling third-party APIs\nto collect up-to-date data."]}),"\n"]}),"\n",(0,a.jsx)(n.h2,{id:"running-your-application",children:"Running your application"}),"\n",(0,a.jsx)(n.p,{children:"To run your application in development mode, run the following\ncommand:"}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"node artisan serve --watch\n"})}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsxs)(n.li,{children:["The ",(0,a.jsx)(n.code,{children:"serve"})," command will look up for your ",(0,a.jsx)(o.A,{father:"bin",child:"main.ts"}),"\nfile to bootstrap your application with predefined configurations."]}),"\n",(0,a.jsxs)(n.li,{children:["The ",(0,a.jsx)(n.code,{children:"watch"})," flag is meant to watch the file system for\nchanges and restart your application automatically when\ndoing some change on it."]}),"\n"]}),"\n",(0,a.jsxs)(n.h3,{id:"what-the--is-pathbinmaints",children:["What the \ud83e\udd2c is ",(0,a.jsx)(n.code,{children:"Path.bin('main.ts')"}),"?"]}),"\n",(0,a.jsx)(n.p,{children:"Well, Athenna tries a lot to be a framework without opinion, one thing\nthat some frameworks do that makes them to have a lot of opinion is defining\ncrucial file paths where you can't touch that file without breaking everything."}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is not totally different from them in this aspect, as you can see,\nthe ",(0,a.jsx)(n.code,{children:"node artisan serve"})," command depends on the existence of the ",(0,a.jsx)(n.code,{children:"./bin/main.ts"}),'\nfile. To be able to be "opinion less" without losing DX (Developer Experience)\nwe ensure that you can easily configure this path the way you want by doing two\nthings:']}),"\n",(0,a.jsxs)(n.ol,{children:["\n",(0,a.jsxs)(n.li,{children:["In all places of our documentation you will see that we always use the\n",(0,a.jsx)(n.a,{href:"/docs/the-basics/helpers#path",children:(0,a.jsx)(n.code,{children:"Path"})})," helper to reference file paths, this\nway is easier for you to understand that the path directory you just see is\nconfigurable."]}),"\n",(0,a.jsxs)(n.li,{children:["Inside ",(0,a.jsx)(n.code,{children:".athennarc.json"})," file or the ",(0,a.jsx)(n.code,{children:"athenna"})," property of your ",(0,a.jsx)(n.code,{children:"package.json"})," you\nhave the ",(0,a.jsxs)(n.a,{href:"/docs/getting-started/athennarc-file#the-directories-property",children:[(0,a.jsx)(n.code,{children:"directories"})," property"]}),"\nwhere you can modify any path supported by the ",(0,a.jsx)(n.a,{href:"/docs/the-basics/helpers#path",children:(0,a.jsx)(n.code,{children:"Path"})})," helper."]}),"\n"]}),"\n",(0,a.jsxs)(n.p,{children:["Ok, now you might be asking yourself: what the \ud83e\udd2c is ",(0,a.jsx)(n.code,{children:".athennarc.json"}),"? Be patient, we\ngonna see later on this documentation page \ud83d\ude06."]}),"\n",(0,a.jsx)(n.h2,{id:"initial-configuration",children:"Initial configuration"}),"\n",(0,a.jsxs)(n.p,{children:["All of the configuration files for the Athenna framework are stored\nin the ",(0,a.jsx)(o.A,{father:"config"})," directory. Each option is documented, so feel\nfree to look through the files and get familiar with the options available\nto you."]}),"\n",(0,a.jsxs)(n.p,{children:["Athenna needs almost no additional configuration out of the box. You are\nfree to get started developing! However, you may wish to review the\n",(0,a.jsx)(o.A,{father:"config",child:"app.ts"})," file and its documentation. It contains several\noptions such as locale that you may wish to change according to your\napplication."]}),"\n",(0,a.jsx)(n.h2,{id:"environment-based-configuration",children:"Environment based configuration"}),"\n",(0,a.jsxs)(n.p,{children:["Since many of Athenna's configuration option values may vary depending on\nwhether your application is running on your local machine or in production,\nmany important configuration values are defined using the ",(0,a.jsx)(n.code,{children:".env"})," file that\nexists at the root of your application."]}),"\n",(0,a.jsxs)(n.p,{children:["Your ",(0,a.jsx)(n.code,{children:".env"})," file should not be committed to your application's source control,\nsince each developer / server using your application could require a different\nenvironment configuration. Furthermore, this would be a security risk in the\nevent an intruder gains access to your source control repository, since any\nsensitive credentials would get exposed."]}),"\n",(0,a.jsx)(n.admonition,{type:"note",children:(0,a.jsxs)(n.p,{children:["For more information about the ",(0,a.jsx)(n.code,{children:".env"})," file and environment based configuration,\ncheck out the full ",(0,a.jsx)(n.a,{href:"/docs/getting-started/configuration",children:"configuration documentation"}),"."]})}),"\n",(0,a.jsx)(n.h2,{id:"workspace-and-runtime-configurations",children:"Workspace and runtime configurations"}),"\n",(0,a.jsxs)(n.p,{children:["Inside the root directory of your project, there is a file\ncalled ",(0,a.jsx)(n.code,{children:".athennarc.json"})," or the ",(0,a.jsx)(n.code,{children:"athenna"})," property of your\n",(0,a.jsx)(n.code,{children:"package.json"}),". These configurations are used by Athenna\nfor configuring the workspace and certain runtime\nsettings of your Athenna application. Basically any configuration\nrelated to how the framework will bootstrap and behave should be\ndefined in these configuration."]}),"\n",(0,a.jsxs)(n.admonition,{type:"note",children:[(0,a.jsxs)(n.p,{children:["Always remember that if you can't find the ",(0,a.jsx)(n.code,{children:".athennarc.json"})," file\nin the root path of you project, it will be defined inside of your\n",(0,a.jsx)(n.code,{children:"package.json"})," in the ",(0,a.jsx)(n.code,{children:"athenna"})," property."]}),(0,a.jsxs)(n.p,{children:["For more information about the RC file, more details about each one\nof its options and also how to define your own properties inside of it\ncheck out the full ",(0,a.jsx)(n.a,{href:"/docs/getting-started/athennarc-file",children:"Athenna RC file documentation"}),"."]})]}),"\n",(0,a.jsx)(n.h2,{id:"next-steps",children:"Next steps"}),"\n",(0,a.jsx)(n.p,{children:"Now that you have created your Athenna project, you may be\nwondering what to learn next. First, we strongly recommend\nbecoming familiar with how Athenna works by reading the\nfollowing documentation:"}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/application-lifecycle",children:"Application lifecycle"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/getting-started/configuration",children:"Configuration"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/getting-started/directory-structure",children:"Directory structure"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/service-container",children:"Service container"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/facades",children:"Facades"})}),"\n"]}),"\n",(0,a.jsx)(n.p,{children:"How you want to use Athenna will also dictate the next steps\non your journey. Since Athenna has different kind of applications,\nthere are a variety of ways to use Athenna, and we'll explore\nthe available ones bellow."}),"\n",(0,a.jsx)(n.admonition,{type:"tip",children:(0,a.jsxs)(n.p,{children:["New to Athenna? Check out the ",(0,a.jsx)(n.a,{href:"https://www.youtube.com/@athennaio",children:"Athenna YouTube channel"}),"\nfor an infinity of hands-on videos!"]})}),"\n",(0,a.jsx)(n.h3,{id:"rest-api-application",children:"REST API application"}),"\n",(0,a.jsx)(n.p,{children:"Use the REST API Application to serve as an API backend to single page application\nor mobile apps. In this context you may use Athenna for data storage / retrieval\nfor your REST API, while also taking advantage of Athenna's powerful services such\nas database, emails, notifications, and more."}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/rest-api-application/routing",children:"routing"})," and the ",(0,a.jsx)(n.a,{href:"/docs/orm/getting-started",children:"ORM"}),"."]}),"\n",(0,a.jsx)(n.h3,{id:"command-line-interface-cli-application",children:"Command Line Interface (CLI) application"}),"\n",(0,a.jsxs)(n.p,{children:["Use the command line interface (CLI) application to create libraries like ",(0,a.jsx)(n.code,{children:"athenna"}),"\nfor NPM where other developers could install it in their terminal. In this context\nyou may use Athenna for automatting process or generating files, while also taking\nadvantage of all Athenna's powerful foundation."]}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/cli-application/commands",children:"commands"}),"."]}),"\n",(0,a.jsx)(n.h3,{id:"cron-job-application",children:"CRON Job application"}),"\n",(0,a.jsx)(n.p,{children:"Use the CRON Job application to create schedulers for running periodic jobs, such\nas for maintenance or calling third-party APIs to collect up-to-date data."}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/cron-application/schedulers",children:"schedulers"}),"."]})]})}function u(e={}){const{wrapper:n}={...(0,i.R)(),...e.components};return n?(0,a.jsx)(n,{...e,children:(0,a.jsx)(d,{...e})}):d(e)}},7049:(e,n,t)=>{t.d(n,{A:()=>o});t(6540);const a={hoverCardContainer:"hoverCardContainer_jqUQ",hoverCardLink:"hoverCardLink_oDZU",hoverCard:"hoverCard_qTDS"};var i=t(4848);function o(e){let n=e.father;switch(e.father){case"storage":n="src/storage";break;case"logs":n="src/storage/logs";break;case"views":n="src/resources/views";break;case"locales":n="src/resources/locales";break;case"static":n="src/resources/static";break;case"config":n="src/config";break;case"database":n="src/database";break;case"seeders":n="src/database/seeders";break;case"migrations":n="src/database/migrations";break;case"console":n="src/console";break;case"commands":n="src/console/commands";break;case"cron":n="src/cron";break;case"schedulers":n="src/cron/schedulers";break;case"models":n="src/models";break;case"services":n="src/services";break;case"repositories":n="src/repositories";break;case"http":n="src/http";break;case"controllers":n="src/http/controllers";break;case"middlewares":n="src/http/middlewares";break;case"interceptors":n="src/http/interceptors";break;case"terminators":n="src/http/terminators";break;case"stubs":n="tests/stubs";break;case"fixtures":n="tests/fixtures";break;case"providers":n="src/providers";break;case"facades":n="src/facades";break;case"routes":n="src/routes"}return(0,i.jsxs)("div",{className:a.hoverCardContainer,children:[(0,i.jsx)("a",{className:a.hoverCardLink,href:`/docs/the-basics/helpers#path${e.father}`,children:(0,i.jsxs)("code",{children:["Path.",e.father,"(",e.child?`'${e.child}'`:"",")"]})}),(0,i.jsx)("div",{className:a.hoverCard,children:(0,i.jsxs)("p",{style:{margin:0},children:["./",n,e.child?`/${e.child}`:""]})})]})}},8453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>s});var a=t(6540);const i={},o=a.createContext(i);function r(e){const n=a.useContext(o);return a.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),a.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/6459b84b.e29b05d3.js b/assets/js/6459b84b.e29b05d3.js new file mode 100644 index 00000000..f62a2238 --- /dev/null +++ b/assets/js/6459b84b.e29b05d3.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[6459],{7112:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>s,default:()=>u,frontMatter:()=>r,metadata:()=>c,toc:()=>h});var a=t(4848),i=t(8453),o=t(7049);const r={title:"Installation",sidebar_position:1,description:"How to install and set up your first Athenna project."},s="Installation",c={id:"getting-started/installation",title:"Installation",description:"How to install and set up your first Athenna project.",source:"@site/docs/getting-started/installation.mdx",sourceDirName:"getting-started",slug:"/getting-started/installation",permalink:"/docs/getting-started/installation",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/installation.mdx",tags:[],version:"current",sidebarPosition:1,frontMatter:{title:"Installation",sidebar_position:1,description:"How to install and set up your first Athenna project."},sidebar:"tutorialSidebar",next:{title:"Athenna RC File",permalink:"/docs/getting-started/athennarc-file"}},l={},h=[{value:"Meet Athenna",id:"meet-athenna",level:2},{value:"Why Athenna?",id:"why-athenna",level:2},{value:"A progressive framework",id:"a-progressive-framework",level:3},{value:"A scalable framework",id:"a-scalable-framework",level:3},{value:"An agnostic framework",id:"an-agnostic-framework",level:3},{value:"A community framework",id:"a-community-framework",level:3},{value:"Your first Athenna project",id:"your-first-athenna-project",level:2},{value:"Prerequisites",id:"prerequisites",level:3},{value:"Installing via package manager",id:"installing-via-package-manager",level:3},{value:"Application type",id:"application-type",level:4},{value:"Running your application",id:"running-your-application",level:2},{value:"What the \ud83e\udd2c is Path.bin('main.ts')?",id:"what-the--is-pathbinmaints",level:3},{value:"Initial configuration",id:"initial-configuration",level:2},{value:"Environment based configuration",id:"environment-based-configuration",level:2},{value:"Workspace and runtime configurations",id:"workspace-and-runtime-configurations",level:2},{value:"Next steps",id:"next-steps",level:2},{value:"REST API application",id:"rest-api-application",level:3},{value:"Command Line Interface (CLI) application",id:"command-line-interface-cli-application",level:3},{value:"CRON Job application",id:"cron-job-application",level:3}];function d(e){const n={a:"a",admonition:"admonition",blockquote:"blockquote",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",li:"li",ol:"ol",p:"p",pre:"pre",ul:"ul",...(0,i.R)(),...e.components};return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(n.header,{children:(0,a.jsx)(n.h1,{id:"installation",children:"Installation"})}),"\n",(0,a.jsx)(n.p,{children:"How to install and set up your first Athenna project."}),"\n",(0,a.jsx)(n.h2,{id:"meet-athenna",children:"Meet Athenna"}),"\n",(0,a.jsx)(n.p,{children:"Athenna is a framework written specially for the Node.js runtime\nwith expressive and elegant syntax. The framework provides a reliable\ncode foundation and starting point for creating your application,\nallowing you to focus on creating something amazing while we sweat the\ndetails."}),"\n",(0,a.jsx)(n.p,{children:"Athenna strives to provide an amazing developer experience while\nproviding powerful features such as thorough dependency injection,\nan expressive database abstraction layer, unit and e2e testing,\nand more."}),"\n",(0,a.jsxs)(n.p,{children:["Whether you are new to Node.js ecosystem or have years of experience,\nAthenna is a framework that can grow with you. We'll help you take\nyour first steps as a developer or give you a boost as you take your\nexpertise to the next level. We can't wait to see what you build, make\nsure to share it with us in our ",(0,a.jsx)(n.a,{href:"https://discord.gg/JdEbBAKw6X",children:"Discord community"}),"."]}),"\n",(0,a.jsx)(n.admonition,{type:"tip",children:(0,a.jsxs)(n.p,{children:["New to Athenna? Check out the ",(0,a.jsx)(n.a,{href:"https://www.youtube.com/@athennaio",children:"Athenna YouTube channel"}),"\nfor an infinity of hands-on videos!"]})}),"\n",(0,a.jsx)(n.h2,{id:"why-athenna",children:"Why Athenna?"}),"\n",(0,a.jsx)(n.p,{children:"There are a variety of tools and frameworks available to you when building\na software using Node.js. However, we believe Athenna is the best choice for\nbuilding modern microservices."}),"\n",(0,a.jsx)(n.h3,{id:"a-progressive-framework",children:"A progressive framework"}),"\n",(0,a.jsx)(n.p,{children:"We like to call Athenna a \"progressive\" framework. By that, we mean that\nAthenna grows with you. If you're just taking your first steps into backend\ndevelopment, Athenna's vast library of documentation, guides, and video\ntutorials will help you learn the ropes without becoming overwhelmed."}),"\n",(0,a.jsx)(n.p,{children:"If you're a senior developer, Athenna gives you robust tools for dependency\ninjection, unit and e2e testing, database abstracted layer and more.\nAthenna is fine-tuned for building professional applications and ready to\nhandle enterprise work loads."}),"\n",(0,a.jsx)(n.h3,{id:"a-scalable-framework",children:"A scalable framework"}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is incredibly scalable. Thanks to the scaling-friendly nature of\nNode.js and great libraries like ",(0,a.jsx)(n.a,{href:"https://fastify.dev/",children:"Fastify"}),", horizontal\nscaling with Athenna is a breeze. In fact, Athenna REST API's have been\neasily scaled to handle hundreds of millions of requests per month."]}),"\n",(0,a.jsx)(n.h3,{id:"an-agnostic-framework",children:"An agnostic framework"}),"\n",(0,a.jsx)(n.p,{children:"Athenna is perpect for building modern software using microservices architecture.\nNo matter what type of application you are creating, be it a REST API, a CLI, a CRON Job,\nAthenna has a reliable solution foundation for each occasion, the only thing you will\nneed to take care is how your application will communicate with the external world."}),"\n",(0,a.jsx)(n.h3,{id:"a-community-framework",children:"A community framework"}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is not reinventing the whell, the framework combines the best packages\nin the Node.js ecosystem to offer the most robust and developer friendly\nframework. We are always seeking for talented developers like you to\nhelp us by ",(0,a.jsx)(n.a,{href:"https://github.com/athennaio",children:"contributing to the framework"})," and\nmake it even better."]}),"\n",(0,a.jsx)(n.h2,{id:"your-first-athenna-project",children:"Your first Athenna project"}),"\n",(0,a.jsx)(n.h3,{id:"prerequisites",children:"Prerequisites"}),"\n",(0,a.jsxs)(n.p,{children:["First, you need to install ",(0,a.jsx)(n.a,{href:"https://nodejs.org/",children:"Node.js"}),".\nWe recommend using ",(0,a.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm",children:"nvm"})," to do that."]}),"\n",(0,a.jsxs)(n.blockquote,{children:["\n",(0,a.jsx)(n.p,{children:(0,a.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm#installing-and-updating",children:"Click here to install nvm and get npm and Node.js\nrunning on your machine."})}),"\n"]}),"\n",(0,a.jsxs)(n.p,{children:["After you install ",(0,a.jsx)(n.code,{children:"nvm"}),", we recommend you to install\nNode.js v20.x or above, but you can still use\nNode.js v16.x and above. Install Node.js v20.x by running:"]}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"nvm install 20\n"})}),"\n",(0,a.jsxs)(n.p,{children:["We recommend setting Node.js v20.x as the default version, to do\nso with ",(0,a.jsx)(n.code,{children:"nvm"})," run:"]}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"nvm alias default 20.8.1\n"})}),"\n",(0,a.jsx)(n.h3,{id:"installing-via-package-manager",children:"Installing via package manager"}),"\n",(0,a.jsx)(n.p,{children:"We want it to be as easy as possible to get started with\nAthenna. With that in mind, we developed a CLI to assist\nin the creation of a new project."}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"npm install -g @athenna/cli\n"})}),"\n",(0,a.jsx)(n.p,{children:"Then you can run this command to generate your project:"}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"athenna new my-project-name\n"})}),"\n",(0,a.jsx)(n.p,{children:"The installation process prompts for the following selections:"}),"\n",(0,a.jsx)(n.h4,{id:"application-type",children:"Application type"}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"REST Api"})," application is ideal for creating a Http server\nusing REST architecture."]}),"\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"CLI"})," application is ideal for creating global CLI's to publish\nin some registry like npm."]}),"\n",(0,a.jsxs)(n.li,{children:[(0,a.jsx)(n.code,{children:"CRON"})," application is ideal for creating schedulers that will run\nperiodic jobs, such as for maintenance or calling third-party APIs\nto collect up-to-date data."]}),"\n"]}),"\n",(0,a.jsx)(n.h2,{id:"running-your-application",children:"Running your application"}),"\n",(0,a.jsx)(n.p,{children:"To run your application in development mode, run the following\ncommand:"}),"\n",(0,a.jsx)(n.pre,{children:(0,a.jsx)(n.code,{className:"language-bash",children:"node artisan serve --watch\n"})}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsxs)(n.li,{children:["The ",(0,a.jsx)(n.code,{children:"serve"})," command will look up for your ",(0,a.jsx)(o.A,{father:"bin",child:"main.ts"}),"\nfile to bootstrap your application with predefined configurations."]}),"\n",(0,a.jsxs)(n.li,{children:["The ",(0,a.jsx)(n.code,{children:"watch"})," flag is meant to watch the file system for\nchanges and restart your application automatically when\ndoing some change on it."]}),"\n"]}),"\n",(0,a.jsxs)(n.h3,{id:"what-the--is-pathbinmaints",children:["What the \ud83e\udd2c is ",(0,a.jsx)(n.code,{children:"Path.bin('main.ts')"}),"?"]}),"\n",(0,a.jsx)(n.p,{children:"Well, Athenna tries a lot to be a framework without opinion, one thing\nthat some frameworks do that makes them to have a lot of opinion is defining\ncrucial file paths where you can't touch that file without breaking everything."}),"\n",(0,a.jsxs)(n.p,{children:["Athenna is not totally different from them in this aspect, as you can see,\nthe ",(0,a.jsx)(n.code,{children:"node artisan serve"})," command depends on the existence of the ",(0,a.jsx)(n.code,{children:"./bin/main.ts"}),'\nfile. To be able to be "opinion less" without losing DX (Developer Experience)\nwe ensure that you can easily configure this path the way you want by doing two\nthings:']}),"\n",(0,a.jsxs)(n.ol,{children:["\n",(0,a.jsxs)(n.li,{children:["In all places of our documentation you will see that we always use the\n",(0,a.jsx)(n.a,{href:"/docs/the-basics/helpers#path",children:(0,a.jsx)(n.code,{children:"Path"})})," helper to reference file paths, this\nway is easier for you to understand that the path directory you just see is\nconfigurable."]}),"\n",(0,a.jsxs)(n.li,{children:["Inside ",(0,a.jsx)(n.code,{children:".athennarc.json"})," file or the ",(0,a.jsx)(n.code,{children:"athenna"})," property of your ",(0,a.jsx)(n.code,{children:"package.json"})," you\nhave the ",(0,a.jsxs)(n.a,{href:"/docs/getting-started/athennarc-file#the-directories-property",children:[(0,a.jsx)(n.code,{children:"directories"})," property"]}),"\nwhere you can modify any path supported by the ",(0,a.jsx)(n.a,{href:"/docs/the-basics/helpers#path",children:(0,a.jsx)(n.code,{children:"Path"})})," helper."]}),"\n"]}),"\n",(0,a.jsxs)(n.p,{children:["Ok, now you might be asking yourself: what the \ud83e\udd2c is ",(0,a.jsx)(n.code,{children:".athennarc.json"}),"? Be patient, we\ngonna see later on this documentation page \ud83d\ude06."]}),"\n",(0,a.jsx)(n.h2,{id:"initial-configuration",children:"Initial configuration"}),"\n",(0,a.jsxs)(n.p,{children:["All of the configuration files for the Athenna framework are stored\nin the ",(0,a.jsx)(o.A,{father:"config"})," directory. Each option is documented, so feel\nfree to look through the files and get familiar with the options available\nto you."]}),"\n",(0,a.jsxs)(n.p,{children:["Athenna needs almost no additional configuration out of the box. You are\nfree to get started developing! However, you may wish to review the\n",(0,a.jsx)(o.A,{father:"config",child:"app.ts"})," file and its documentation. It contains several\noptions such as locale that you may wish to change according to your\napplication."]}),"\n",(0,a.jsx)(n.h2,{id:"environment-based-configuration",children:"Environment based configuration"}),"\n",(0,a.jsxs)(n.p,{children:["Since many of Athenna's configuration option values may vary depending on\nwhether your application is running on your local machine or in production,\nmany important configuration values are defined using the ",(0,a.jsx)(n.code,{children:".env"})," file that\nexists at the root of your application."]}),"\n",(0,a.jsxs)(n.p,{children:["Your ",(0,a.jsx)(n.code,{children:".env"})," file should not be committed to your application's source control,\nsince each developer / server using your application could require a different\nenvironment configuration. Furthermore, this would be a security risk in the\nevent an intruder gains access to your source control repository, since any\nsensitive credentials would get exposed."]}),"\n",(0,a.jsx)(n.admonition,{type:"note",children:(0,a.jsxs)(n.p,{children:["For more information about the ",(0,a.jsx)(n.code,{children:".env"})," file and environment based configuration,\ncheck out the full ",(0,a.jsx)(n.a,{href:"/docs/getting-started/configuration",children:"configuration documentation"}),"."]})}),"\n",(0,a.jsx)(n.h2,{id:"workspace-and-runtime-configurations",children:"Workspace and runtime configurations"}),"\n",(0,a.jsxs)(n.p,{children:["Inside the root directory of your project, there is a file\ncalled ",(0,a.jsx)(n.code,{children:".athennarc.json"})," or the ",(0,a.jsx)(n.code,{children:"athenna"})," property of your\n",(0,a.jsx)(n.code,{children:"package.json"}),". These configurations are used by Athenna\nfor configuring the workspace and certain runtime\nsettings of your Athenna application. Basically any configuration\nrelated to how the framework will bootstrap and behave should be\ndefined in these configuration."]}),"\n",(0,a.jsxs)(n.admonition,{type:"note",children:[(0,a.jsxs)(n.p,{children:["Always remember that if you can't find the ",(0,a.jsx)(n.code,{children:".athennarc.json"})," file\nin the root path of you project, it will be defined inside of your\n",(0,a.jsx)(n.code,{children:"package.json"})," in the ",(0,a.jsx)(n.code,{children:"athenna"})," property."]}),(0,a.jsxs)(n.p,{children:["For more information about the RC file, more details about each one\nof its options and also how to define your own properties inside of it\ncheck out the full ",(0,a.jsx)(n.a,{href:"/docs/getting-started/athennarc-file",children:"Athenna RC file documentation"}),"."]})]}),"\n",(0,a.jsx)(n.h2,{id:"next-steps",children:"Next steps"}),"\n",(0,a.jsx)(n.p,{children:"Now that you have created your Athenna project, you may be\nwondering what to learn next. First, we strongly recommend\nbecoming familiar with how Athenna works by reading the\nfollowing documentation:"}),"\n",(0,a.jsxs)(n.ul,{children:["\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/application-lifecycle",children:"Application lifecycle"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/getting-started/configuration",children:"Configuration"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/getting-started/directory-structure",children:"Directory structure"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/service-container",children:"Service container"})}),"\n",(0,a.jsx)(n.li,{children:(0,a.jsx)(n.a,{href:"/docs/architecture-concepts/facades",children:"Facades"})}),"\n"]}),"\n",(0,a.jsx)(n.p,{children:"How you want to use Athenna will also dictate the next steps\non your journey. Since Athenna has different kind of applications,\nthere are a variety of ways to use Athenna, and we'll explore\nthe available ones bellow."}),"\n",(0,a.jsx)(n.admonition,{type:"tip",children:(0,a.jsxs)(n.p,{children:["New to Athenna? Check out the ",(0,a.jsx)(n.a,{href:"https://www.youtube.com/@athennaio",children:"Athenna YouTube channel"}),"\nfor an infinity of hands-on videos!"]})}),"\n",(0,a.jsx)(n.h3,{id:"rest-api-application",children:"REST API application"}),"\n",(0,a.jsx)(n.p,{children:"Use the REST API Application to serve as an API backend to single page application\nor mobile apps. In this context you may use Athenna for data storage / retrieval\nfor your REST API, while also taking advantage of Athenna's powerful services such\nas database, emails, notifications, and more."}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/rest-api-application/routing",children:"routing"})," and the ",(0,a.jsx)(n.a,{href:"/docs/orm/getting-started",children:"ORM"}),"."]}),"\n",(0,a.jsx)(n.h3,{id:"command-line-interface-cli-application",children:"Command Line Interface (CLI) application"}),"\n",(0,a.jsxs)(n.p,{children:["Use the command line interface (CLI) application to create libraries like ",(0,a.jsx)(n.code,{children:"athenna"}),"\nfor NPM where other developers could install it in their terminal. In this context\nyou may use Athenna for automatting process or generating files, while also taking\nadvantage of all Athenna's powerful foundation."]}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/cli-application/commands",children:"commands"}),"."]}),"\n",(0,a.jsx)(n.h3,{id:"cron-job-application",children:"CRON Job application"}),"\n",(0,a.jsx)(n.p,{children:"Use the CRON Job application to create schedulers for running periodic jobs, such\nas for maintenance or calling third-party APIs to collect up-to-date data."}),"\n",(0,a.jsxs)(n.p,{children:["If this is how you plan to use Athenna, you may want to check out our documentation\non ",(0,a.jsx)(n.a,{href:"/docs/cron-application/schedulers",children:"schedulers"}),"."]})]})}function u(e={}){const{wrapper:n}={...(0,i.R)(),...e.components};return n?(0,a.jsx)(n,{...e,children:(0,a.jsx)(d,{...e})}):d(e)}},7049:(e,n,t)=>{t.d(n,{A:()=>o});t(6540);const a={hoverCardContainer:"hoverCardContainer_jqUQ",hoverCardLink:"hoverCardLink_oDZU",hoverCard:"hoverCard_qTDS"};var i=t(4848);function o(e){let n=e.father;switch(e.father){case"storage":n="src/storage";break;case"logs":n="src/storage/logs";break;case"views":n="src/resources/views";break;case"locales":n="src/resources/locales";break;case"static":n="src/resources/static";break;case"config":n="src/config";break;case"database":n="src/database";break;case"seeders":n="src/database/seeders";break;case"migrations":n="src/database/migrations";break;case"console":n="src/console";break;case"commands":n="src/console/commands";break;case"cron":n="src/cron";break;case"schedulers":n="src/cron/schedulers";break;case"models":n="src/models";break;case"services":n="src/services";break;case"repositories":n="src/repositories";break;case"http":n="src/http";break;case"controllers":n="src/http/controllers";break;case"middlewares":n="src/http/middlewares";break;case"interceptors":n="src/http/interceptors";break;case"terminators":n="src/http/terminators";break;case"stubs":n="tests/stubs";break;case"fixtures":n="tests/fixtures";break;case"providers":n="src/providers";break;case"facades":n="src/facades";break;case"routes":n="src/routes"}return(0,i.jsxs)("div",{className:a.hoverCardContainer,children:[(0,i.jsx)("a",{className:a.hoverCardLink,href:`/docs/the-basics/helpers#path${e.father}`,children:(0,i.jsxs)("code",{children:["Path.",e.father,"(",e.child?`'${e.child}'`:"",")"]})}),(0,i.jsx)("div",{className:a.hoverCard,children:(0,i.jsxs)("p",{style:{margin:0},children:["./",n,e.child?`/${e.child}`:""]})})]})}},8453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>s});var a=t(6540);const i={},o=a.createContext(i);function r(e){const n=a.useContext(o);return a.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),a.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/98b8cc29.41262bc9.js b/assets/js/98b8cc29.41262bc9.js deleted file mode 100644 index 726cab29..00000000 --- a/assets/js/98b8cc29.41262bc9.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[6427],{6557:(e,n,r)=>{r.r(n),r.d(n,{assets:()=>d,contentTitle:()=>i,default:()=>h,frontMatter:()=>o,metadata:()=>a,toc:()=>c});var t=r(4848),s=r(8453);const o={title:"AthennaRC File",sidebar_position:3,description:"Understand what is the purpose of the .athennarc.json file."},i="AthennaRC File",a={id:"getting-started/athennarc-file",title:"AthennaRC File",description:"Understand what is the purpose of the .athennarc.json file.",source:"@site/docs/getting-started/athennarc-file.mdx",sourceDirName:"getting-started",slug:"/getting-started/athennarc-file",permalink:"/docs/getting-started/athennarc-file",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/athennarc-file.mdx",tags:[],version:"current",sidebarPosition:3,frontMatter:{title:"AthennaRC File",sidebar_position:3,description:"Understand what is the purpose of the .athennarc.json file."},sidebar:"tutorialSidebar",previous:{title:"Installation",permalink:"/docs/getting-started/installation"},next:{title:"Configuration",permalink:"/docs/getting-started/configuration"}},d={},c=[{value:"Introduction",id:"introduction",level:2},{value:"RC file vs Configurations",id:"rc-file-vs-configurations",level:2},{value:"Custom RC file path",id:"custom-rc-file-path",level:2},{value:"Using RC file in package.json",id:"using-rc-file-in-packagejson",level:2},{value:"The preloads property",id:"the-preloads-property",level:2},{value:"The providers property",id:"the-providers-property",level:2},{value:"The services property",id:"the-services-property",level:2},{value:"The commands property",id:"the-commands-property",level:2},{value:"The templates property",id:"the-templates-property",level:2},{value:"The directories property",id:"the-directories-property",level:2},{value:"The controllers property",id:"the-controllers-property",level:2},{value:"The middlewares property",id:"the-middlewares-property",level:2},{value:"The namedMiddlewares property",id:"the-namedmiddlewares-property",level:2},{value:"The globalMiddlewares property",id:"the-globalmiddlewares-property",level:2},{value:"The artisan property",id:"the-artisan-property",level:2}];function l(e){const n={a:"a",admonition:"admonition",code:"code",h1:"h1",h2:"h2",header:"header",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,s.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"athennarc-file",children:"AthennaRC File"})}),"\n",(0,t.jsx)(n.p,{children:"Understand what is the purpose of the .athennarc.json file."}),"\n",(0,t.jsx)(n.h2,{id:"introduction",children:"Introduction"}),"\n",(0,t.jsxs)(n.p,{children:["Inside the root directory of your project, there is a file\ncalled ",(0,t.jsx)(n.code,{children:".athennarc.json"})," which is responsible for configuring\nthe workspace and certain runtime settings of your Athenna\napplication."]}),"\n",(0,t.jsx)(n.h2,{id:"rc-file-vs-configurations",children:"RC file vs Configurations"}),"\n",(0,t.jsxs)(n.p,{children:["The responsible for the RC file is configuring the workspace\nand certain runtime settings to bootstrap your Athenna\napplication properly. Also, when working with ",(0,t.jsx)(n.code,{children:".json"})," files is\nvery easy to manipulate the values of it, making it possible\nto make changes on the file in runtime. Let's see an example\nwhere Athenna manipulates your RC file:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-bash",children:"node artisan make:service UserService\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Athenna will create the service file in your project but\nalso register it inside your ",(0,t.jsx)(n.code,{children:".athennarc.json"})," file:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "services": [\n "#src/services/UserService"\n ]\n}\n'})}),"\n",(0,t.jsx)(n.h2,{id:"custom-rc-file-path",children:"Custom RC file path"}),"\n",(0,t.jsxs)(n.p,{children:["You can change the name and the path of your RC file or even\ncreate customized ones for each environement (",(0,t.jsx)(n.code,{children:".athennarc.dev.json"}),",\n",(0,t.jsx)(n.code,{children:".athennarc.prod.json"}),"). To do that you need to set the new\npath to ",(0,t.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('dev.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n athennaRcPath: './bin/athennadevrc.json' \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsx)(n.p,{children:"Always remember that when using relative paths to set something\nin Athenna, you need to use your project root path as reference,\njust like in the example above."})}),"\n",(0,t.jsxs)(n.h2,{id:"using-rc-file-in-packagejson",children:["Using RC file in ",(0,t.jsx)(n.code,{children:"package.json"})]}),"\n",(0,t.jsxs)(n.p,{children:["You can also use the RC file inside ",(0,t.jsx)(n.code,{children:"package.json"}),". By default,\nIf you don't specify the path of your RC file, and the default\n",(0,t.jsx)(n.code,{children:".athennarc.json"})," cannot be found in the root path of your application,\nAthenna will check if the ",(0,t.jsx)(n.code,{children:"athenna"})," property exists in your ",(0,t.jsx)(n.code,{children:"package.json"}),":"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title="package.json"',children:'{\n "athenna": {\n "providers": [\n "@athenna/core/providers/CoreProvider",\n "@athenna/http/providers/HttpRouteProvider",\n "@athenna/http/providers/HttpServerProvider"\n ],\n "directories": {\n "bin": "bootstrap"\n }\n }\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-preloads-property",children:["The ",(0,t.jsx)(n.code,{children:"preloads"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array of files that will be loaded when your application\nis bootstrapping. The files are loaded after booting the\nservice providers. You can do anything you want in preload\nfiles. Check the example bellow:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-typescript",metastring:'title="say-hello.ts"',children:"import { Log } from '@athenna/logger'\nimport { Config } from '@athenna/core'\n\nLog.info(`Hello from ${Config.get('app.name')} application!`)\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "preloads": [\n "./bin/preloads/say-hello.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-providers-property",children:["The ",(0,t.jsx)(n.code,{children:"providers"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array of service providers to load when the application\nis bootstrapping:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "providers": [\n "@athenna/core/providers/CoreProvider",\n "@athenna/http/providers/HttpRouteProvider",\n "@athenna/http/providers/HttpServerProvider"\n ]\n}\n'})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about service providers could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/the-basics/http/controllers",children:"service providers documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h2,{id:"the-services-property",children:["The ",(0,t.jsx)(n.code,{children:"services"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["This property is responsible to register your application\nservices or from some library inside the service container.\nIn most cases, it is better to simply instantiate your services,\nbut in others you might have different implementations for\nsome interface, using ",(0,t.jsx)(n.a,{href:"https://www.educative.io/answers/what-is-inversion-of-control",children:"inversion of control"}),"\nin these cases could be an exceptional idea to register your\nservices in the container:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "services": [\n "#src/services/AppService",\n "./src/services/OtherService.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-commands-property",children:["The ",(0,t.jsx)(n.code,{children:"commands"})," property"]}),"\n",(0,t.jsx)(n.p,{children:'An object that is responsible to register your application\ncommands and their respective settings. The key of the\ncommands object needs to be exactly your command signature\nwithout any arguments, flags or spaces. Also, the value\nof it could be the command path or an object with the\n"path" key inside:'}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "commands": {\n "make:exception": "@athenna/core/commands/MakeExceptionCommand",\n "make:facade": {\n "path": "@athenna/core/commands/MakeFacadeCommand",\n "env": "local",\n "destination": "./src/providers/facades",\n "loadApp": false,\n "stayAlive": false,\n "loadAllCommands": false,\n "environments": ["console"]\n }\n }\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Depending on the command you are running, it got their\nown configurations. Commands like ",(0,t.jsx)(n.code,{children:"make:..."})," for example,\nreads the ",(0,t.jsx)(n.code,{children:"destination"})," property to the file generated\nfor a different path from it default."]}),"\n",(0,t.jsx)(n.p,{children:"There are properties that you can define whatever your\ncommand. Let's see who they are and what they do:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"path"})})," - Defines the path to your command, this field is mandatory\nif you are using an object to define your command."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"loadApp"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", the ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method will be called\nuntil running your command. The ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method is responsible\nto load your env file and configuration files and also boot your\nservice providers."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"env"})})," - Defines which env file will be used to run your command.\nThis field is not required and it will only be relevant when the\n",(0,t.jsx)(n.code,{children:"loadApp"})," property is ",(0,t.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"stayAlive"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", your command will stay running until the\nevent loop is not in use anymore. Very useful for keep running\nbackground tasks from commands."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"loadAllCommands"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", all the commands inside the ",(0,t.jsx)(n.code,{children:"commands"}),"\nproperty will be loaded. Useful when you command needs to call other command\nprogrammatically."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"environments"})})," - Only relevant when ",(0,t.jsx)(n.code,{children:"loadApp"})," is ",(0,t.jsx)(n.code,{children:"true"}),". The environments\nset will be used as parameter for ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method and will help\nAthenna to select the service providers that should or shouldn't be booted."]}),"\n"]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about commands could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/the-basics/cli/commands",children:"cli commands documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h2,{id:"the-templates-property",children:["The ",(0,t.jsx)(n.code,{children:"templates"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map your application commands templates with their respective\npath. The templates mapped in this object will be used by your\n",(0,t.jsx)(n.code,{children:"make"})," commands to generate the resource with some specific\ncode template:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "templates": {\n "exception": "node_modules/@athenna/core/templates/exception.edge",\n "facade": "node_modules/@athenna/core/templates/facade.edge"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-directories-property",children:["The ",(0,t.jsx)(n.code,{children:"directories"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map your application directories with their respective\nbase path. The ",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,t.jsx)(n.code,{children:"Path"})}),"\nclass will use the directories mapped in this object to\nresolve the paths of your application:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "directories": {\n "bin": "bin",\n "src": "src",\n "app": "app",\n "bootstrap": "bootstrap",\n "public": "public",\n "static": "public/static",\n "assets": "public/assets",\n "nodeModules": "node_modules",\n "nodeModulesBin": "node_modules/.bin",\n "tests": "tests",\n "stubs": "tests/stubs",\n "fixtures": "tests/fixtures",\n "models": "src/models",\n "services": "src/services",\n "jobs": "src/jobs",\n "workers": "src/workers",\n "exceptions": "src/exceptions",\n "repositories": "src/repositories",\n "console": "src/console",\n "commands": "src/console/commands",\n "http": "src/http",\n "guards": "src/http/guards",\n "controllers": "src/http/controllers",\n "middlewares": "src/http/middlewares",\n "interceptors": "src/http/interceptors",\n "terminators": "src/http/terminators",\n "validators": "src/validators",\n "cron": "src/cron",\n "schedulers": "src/cron/schedulers",\n "config": "src/config",\n "database": "src/database",\n "seeders": "src/database/seeders",\n "migrations": "src/database/migrations",\n "lang": "src/lang",\n "resources": "src/resources",\n "views": "src/resources/views",\n "locales": "src/resources/locales", \n "providers": "src/providers",\n "facades": "src/facades", \n "routes": "src/routes",\n "storage": "src/storage",\n "logs": "src/storage/logs"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["The paths above are the default ones used by Athenna to resolve\nyour application paths. You can change one of them or many\nif you want, your directories defined in ",(0,t.jsx)(n.code,{children:"directories"})," property\nwill always be merged with the defaults."]}),"\n",(0,t.jsxs)(n.p,{children:["Athenna always rely on ",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,t.jsx)(n.code,{children:"Path"})}),"\nclass methods to find files and directories that are used\ninternally by the framework, like configuration file, route\nfiles, entry points and many others. Changing the ",(0,t.jsx)(n.code,{children:"directories"})," property\ncould be very useful when you are building your own project structure."]}),"\n",(0,t.jsxs)(n.h2,{id:"the-controllers-property",children:["The ",(0,t.jsx)(n.code,{children:"controllers"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array with the controllers of your application. The\ncontrollers registered in this array will be registered\nin the service container to be accessed easily by your\nRoute facade:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "controllers": [\n "#src/http/controllers/AppController",\n "./src/http/controllers/OtherController.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-middlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"middlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An array with the middlewares of your application. The\nmiddlewares registered in this array will be registered\nin the service container to be accessed easily by your\n",(0,t.jsx)(n.code,{children:"Route"})," facade:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "middlewares": [\n "#src/http/middlewares/AppMiddleware",\n "./src/http/interceptors/AppInterceptor.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Athenna expects that the middlewares set in this property\nto be annotated with ",(0,t.jsx)(n.code,{children:"@Middleware()"}),", ",(0,t.jsx)(n.code,{children:"@Interceptor()"})," or ",(0,t.jsx)(n.code,{children:"@Terminator()"}),"\nannotations, this is not mandatory, but you will only\nbe able to set the name of the middleware or if it is global or not using\nthe annotations."]}),"\n",(0,t.jsxs)(n.p,{children:["If you are not using TypeScript in your application, you can use the\n",(0,t.jsx)(n.a,{href:"/docs/getting-started/athennarc-file#the-namedmiddlewares-property",children:(0,t.jsx)(n.code,{children:"namedMiddlewares"})}),"\nproperty to register named middlewares and the\n",(0,t.jsx)(n.a,{href:"/docs/getting-started/athennarc-file#the-globalmiddlewares-property",children:(0,t.jsx)(n.code,{children:"globalMiddlewares"})}),"\nproperty to register global middlewares."]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about middlewares could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/rest-api-application/middlewares",children:"http middlewares documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h2,{id:"the-namedmiddlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"namedMiddlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map the named middlewares of your application. Named\nmiddlewares could be configured using the ",(0,t.jsx)(n.code,{children:"@Middleware"}),"\nannotation, but if you are not using TypeScript in your\napplication, you can use this object to map your named\nmiddlewares. Named middlewares are very useful to be used\nin your route declaration:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "namedMiddlewares": {\n "app": "#src/http/middlewares/AppMiddleware",\n "intercept": "./src/http/interceptors/AppInterceptor.js"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-globalmiddlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"globalMiddlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An array with the global middlewares of your application.\nGlobal middlewares could be configured using the ",(0,t.jsx)(n.code,{children:"@Middleware"}),"\nannotation, but if you are not using TypeScript in your\napplication, you can use this object to map your named\nmiddlewares. Global middlewares are executed every time\nin any request of your application:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "globalMiddlewares": [\n "#src/http/middlewares/AppMiddleware",\n "./src/http/interceptors/AppInterceptor.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h2,{id:"the-artisan-property",children:["The ",(0,t.jsx)(n.code,{children:"artisan"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An object with a variety of Artisan configurations.\nAt this point the only configurations accepted are ",(0,t.jsx)(n.code,{children:"artisan.child.executor"}),"\nand ",(0,t.jsx)(n.code,{children:"artisan.child.path"}),". Both configurations are used to define\nhow the ",(0,t.jsx)(n.code,{children:"Artisan.callInChild()"})," method will behave when no options\nare set to it as second argument:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "artisan": {\n "child": {\n "executor": "node --inspect",\n "path": "./bin/artisan.ts"\n }\n }\n}\n'})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["Athenna will automatically parse the ",(0,t.jsx)(n.code,{children:"artisan.child.path"})," using\nthe ",(0,t.jsx)(n.code,{children:"Path.ext()"})," method, so you don't need to worry about if the\nextension is ",(0,t.jsx)(n.code,{children:".js"})," or ",(0,t.jsx)(n.code,{children:".ts"}),"."]})})]})}function h(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(l,{...e})}):l(e)}},8453:(e,n,r)=>{r.d(n,{R:()=>i,x:()=>a});var t=r(6540);const s={},o=t.createContext(s);function i(e){const n=t.useContext(o);return t.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:i(e.components),t.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/98b8cc29.6c61093b.js b/assets/js/98b8cc29.6c61093b.js new file mode 100644 index 00000000..a48e2b7f --- /dev/null +++ b/assets/js/98b8cc29.6c61093b.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunk_athenna_docs=self.webpackChunk_athenna_docs||[]).push([[6427],{6557:(e,n,r)=>{r.r(n),r.d(n,{assets:()=>d,contentTitle:()=>i,default:()=>h,frontMatter:()=>o,metadata:()=>a,toc:()=>c});var t=r(4848),s=r(8453);const o={title:"Athenna RC File",sidebar_position:3,description:"Understand what is the purpose of the .athennarc.json file."},i="Athenna RC File",a={id:"getting-started/athennarc-file",title:"Athenna RC File",description:"Understand what is the purpose of the .athennarc.json file.",source:"@site/docs/getting-started/athennarc-file.mdx",sourceDirName:"getting-started",slug:"/getting-started/athennarc-file",permalink:"/docs/getting-started/athennarc-file",draft:!1,unlisted:!1,editUrl:"https://github.com/AthennaIO/Docs/tree/main/docs/getting-started/athennarc-file.mdx",tags:[],version:"current",sidebarPosition:3,frontMatter:{title:"Athenna RC File",sidebar_position:3,description:"Understand what is the purpose of the .athennarc.json file."},sidebar:"tutorialSidebar",previous:{title:"Installation",permalink:"/docs/getting-started/installation"},next:{title:"Configuration",permalink:"/docs/getting-started/configuration"}},d={},c=[{value:"Introduction",id:"introduction",level:2},{value:"RC file vs Configurations",id:"rc-file-vs-configurations",level:2},{value:"Custom RC file path",id:"custom-rc-file-path",level:2},{value:"Using RC file in package.json",id:"using-rc-file-in-packagejson",level:2},{value:"RC File properties",id:"rc-file-properties",level:2},{value:"The preloads property",id:"the-preloads-property",level:3},{value:"The providers property",id:"the-providers-property",level:3},{value:"The services property",id:"the-services-property",level:3},{value:"The commands property",id:"the-commands-property",level:3},{value:"The templates property",id:"the-templates-property",level:3},{value:"The directories property",id:"the-directories-property",level:3},{value:"The controllers property",id:"the-controllers-property",level:3},{value:"The middlewares property",id:"the-middlewares-property",level:3},{value:"The namedMiddlewares property",id:"the-namedmiddlewares-property",level:3},{value:"The globalMiddlewares property",id:"the-globalmiddlewares-property",level:3},{value:"The artisan property",id:"the-artisan-property",level:3}];function l(e){const n={a:"a",admonition:"admonition",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,s.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"athenna-rc-file",children:"Athenna RC File"})}),"\n",(0,t.jsx)(n.p,{children:"Understand what is the purpose of the .athennarc.json file."}),"\n",(0,t.jsx)(n.h2,{id:"introduction",children:"Introduction"}),"\n",(0,t.jsxs)(n.p,{children:["Inside the root directory of your project, there is a file\ncalled ",(0,t.jsx)(n.code,{children:".athennarc.json"})," which is responsible for configuring\nthe workspace and certain runtime settings of your Athenna\napplication."]}),"\n",(0,t.jsx)(n.h2,{id:"rc-file-vs-configurations",children:"RC file vs Configurations"}),"\n",(0,t.jsxs)(n.p,{children:["The responsible of the ",(0,t.jsx)(n.code,{children:".athennarc.json"})," is configuring the workspace\nand certain runtime settings to bootstrap your Athenna\napplication properly. Also, when working with ",(0,t.jsx)(n.code,{children:".json"})," files is\nvery easy to manipulate the values of it, making it possible\nto make changes on the file in runtime. Let's see an example\nwhere Athenna manipulates your ",(0,t.jsx)(n.code,{children:".athennarc.json"}),":"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-bash",children:"node artisan make:service UserService\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Athenna will create the service file in your project but\nalso register it inside your ",(0,t.jsx)(n.code,{children:".athennarc.json"})," file:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "services": [\n "#src/services/UserService"\n ]\n}\n'})}),"\n",(0,t.jsx)(n.h2,{id:"custom-rc-file-path",children:"Custom RC file path"}),"\n",(0,t.jsxs)(n.p,{children:["You can change the name and the path of your RC file or even\ncreate customized ones for each environment (",(0,t.jsx)(n.code,{children:".athennarc.dev.json"}),",\n",(0,t.jsx)(n.code,{children:".athennarc.prod.json"}),"). To do that you need to set the new\npath to ",(0,t.jsx)(n.code,{children:"Ignite::load()"})," static method:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-typescript",metastring:"title=\"Path.bin('dev.ts')\"",children:"import { Ignite } from '@athenna/core'\n\nconst ignite = await new Ignite().load(import.meta.url, {\n athennaRcPath: './bin/athennadevrc.json' \ud83d\udc48\n})\n\nawait ignite.httpServer()\n"})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsx)(n.p,{children:"Always remember that when using relative paths to set something\nin Athenna, you need to use your project root path as reference,\njust like in the example above."})}),"\n",(0,t.jsxs)(n.h2,{id:"using-rc-file-in-packagejson",children:["Using RC file in ",(0,t.jsx)(n.code,{children:"package.json"})]}),"\n",(0,t.jsxs)(n.p,{children:["You can also use the RC file inside ",(0,t.jsx)(n.code,{children:"package.json"}),". By default,\nIf you don't specify the path of your RC file, and the default\n",(0,t.jsx)(n.code,{children:".athennarc.json"})," cannot be found in the root path of your application,\nAthenna will check if the ",(0,t.jsx)(n.code,{children:"athenna"})," property exists in your ",(0,t.jsx)(n.code,{children:"package.json"}),":"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title="package.json"',children:'{\n "athenna": {\n "providers": [\n "@athenna/core/providers/CoreProvider",\n "@athenna/http/providers/HttpRouteProvider",\n "@athenna/http/providers/HttpServerProvider"\n ],\n "directories": {\n "bin": "bootstrap"\n }\n }\n}\n'})}),"\n",(0,t.jsx)(n.h2,{id:"rc-file-properties",children:"RC File properties"}),"\n",(0,t.jsxs)(n.p,{children:["Let's cover all of the ",(0,t.jsx)(n.code,{children:".athennarc.json"})," file properties and\nunderstand how to use each one of them:"]}),"\n",(0,t.jsxs)(n.h3,{id:"the-preloads-property",children:["The ",(0,t.jsx)(n.code,{children:"preloads"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array of files that will be loaded when your application\nis bootstrapping. The files are loaded after booting the\nservice providers. You can do anything you want in preload\nfiles. Check the example bellow:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-typescript",metastring:'title="say-hello.ts"',children:"import { Log } from '@athenna/logger'\nimport { Config } from '@athenna/core'\n\nLog.info(`Hello from ${Config.get('app.name')} application!`)\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "preloads": [\n "./bin/preloads/say-hello.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-providers-property",children:["The ",(0,t.jsx)(n.code,{children:"providers"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array of service providers to load when the application\nis bootstrapping:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "providers": [\n "@athenna/core/providers/CoreProvider",\n "@athenna/http/providers/HttpRouteProvider",\n "@athenna/http/providers/HttpServerProvider"\n ]\n}\n'})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about service providers could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/the-basics/http/controllers",children:"service providers documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h3,{id:"the-services-property",children:["The ",(0,t.jsx)(n.code,{children:"services"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["This property is responsible to register your application\nservices or from some library inside the service container.\nIn most cases, it is better to simply instantiate your services,\nbut in others you might have different implementations for\nsome interface, using ",(0,t.jsx)(n.a,{href:"https://www.educative.io/answers/what-is-inversion-of-control",children:"inversion of control"}),"\nin these cases could be an exceptional idea to register your\nservices in the container:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "services": [\n "#src/services/AppService",\n "./src/services/OtherService.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-commands-property",children:["The ",(0,t.jsx)(n.code,{children:"commands"})," property"]}),"\n",(0,t.jsx)(n.p,{children:'An object that is responsible to register your application\ncommands and their respective settings. The key of the\ncommands object needs to be exactly your command signature\nwithout any arguments, flags or spaces. Also, the value\nof it could be the command path or an object with the\n"path" key inside:'}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "commands": {\n "make:exception": "@athenna/core/commands/MakeExceptionCommand",\n "make:facade": {\n "path": "@athenna/core/commands/MakeFacadeCommand",\n "env": "local",\n "destination": "./src/providers/facades",\n "loadApp": false,\n "stayAlive": false,\n "loadAllCommands": false,\n "environments": ["console"]\n }\n }\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Depending on the command you are running, it got their\nown configurations. Commands like ",(0,t.jsx)(n.code,{children:"make:..."})," for example,\nreads the ",(0,t.jsx)(n.code,{children:"destination"})," property to the file generated\nfor a different path from it default."]}),"\n",(0,t.jsx)(n.p,{children:"There are properties that you can define whatever your\ncommand. Let's see who they are and what they do:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"path"})})," - Defines the path to your command, this field is mandatory\nif you are using an object to define your command."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"loadApp"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", the ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method will be called\nuntil running your command. The ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method is responsible\nto load your env file and configuration files and also boot your\nservice providers."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"env"})})," - Defines which env file will be used to run your command.\nThis field is not required and it will only be relevant when the\n",(0,t.jsx)(n.code,{children:"loadApp"})," property is ",(0,t.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"stayAlive"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", your command will stay running until the\nevent loop is not in use anymore. Very useful for keep running\nbackground tasks from commands."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"loadAllCommands"})})," - If ",(0,t.jsx)(n.code,{children:"true"}),", all the commands inside the ",(0,t.jsx)(n.code,{children:"commands"}),"\nproperty will be loaded. Useful when you command needs to call other command\nprogrammatically."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"environments"})})," - Only relevant when ",(0,t.jsx)(n.code,{children:"loadApp"})," is ",(0,t.jsx)(n.code,{children:"true"}),". The environments\nset will be used as parameter for ",(0,t.jsx)(n.code,{children:"Ignite.fire()"})," method and will help\nAthenna to select the service providers that should or shouldn't be booted."]}),"\n"]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about commands could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/the-basics/cli/commands",children:"cli commands documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h3,{id:"the-templates-property",children:["The ",(0,t.jsx)(n.code,{children:"templates"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map your application commands templates with their respective\npath. The templates mapped in this object will be used by your\n",(0,t.jsx)(n.code,{children:"make"})," commands to generate the resource with some specific\ncode template:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "templates": {\n "exception": "node_modules/@athenna/core/templates/exception.edge",\n "facade": "node_modules/@athenna/core/templates/facade.edge"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-directories-property",children:["The ",(0,t.jsx)(n.code,{children:"directories"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map your application directories with their respective\nbase path. The ",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,t.jsx)(n.code,{children:"Path"})}),"\nclass will use the directories mapped in this object to\nresolve the paths of your application:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "directories": {\n "bin": "bin",\n "src": "src",\n "app": "app",\n "bootstrap": "bootstrap",\n "public": "public",\n "static": "public/static",\n "assets": "public/assets",\n "nodeModules": "node_modules",\n "nodeModulesBin": "node_modules/.bin",\n "tests": "tests",\n "stubs": "tests/stubs",\n "fixtures": "tests/fixtures",\n "models": "src/models",\n "services": "src/services",\n "jobs": "src/jobs",\n "workers": "src/workers",\n "exceptions": "src/exceptions",\n "repositories": "src/repositories",\n "console": "src/console",\n "commands": "src/console/commands",\n "http": "src/http",\n "guards": "src/http/guards",\n "controllers": "src/http/controllers",\n "middlewares": "src/http/middlewares",\n "interceptors": "src/http/interceptors",\n "terminators": "src/http/terminators",\n "validators": "src/validators",\n "cron": "src/cron",\n "schedulers": "src/cron/schedulers",\n "config": "src/config",\n "database": "src/database",\n "seeders": "src/database/seeders",\n "migrations": "src/database/migrations",\n "lang": "src/lang",\n "resources": "src/resources",\n "views": "src/resources/views",\n "locales": "src/resources/locales", \n "providers": "src/providers",\n "facades": "src/facades", \n "routes": "src/routes",\n "storage": "src/storage",\n "logs": "src/storage/logs"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["The paths above are the default ones used by Athenna to resolve\nyour application paths. You can change one of them or many\nif you want, your directories defined in ",(0,t.jsx)(n.code,{children:"directories"})," property\nwill always be merged with the defaults."]}),"\n",(0,t.jsxs)(n.p,{children:["Athenna always rely on ",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/digging-deeper/helpers#path",children:(0,t.jsx)(n.code,{children:"Path"})}),"\nclass methods to find files and directories that are used\ninternally by the framework, like configuration file, route\nfiles, entry points and many others. Changing the ",(0,t.jsx)(n.code,{children:"directories"})," property\ncould be very useful when you are building your own project structure."]}),"\n",(0,t.jsxs)(n.h3,{id:"the-controllers-property",children:["The ",(0,t.jsx)(n.code,{children:"controllers"})," property"]}),"\n",(0,t.jsx)(n.p,{children:"An array with the controllers of your application. The\ncontrollers registered in this array will be registered\nin the service container to be accessed easily by your\nRoute facade:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "controllers": [\n "#src/http/controllers/AppController",\n "./src/http/controllers/OtherController.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-middlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"middlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An array with the middlewares of your application. The\nmiddlewares registered in this array will be registered\nin the service container to be accessed easily by your\n",(0,t.jsx)(n.code,{children:"Route"})," facade:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "middlewares": [\n "#src/http/middlewares/AppMiddleware",\n "./src/http/interceptors/AppInterceptor.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Athenna expects that the middlewares set in this property\nto be annotated with ",(0,t.jsx)(n.code,{children:"@Middleware()"}),", ",(0,t.jsx)(n.code,{children:"@Interceptor()"})," or ",(0,t.jsx)(n.code,{children:"@Terminator()"}),"\nannotations, this is not mandatory, but you will only\nbe able to set the name of the middleware or if it is global or not using\nthe annotations."]}),"\n",(0,t.jsxs)(n.p,{children:["If you are not using TypeScript in your application, you can use the\n",(0,t.jsx)(n.a,{href:"/docs/getting-started/athennarc-file#the-namedmiddlewares-property",children:(0,t.jsx)(n.code,{children:"namedMiddlewares"})}),"\nproperty to register named middlewares and the\n",(0,t.jsx)(n.a,{href:"/docs/getting-started/athennarc-file#the-globalmiddlewares-property",children:(0,t.jsx)(n.code,{children:"globalMiddlewares"})}),"\nproperty to register global middlewares."]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["More information about middlewares could be found at\n",(0,t.jsx)(n.a,{href:"https://athenna.io/docs/rest-api-application/middlewares",children:"http middlewares documentation section"}),"."]})}),"\n",(0,t.jsxs)(n.h3,{id:"the-namedmiddlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"namedMiddlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["Map the named middlewares of your application. Named\nmiddlewares could be configured using the ",(0,t.jsx)(n.code,{children:"@Middleware()"}),"\nannotation, but if you are not using TypeScript in your\napplication, you can use this object to map your named\nmiddlewares. Named middlewares are very useful to be used\nin your route declaration:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "namedMiddlewares": {\n "app": "#src/http/middlewares/AppMiddleware",\n "intercept": "./src/http/interceptors/AppInterceptor.js"\n }\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-globalmiddlewares-property",children:["The ",(0,t.jsx)(n.code,{children:"globalMiddlewares"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An array with the global middlewares of your application.\nGlobal middlewares could be configured using the ",(0,t.jsx)(n.code,{children:"@Middleware()"}),"\nannotation, but if you are not using TypeScript in your\napplication, you can use this object to map your named\nmiddlewares. Global middlewares are executed every time\nin any request of your application:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "globalMiddlewares": [\n "#src/http/middlewares/AppMiddleware",\n "./src/http/interceptors/AppInterceptor.js"\n ]\n}\n'})}),"\n",(0,t.jsxs)(n.h3,{id:"the-artisan-property",children:["The ",(0,t.jsx)(n.code,{children:"artisan"})," property"]}),"\n",(0,t.jsxs)(n.p,{children:["An object with a variety of Artisan configurations.\nAt this point the only configurations accepted are ",(0,t.jsx)(n.code,{children:"artisan.child.executor"}),"\nand ",(0,t.jsx)(n.code,{children:"artisan.child.path"}),". Both configurations are used to define\nhow the ",(0,t.jsx)(n.code,{children:"Artisan.callInChild()"})," method will behave when no options\nare set to it as second argument:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",metastring:'title=".athennarc.json"',children:'{\n "artisan": {\n "child": {\n "executor": "node --inspect",\n "path": "./bin/artisan.ts"\n }\n }\n}\n'})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["Athenna will automatically parse the ",(0,t.jsx)(n.code,{children:"artisan.child.path"})," using\nthe ",(0,t.jsx)(n.code,{children:"Path.ext()"})," method, so you don't need to worry about if the\nextension is ",(0,t.jsx)(n.code,{children:".js"})," or ",(0,t.jsx)(n.code,{children:".ts"}),"."]})})]})}function h(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(l,{...e})}):l(e)}},8453:(e,n,r)=>{r.d(n,{R:()=>i,x:()=>a});var t=r(6540);const s={},o=t.createContext(s);function i(e){const n=t.useContext(o);return t.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:i(e.components),t.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.2bb103db.js b/assets/js/runtime~main.e4720fcb.js similarity index 97% rename from assets/js/runtime~main.2bb103db.js rename to assets/js/runtime~main.e4720fcb.js index 266ccd48..fa77170f 100644 --- a/assets/js/runtime~main.2bb103db.js +++ b/assets/js/runtime~main.e4720fcb.js @@ -1 +1 @@ -(()=>{"use strict";var e,a,c,r,d,f={},t={};function b(e){var a=t[e];if(void 0!==a)return a.exports;var c=t[e]={exports:{}};return f[e].call(c.exports,c,c.exports,b),c.exports}b.m=f,e=[],b.O=(a,c,r,d)=>{if(!c){var f=1/0;for(n=0;n=d)&&Object.keys(b.O).every((e=>b.O[e](c[o])))?c.splice(o--,1):(t=!1,d0&&e[n-1][2]>d;n--)e[n]=e[n-1];e[n]=[c,r,d]},b.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return b.d(a,{a:a}),a},c=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,b.t=function(e,r){if(1&r&&(e=this(e)),8&r)return e;if("object"==typeof e&&e){if(4&r&&e.__esModule)return e;if(16&r&&"function"==typeof e.then)return e}var d=Object.create(null);b.r(d);var f={};a=a||[null,c({}),c([]),c(c)];for(var t=2&r&&e;"object"==typeof t&&!~a.indexOf(t);t=c(t))Object.getOwnPropertyNames(t).forEach((a=>f[a]=()=>e[a]));return f.default=()=>e,b.d(d,f),d},b.d=(e,a)=>{for(var c in a)b.o(a,c)&&!b.o(e,c)&&Object.defineProperty(e,c,{enumerable:!0,get:a[c]})},b.f={},b.e=e=>Promise.all(Object.keys(b.f).reduce(((a,c)=>(b.f[c](e,a),a)),[])),b.u=e=>"assets/js/"+({177:"f3e8f525",217:"cc784980",755:"a4d3e054",773:"94e1aa3d",794:"e6388bba",849:"0058b4c6",918:"8f88d278",1235:"a7456010",1438:"3ebc3d67",1746:"391d9c20",1992:"801e3691",2042:"reactPlayerTwitch",2106:"ad6f9ff5",2190:"1cb4e3ed",2350:"3fb3fb7f",2723:"reactPlayerMux",3077:"77db9a17",3082:"50bfad63",3217:"96c97f44",3392:"reactPlayerVidyard",3582:"08133570",3699:"2ab4562b",3737:"94cba6ed",3892:"c24fb5a2",4583:"1df93b7f",4620:"84dac133",4638:"60418129",4881:"1bfe5704",4890:"30d27832",5202:"8416df86",5312:"26455d6c",5390:"25ef7947",5424:"e71332dd",5583:"f991d430",5742:"aba21aa0",6061:"1f391b9e",6173:"reactPlayerVimeo",6295:"21dc2778",6328:"reactPlayerDailyMotion",6353:"reactPlayerPreview",6427:"98b8cc29",6459:"6459b84b",6463:"reactPlayerKaltura",6563:"4a3e27c5",6619:"bfc576a4",6701:"28a593c7",6742:"807b6800",6887:"reactPlayerFacebook",6927:"1b2a18d5",7033:"ead3bc46",7098:"a7bd4aaa",7170:"ba3d4959",7182:"e2eeca55",7336:"3aefb2a9",7345:"f210ba15",7458:"reactPlayerFilePlayer",7570:"reactPlayerMixcloud",7627:"reactPlayerStreamable",7729:"76be5683",7802:"d295d49b",7867:"4ca25bb5",7952:"59190afd",8194:"a2c501c8",8401:"17896441",8446:"reactPlayerYouTube",8459:"0b6406e9",8475:"31058b07",8849:"751a75f6",8995:"cbe663fe",9048:"a94703ab",9116:"d550161c",9195:"bfc34fa7",9257:"3f163355",9340:"reactPlayerWistia",9346:"62e3c86b",9647:"5e95c892",9655:"a21dcd43",9979:"reactPlayerSoundCloud"}[e]||e)+"."+{177:"b365d540",217:"8e829d8e",755:"763b2a62",773:"4adec87c",794:"9f60e687",849:"3377be63",918:"aea094c1",1169:"da74cb51",1176:"3fa1f1a0",1235:"899d59d8",1245:"0093dbf5",1303:"3d23202d",1331:"b04ae620",1398:"b34f2eb3",1438:"710b7f21",1746:"80cef5a3",1946:"c1f898cd",1992:"ae983ce3",2042:"55051469",2106:"a065e090",2130:"e743b6f2",2190:"4ef422e6",2237:"e35c7faa",2350:"82cdc3c7",2362:"92df48f8",2376:"89729940",2453:"4efe8b38",2548:"b0f7735f",2723:"7e2e3f86",2843:"88f75fed",2925:"9d15cf56",2983:"44d58df4",3068:"a770ed84",3077:"cfdd980b",3082:"bed9431a",3217:"777fb791",3392:"090f167f",3582:"7810342c",3626:"5a9700a2",3699:"80d506c1",3706:"59d3c775",3737:"a30398e0",3892:"e249ccb5",4132:"f281ab14",4162:"d00bd1cc",4334:"66dbfd08",4583:"50f773da",4620:"690e12ba",4638:"0dee389f",4741:"8e6c530a",4881:"a89fa12e",4890:"8c09266b",4943:"60400b58",5202:"1f125844",5312:"54502484",5390:"1714955d",5424:"083d306c",5583:"a6763569",5742:"9e876600",6061:"c2f91780",6173:"6ee50fc8",6295:"d0235ee8",6328:"ea480b77",6353:"9d736f65",6420:"f2879ce4",6427:"41262bc9",6459:"41354f80",6463:"065e45b1",6563:"dedb7468",6619:"5f25d85d",6701:"2deffc28",6742:"dca80451",6788:"80aec321",6803:"73ef982b",6887:"1dd4863c",6927:"3697ed96",7033:"2dc2ba37",7098:"288b6c02",7170:"1dc5260b",7182:"706e84f8",7336:"5e2f1293",7345:"27977df4",7426:"280729df",7458:"bef56796",7570:"c3e2db4a",7627:"d93f5569",7729:"52c4503c",7802:"7eea1fd8",7867:"db049c7f",7952:"a05b5ce2",8055:"b792f2c6",8194:"3cf5e88c",8337:"8759c63e",8401:"c1b96131",8446:"6664bacb",8459:"a27a161b",8475:"0f0edd0e",8478:"e7701138",8635:"75cbcfbe",8676:"ee4f37b1",8810:"92902e39",8849:"4bd9a970",8869:"9590b926",8995:"c30f3794",9048:"27dd5686",9116:"42e59631",9195:"f99ddc31",9257:"68d2b803",9340:"5498c2c2",9346:"35430c50",9647:"506f1936",9655:"bf6ad62c",9689:"74e05a4b",9730:"aa51d1f6",9979:"d89f0440"}[e]+".js",b.miniCssF=e=>{},b.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),b.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),r={},d="@athenna/docs:",b.l=(e,a,c,f)=>{if(r[e])r[e].push(a);else{var t,o;if(void 0!==c)for(var l=document.getElementsByTagName("script"),n=0;n{t.onerror=t.onload=null,clearTimeout(s);var d=r[e];if(delete r[e],t.parentNode&&t.parentNode.removeChild(t),d&&d.forEach((e=>e(c))),a)return a(c)},s=setTimeout(u.bind(null,void 0,{type:"timeout",target:t}),12e4);t.onerror=u.bind(null,t.onerror),t.onload=u.bind(null,t.onload),o&&document.head.appendChild(t)}},b.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},b.p="/",b.gca=function(e){return e={17896441:"8401",60418129:"4638",f3e8f525:"177",cc784980:"217",a4d3e054:"755","94e1aa3d":"773",e6388bba:"794","0058b4c6":"849","8f88d278":"918",a7456010:"1235","3ebc3d67":"1438","391d9c20":"1746","801e3691":"1992",reactPlayerTwitch:"2042",ad6f9ff5:"2106","1cb4e3ed":"2190","3fb3fb7f":"2350",reactPlayerMux:"2723","77db9a17":"3077","50bfad63":"3082","96c97f44":"3217",reactPlayerVidyard:"3392","08133570":"3582","2ab4562b":"3699","94cba6ed":"3737",c24fb5a2:"3892","1df93b7f":"4583","84dac133":"4620","1bfe5704":"4881","30d27832":"4890","8416df86":"5202","26455d6c":"5312","25ef7947":"5390",e71332dd:"5424",f991d430:"5583",aba21aa0:"5742","1f391b9e":"6061",reactPlayerVimeo:"6173","21dc2778":"6295",reactPlayerDailyMotion:"6328",reactPlayerPreview:"6353","98b8cc29":"6427","6459b84b":"6459",reactPlayerKaltura:"6463","4a3e27c5":"6563",bfc576a4:"6619","28a593c7":"6701","807b6800":"6742",reactPlayerFacebook:"6887","1b2a18d5":"6927",ead3bc46:"7033",a7bd4aaa:"7098",ba3d4959:"7170",e2eeca55:"7182","3aefb2a9":"7336",f210ba15:"7345",reactPlayerFilePlayer:"7458",reactPlayerMixcloud:"7570",reactPlayerStreamable:"7627","76be5683":"7729",d295d49b:"7802","4ca25bb5":"7867","59190afd":"7952",a2c501c8:"8194",reactPlayerYouTube:"8446","0b6406e9":"8459","31058b07":"8475","751a75f6":"8849",cbe663fe:"8995",a94703ab:"9048",d550161c:"9116",bfc34fa7:"9195","3f163355":"9257",reactPlayerWistia:"9340","62e3c86b":"9346","5e95c892":"9647",a21dcd43:"9655",reactPlayerSoundCloud:"9979"}[e]||e,b.p+b.u(e)},(()=>{var e={5354:0,1869:0};b.f.j=(a,c)=>{var r=b.o(e,a)?e[a]:void 0;if(0!==r)if(r)c.push(r[2]);else if(/^(1869|5354)$/.test(a))e[a]=0;else{var d=new Promise(((c,d)=>r=e[a]=[c,d]));c.push(r[2]=d);var f=b.p+b.u(a),t=new Error;b.l(f,(c=>{if(b.o(e,a)&&(0!==(r=e[a])&&(e[a]=void 0),r)){var d=c&&("load"===c.type?"missing":c.type),f=c&&c.target&&c.target.src;t.message="Loading chunk "+a+" failed.\n("+d+": "+f+")",t.name="ChunkLoadError",t.type=d,t.request=f,r[1](t)}}),"chunk-"+a,a)}},b.O.j=a=>0===e[a];var a=(a,c)=>{var r,d,f=c[0],t=c[1],o=c[2],l=0;if(f.some((a=>0!==e[a]))){for(r in t)b.o(t,r)&&(b.m[r]=t[r]);if(o)var n=o(b)}for(a&&a(c);l{"use strict";var e,a,c,r,d,f={},t={};function b(e){var a=t[e];if(void 0!==a)return a.exports;var c=t[e]={exports:{}};return f[e].call(c.exports,c,c.exports,b),c.exports}b.m=f,e=[],b.O=(a,c,r,d)=>{if(!c){var f=1/0;for(n=0;n=d)&&Object.keys(b.O).every((e=>b.O[e](c[o])))?c.splice(o--,1):(t=!1,d0&&e[n-1][2]>d;n--)e[n]=e[n-1];e[n]=[c,r,d]},b.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return b.d(a,{a:a}),a},c=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,b.t=function(e,r){if(1&r&&(e=this(e)),8&r)return e;if("object"==typeof e&&e){if(4&r&&e.__esModule)return e;if(16&r&&"function"==typeof e.then)return e}var d=Object.create(null);b.r(d);var f={};a=a||[null,c({}),c([]),c(c)];for(var t=2&r&&e;"object"==typeof t&&!~a.indexOf(t);t=c(t))Object.getOwnPropertyNames(t).forEach((a=>f[a]=()=>e[a]));return f.default=()=>e,b.d(d,f),d},b.d=(e,a)=>{for(var c in a)b.o(a,c)&&!b.o(e,c)&&Object.defineProperty(e,c,{enumerable:!0,get:a[c]})},b.f={},b.e=e=>Promise.all(Object.keys(b.f).reduce(((a,c)=>(b.f[c](e,a),a)),[])),b.u=e=>"assets/js/"+({177:"f3e8f525",217:"cc784980",755:"a4d3e054",773:"94e1aa3d",794:"e6388bba",849:"0058b4c6",918:"8f88d278",1235:"a7456010",1438:"3ebc3d67",1746:"391d9c20",1992:"801e3691",2042:"reactPlayerTwitch",2106:"ad6f9ff5",2190:"1cb4e3ed",2350:"3fb3fb7f",2723:"reactPlayerMux",3077:"77db9a17",3082:"50bfad63",3217:"96c97f44",3392:"reactPlayerVidyard",3582:"08133570",3699:"2ab4562b",3737:"94cba6ed",3892:"c24fb5a2",4583:"1df93b7f",4620:"84dac133",4638:"60418129",4881:"1bfe5704",4890:"30d27832",5202:"8416df86",5312:"26455d6c",5390:"25ef7947",5424:"e71332dd",5583:"f991d430",5742:"aba21aa0",6061:"1f391b9e",6173:"reactPlayerVimeo",6295:"21dc2778",6328:"reactPlayerDailyMotion",6353:"reactPlayerPreview",6427:"98b8cc29",6459:"6459b84b",6463:"reactPlayerKaltura",6563:"4a3e27c5",6619:"bfc576a4",6701:"28a593c7",6742:"807b6800",6887:"reactPlayerFacebook",6927:"1b2a18d5",7033:"ead3bc46",7098:"a7bd4aaa",7170:"ba3d4959",7182:"e2eeca55",7336:"3aefb2a9",7345:"f210ba15",7458:"reactPlayerFilePlayer",7570:"reactPlayerMixcloud",7627:"reactPlayerStreamable",7729:"76be5683",7802:"d295d49b",7867:"4ca25bb5",7952:"59190afd",8194:"a2c501c8",8401:"17896441",8446:"reactPlayerYouTube",8459:"0b6406e9",8475:"31058b07",8849:"751a75f6",8995:"cbe663fe",9048:"a94703ab",9116:"d550161c",9195:"bfc34fa7",9257:"3f163355",9340:"reactPlayerWistia",9346:"62e3c86b",9647:"5e95c892",9655:"a21dcd43",9979:"reactPlayerSoundCloud"}[e]||e)+"."+{177:"b365d540",217:"8e829d8e",755:"763b2a62",773:"4adec87c",794:"9f60e687",849:"e7eea27f",918:"aea094c1",1169:"da74cb51",1176:"3fa1f1a0",1235:"899d59d8",1245:"0093dbf5",1303:"3d23202d",1331:"b04ae620",1398:"b34f2eb3",1438:"710b7f21",1746:"80cef5a3",1946:"c1f898cd",1992:"ae983ce3",2042:"55051469",2106:"a065e090",2130:"e743b6f2",2190:"4ef422e6",2237:"e35c7faa",2350:"82cdc3c7",2362:"92df48f8",2376:"89729940",2453:"4efe8b38",2548:"b0f7735f",2723:"7e2e3f86",2843:"88f75fed",2925:"9d15cf56",2983:"44d58df4",3068:"a770ed84",3077:"cfdd980b",3082:"bed9431a",3217:"777fb791",3392:"090f167f",3582:"7810342c",3626:"5a9700a2",3699:"80d506c1",3706:"59d3c775",3737:"a30398e0",3892:"e249ccb5",4132:"f281ab14",4162:"d00bd1cc",4334:"66dbfd08",4583:"50f773da",4620:"690e12ba",4638:"0dee389f",4741:"8e6c530a",4881:"a89fa12e",4890:"8b98abf6",4943:"60400b58",5202:"1f125844",5312:"54502484",5390:"1714955d",5424:"083d306c",5583:"a6763569",5742:"9e876600",6061:"c2f91780",6173:"6ee50fc8",6295:"d0235ee8",6328:"ea480b77",6353:"9d736f65",6420:"f2879ce4",6427:"6c61093b",6459:"e29b05d3",6463:"065e45b1",6563:"dedb7468",6619:"5f25d85d",6701:"2deffc28",6742:"dca80451",6788:"80aec321",6803:"73ef982b",6887:"1dd4863c",6927:"3697ed96",7033:"2dc2ba37",7098:"288b6c02",7170:"1dc5260b",7182:"706e84f8",7336:"5e2f1293",7345:"27977df4",7426:"280729df",7458:"bef56796",7570:"c3e2db4a",7627:"d93f5569",7729:"52c4503c",7802:"7eea1fd8",7867:"db049c7f",7952:"a05b5ce2",8055:"b792f2c6",8194:"3cf5e88c",8337:"8759c63e",8401:"c1b96131",8446:"6664bacb",8459:"a27a161b",8475:"0f0edd0e",8478:"e7701138",8635:"75cbcfbe",8676:"ee4f37b1",8810:"92902e39",8849:"4bd9a970",8869:"9590b926",8995:"c30f3794",9048:"27dd5686",9116:"42e59631",9195:"f99ddc31",9257:"68d2b803",9340:"5498c2c2",9346:"35430c50",9647:"506f1936",9655:"bf6ad62c",9689:"74e05a4b",9730:"aa51d1f6",9979:"d89f0440"}[e]+".js",b.miniCssF=e=>{},b.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),b.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),r={},d="@athenna/docs:",b.l=(e,a,c,f)=>{if(r[e])r[e].push(a);else{var t,o;if(void 0!==c)for(var l=document.getElementsByTagName("script"),n=0;n{t.onerror=t.onload=null,clearTimeout(s);var d=r[e];if(delete r[e],t.parentNode&&t.parentNode.removeChild(t),d&&d.forEach((e=>e(c))),a)return a(c)},s=setTimeout(u.bind(null,void 0,{type:"timeout",target:t}),12e4);t.onerror=u.bind(null,t.onerror),t.onload=u.bind(null,t.onload),o&&document.head.appendChild(t)}},b.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},b.p="/",b.gca=function(e){return e={17896441:"8401",60418129:"4638",f3e8f525:"177",cc784980:"217",a4d3e054:"755","94e1aa3d":"773",e6388bba:"794","0058b4c6":"849","8f88d278":"918",a7456010:"1235","3ebc3d67":"1438","391d9c20":"1746","801e3691":"1992",reactPlayerTwitch:"2042",ad6f9ff5:"2106","1cb4e3ed":"2190","3fb3fb7f":"2350",reactPlayerMux:"2723","77db9a17":"3077","50bfad63":"3082","96c97f44":"3217",reactPlayerVidyard:"3392","08133570":"3582","2ab4562b":"3699","94cba6ed":"3737",c24fb5a2:"3892","1df93b7f":"4583","84dac133":"4620","1bfe5704":"4881","30d27832":"4890","8416df86":"5202","26455d6c":"5312","25ef7947":"5390",e71332dd:"5424",f991d430:"5583",aba21aa0:"5742","1f391b9e":"6061",reactPlayerVimeo:"6173","21dc2778":"6295",reactPlayerDailyMotion:"6328",reactPlayerPreview:"6353","98b8cc29":"6427","6459b84b":"6459",reactPlayerKaltura:"6463","4a3e27c5":"6563",bfc576a4:"6619","28a593c7":"6701","807b6800":"6742",reactPlayerFacebook:"6887","1b2a18d5":"6927",ead3bc46:"7033",a7bd4aaa:"7098",ba3d4959:"7170",e2eeca55:"7182","3aefb2a9":"7336",f210ba15:"7345",reactPlayerFilePlayer:"7458",reactPlayerMixcloud:"7570",reactPlayerStreamable:"7627","76be5683":"7729",d295d49b:"7802","4ca25bb5":"7867","59190afd":"7952",a2c501c8:"8194",reactPlayerYouTube:"8446","0b6406e9":"8459","31058b07":"8475","751a75f6":"8849",cbe663fe:"8995",a94703ab:"9048",d550161c:"9116",bfc34fa7:"9195","3f163355":"9257",reactPlayerWistia:"9340","62e3c86b":"9346","5e95c892":"9647",a21dcd43:"9655",reactPlayerSoundCloud:"9979"}[e]||e,b.p+b.u(e)},(()=>{var e={5354:0,1869:0};b.f.j=(a,c)=>{var r=b.o(e,a)?e[a]:void 0;if(0!==r)if(r)c.push(r[2]);else if(/^(1869|5354)$/.test(a))e[a]=0;else{var d=new Promise(((c,d)=>r=e[a]=[c,d]));c.push(r[2]=d);var f=b.p+b.u(a),t=new Error;b.l(f,(c=>{if(b.o(e,a)&&(0!==(r=e[a])&&(e[a]=void 0),r)){var d=c&&("load"===c.type?"missing":c.type),f=c&&c.target&&c.target.src;t.message="Loading chunk "+a+" failed.\n("+d+": "+f+")",t.name="ChunkLoadError",t.type=d,t.request=f,r[1](t)}}),"chunk-"+a,a)}},b.O.j=a=>0===e[a];var a=(a,c)=>{var r,d,f=c[0],t=c[1],o=c[2],l=0;if(f.some((a=>0!==e[a]))){for(r in t)b.o(t,r)&&(b.m[r]=t[r]);if(o)var n=o(b)}for(a&&a(c);l - + diff --git a/docs/architecture-concepts/application-lifecycle.html b/docs/architecture-concepts/application-lifecycle.html index 6c047a93..dd54cbbf 100644 --- a/docs/architecture-concepts/application-lifecycle.html +++ b/docs/architecture-concepts/application-lifecycle.html @@ -18,7 +18,7 @@ - + diff --git a/docs/architecture-concepts/facades.html b/docs/architecture-concepts/facades.html index 5b082835..1b8cc9d5 100644 --- a/docs/architecture-concepts/facades.html +++ b/docs/architecture-concepts/facades.html @@ -18,7 +18,7 @@ - + diff --git a/docs/architecture-concepts/service-container.html b/docs/architecture-concepts/service-container.html index 3c5ea384..8040e0f4 100644 --- a/docs/architecture-concepts/service-container.html +++ b/docs/architecture-concepts/service-container.html @@ -18,7 +18,7 @@ - + diff --git a/docs/architecture-concepts/service-providers.html b/docs/architecture-concepts/service-providers.html index 6f6348a9..793dd0a4 100644 --- a/docs/architecture-concepts/service-providers.html +++ b/docs/architecture-concepts/service-providers.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cli-application/annotations.html b/docs/cli-application/annotations.html index 5f871912..e33e95e2 100644 --- a/docs/cli-application/annotations.html +++ b/docs/cli-application/annotations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cli-application/commands.html b/docs/cli-application/commands.html index fb14dbf2..3f16af17 100644 --- a/docs/cli-application/commands.html +++ b/docs/cli-application/commands.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cli-application/error-handling.html b/docs/cli-application/error-handling.html index 500d5053..491b60c4 100644 --- a/docs/cli-application/error-handling.html +++ b/docs/cli-application/error-handling.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cli-application/publishing.html b/docs/cli-application/publishing.html index 7a703d34..a3742533 100644 --- a/docs/cli-application/publishing.html +++ b/docs/cli-application/publishing.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cli-application/running.html b/docs/cli-application/running.html index b22679e3..028aa1b1 100644 --- a/docs/cli-application/running.html +++ b/docs/cli-application/running.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cron-application/annotations.html b/docs/cron-application/annotations.html index 43f7d1cc..b69898f2 100644 --- a/docs/cron-application/annotations.html +++ b/docs/cron-application/annotations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cron-application/cron-context.html b/docs/cron-application/cron-context.html index c91b41ea..23dd1498 100644 --- a/docs/cron-application/cron-context.html +++ b/docs/cron-application/cron-context.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cron-application/error-handling.html b/docs/cron-application/error-handling.html index b56b896b..6b298e4c 100644 --- a/docs/cron-application/error-handling.html +++ b/docs/cron-application/error-handling.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cron-application/schedulers.html b/docs/cron-application/schedulers.html index eb62beed..ff30aa3b 100644 --- a/docs/cron-application/schedulers.html +++ b/docs/cron-application/schedulers.html @@ -18,7 +18,7 @@ - + diff --git a/docs/cron-application/tracing-executions.html b/docs/cron-application/tracing-executions.html index 72053ea5..64629fd8 100644 --- a/docs/cron-application/tracing-executions.html +++ b/docs/cron-application/tracing-executions.html @@ -18,7 +18,7 @@ - + diff --git a/docs/database/getting-started.html b/docs/database/getting-started.html index e7e8201c..75a35347 100644 --- a/docs/database/getting-started.html +++ b/docs/database/getting-started.html @@ -18,7 +18,7 @@ - + diff --git a/docs/database/migrations.html b/docs/database/migrations.html index cd59d6b4..45a144d2 100644 --- a/docs/database/migrations.html +++ b/docs/database/migrations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/database/query-builder.html b/docs/database/query-builder.html index 44152515..88555160 100644 --- a/docs/database/query-builder.html +++ b/docs/database/query-builder.html @@ -18,7 +18,7 @@ - + diff --git a/docs/database/seeding.html b/docs/database/seeding.html index c40dcc45..0ff6e39b 100644 --- a/docs/database/seeding.html +++ b/docs/database/seeding.html @@ -18,7 +18,7 @@ - + diff --git a/docs/digging-deeper/collections.html b/docs/digging-deeper/collections.html index 4debfabc..726d1df9 100644 --- a/docs/digging-deeper/collections.html +++ b/docs/digging-deeper/collections.html @@ -18,7 +18,7 @@ - + diff --git a/docs/digging-deeper/graceful-shutdown.html b/docs/digging-deeper/graceful-shutdown.html index 717501fa..408a782e 100644 --- a/docs/digging-deeper/graceful-shutdown.html +++ b/docs/digging-deeper/graceful-shutdown.html @@ -18,7 +18,7 @@ - + diff --git a/docs/digging-deeper/library-development.html b/docs/digging-deeper/library-development.html index b78462ff..093fe4dc 100644 --- a/docs/digging-deeper/library-development.html +++ b/docs/digging-deeper/library-development.html @@ -18,7 +18,7 @@ - + diff --git a/docs/digging-deeper/mail.html b/docs/digging-deeper/mail.html index 21b84677..895b70e7 100644 --- a/docs/digging-deeper/mail.html +++ b/docs/digging-deeper/mail.html @@ -18,7 +18,7 @@ - + diff --git a/docs/digging-deeper/repl.html b/docs/digging-deeper/repl.html index c5e5fb45..58d9b022 100644 --- a/docs/digging-deeper/repl.html +++ b/docs/digging-deeper/repl.html @@ -18,7 +18,7 @@ - + diff --git a/docs/getting-started/athennarc-file.html b/docs/getting-started/athennarc-file.html index 7ccf91ff..182d534f 100644 --- a/docs/getting-started/athennarc-file.html +++ b/docs/getting-started/athennarc-file.html @@ -3,7 +3,7 @@ -AthennaRC File | Athenna Framework +Athenna RC File | Athenna Framework @@ -18,31 +18,31 @@ - + -

AthennaRC File

+

Athenna RC File

Understand what is the purpose of the .athennarc.json file.

-

Introduction​

+

Introduction​

Inside the root directory of your project, there is a file called .athennarc.json which is responsible for configuring the workspace and certain runtime settings of your Athenna application.

RC file vs Configurations​

-

The responsible for the RC file is configuring the workspace +

The responsible of the .athennarc.json is configuring the workspace and certain runtime settings to bootstrap your Athenna application properly. Also, when working with .json files is very easy to manipulate the values of it, making it possible to make changes on the file in runtime. Let's see an example -where Athenna manipulates your RC file:

+where Athenna manipulates your .athennarc.json:

node artisan make:service UserService

Athenna will create the service file in your project but also register it inside your .athennarc.json file:

{
"services": [
"#src/services/UserService"
]
}

Custom RC file path​

You can change the name and the path of your RC file or even -create customized ones for each environement (.athennarc.dev.json, +create customized ones for each environment (.athennarc.dev.json, .athennarc.prod.json). To do that you need to set the new path to Ignite::load() static method:

Path.bin('dev.ts')
import { Ignite } from '@athenna/core'

const ignite = await new Ignite().load(import.meta.url, {
athennaRcPath: './bin/athennadevrc.json' 👈
})

await ignite.httpServer()
@@ -55,20 +55,23 @@

.athennarc.json cannot be found in the root path of your application, Athenna will check if the athenna property exists in your package.json:

package.json
{
"athenna": {
"providers": [
"@athenna/core/providers/CoreProvider",
"@athenna/http/providers/HttpRouteProvider",
"@athenna/http/providers/HttpServerProvider"
],
"directories": {
"bin": "bootstrap"
}
}
}
-

The preloads property​

+

RC File properties​

+

Let's cover all of the .athennarc.json file properties and +understand how to use each one of them:

+

The preloads property​

An array of files that will be loaded when your application is bootstrapping. The files are loaded after booting the service providers. You can do anything you want in preload files. Check the example bellow:

say-hello.ts
import { Log } from '@athenna/logger'
import { Config } from '@athenna/core'

Log.info(`Hello from ${Config.get('app.name')} application!`)
-
{
"preloads": [
"./bin/preloads/say-hello.js"
]
}
-

The providers property​

+
.athennarc.json
{
"preloads": [
"./bin/preloads/say-hello.js"
]
}
+

The providers property​

An array of service providers to load when the application is bootstrapping:

-
{
"providers": [
"@athenna/core/providers/CoreProvider",
"@athenna/http/providers/HttpRouteProvider",
"@athenna/http/providers/HttpServerProvider"
]
}
+
.athennarc.json
{
"providers": [
"@athenna/core/providers/CoreProvider",
"@athenna/http/providers/HttpRouteProvider",
"@athenna/http/providers/HttpServerProvider"
]
}
tip

More information about service providers could be found at service providers documentation section.

-

The services property​

+

The services property​

This property is responsible to register your application services or from some library inside the service container. In most cases, it is better to simply instantiate your services, @@ -76,15 +79,15 @@

The inversion of control in these cases could be an exceptional idea to register your services in the container:

-
{
"services": [
"#src/services/AppService",
"./src/services/OtherService.js"
]
}
-

The commands property​

+
.athennarc.json
{
"services": [
"#src/services/AppService",
"./src/services/OtherService.js"
]
}
+

The commands property​

An object that is responsible to register your application commands and their respective settings. The key of the commands object needs to be exactly your command signature without any arguments, flags or spaces. Also, the value of it could be the command path or an object with the "path" key inside:

-
{
"commands": {
"make:exception": "@athenna/core/commands/MakeExceptionCommand",
"make:facade": {
"path": "@athenna/core/commands/MakeFacadeCommand",
"env": "local",
"destination": "./src/providers/facades",
"loadApp": false,
"stayAlive": false,
"loadAllCommands": false,
"environments": ["console"]
}
}
}
+
.athennarc.json
{
"commands": {
"make:exception": "@athenna/core/commands/MakeExceptionCommand",
"make:facade": {
"path": "@athenna/core/commands/MakeFacadeCommand",
"env": "local",
"destination": "./src/providers/facades",
"loadApp": false,
"stayAlive": false,
"loadAllCommands": false,
"environments": ["console"]
}
}
}

Depending on the command you are running, it got their own configurations. Commands like make:... for example, reads the destination property to the file generated @@ -113,18 +116,18 @@

The
tip

More information about commands could be found at cli commands documentation section.

-

The templates property​

+

The templates property​

Map your application commands templates with their respective path. The templates mapped in this object will be used by your make commands to generate the resource with some specific code template:

-
{
"templates": {
"exception": "node_modules/@athenna/core/templates/exception.edge",
"facade": "node_modules/@athenna/core/templates/facade.edge"
}
}
-

The directories property​

+
.athennarc.json
{
"templates": {
"exception": "node_modules/@athenna/core/templates/exception.edge",
"facade": "node_modules/@athenna/core/templates/facade.edge"
}
}
+

The directories property​

Map your application directories with their respective base path. The Path class will use the directories mapped in this object to resolve the paths of your application:

-
{
"directories": {
"bin": "bin",
"src": "src",
"app": "app",
"bootstrap": "bootstrap",
"public": "public",
"static": "public/static",
"assets": "public/assets",
"nodeModules": "node_modules",
"nodeModulesBin": "node_modules/.bin",
"tests": "tests",
"stubs": "tests/stubs",
"fixtures": "tests/fixtures",
"models": "src/models",
"services": "src/services",
"jobs": "src/jobs",
"workers": "src/workers",
"exceptions": "src/exceptions",
"repositories": "src/repositories",
"console": "src/console",
"commands": "src/console/commands",
"http": "src/http",
"guards": "src/http/guards",
"controllers": "src/http/controllers",
"middlewares": "src/http/middlewares",
"interceptors": "src/http/interceptors",
"terminators": "src/http/terminators",
"validators": "src/validators",
"cron": "src/cron",
"schedulers": "src/cron/schedulers",
"config": "src/config",
"database": "src/database",
"seeders": "src/database/seeders",
"migrations": "src/database/migrations",
"lang": "src/lang",
"resources": "src/resources",
"views": "src/resources/views",
"locales": "src/resources/locales",
"providers": "src/providers",
"facades": "src/facades",
"routes": "src/routes",
"storage": "src/storage",
"logs": "src/storage/logs"
}
}
+
.athennarc.json
{
"directories": {
"bin": "bin",
"src": "src",
"app": "app",
"bootstrap": "bootstrap",
"public": "public",
"static": "public/static",
"assets": "public/assets",
"nodeModules": "node_modules",
"nodeModulesBin": "node_modules/.bin",
"tests": "tests",
"stubs": "tests/stubs",
"fixtures": "tests/fixtures",
"models": "src/models",
"services": "src/services",
"jobs": "src/jobs",
"workers": "src/workers",
"exceptions": "src/exceptions",
"repositories": "src/repositories",
"console": "src/console",
"commands": "src/console/commands",
"http": "src/http",
"guards": "src/http/guards",
"controllers": "src/http/controllers",
"middlewares": "src/http/middlewares",
"interceptors": "src/http/interceptors",
"terminators": "src/http/terminators",
"validators": "src/validators",
"cron": "src/cron",
"schedulers": "src/cron/schedulers",
"config": "src/config",
"database": "src/database",
"seeders": "src/database/seeders",
"migrations": "src/database/migrations",
"lang": "src/lang",
"resources": "src/resources",
"views": "src/resources/views",
"locales": "src/resources/locales",
"providers": "src/providers",
"facades": "src/facades",
"routes": "src/routes",
"storage": "src/storage",
"logs": "src/storage/logs"
}
}

The paths above are the default ones used by Athenna to resolve your application paths. You can change one of them or many if you want, your directories defined in directories property @@ -134,18 +137,18 @@

The internally by the framework, like configuration file, route files, entry points and many others. Changing the directories property could be very useful when you are building your own project structure.

-

The controllers property​

+

The controllers property​

An array with the controllers of your application. The controllers registered in this array will be registered in the service container to be accessed easily by your Route facade:

-
{
"controllers": [
"#src/http/controllers/AppController",
"./src/http/controllers/OtherController.js"
]
}
-

The middlewares property​

+
.athennarc.json
{
"controllers": [
"#src/http/controllers/AppController",
"./src/http/controllers/OtherController.js"
]
}
+

The middlewares property​

An array with the middlewares of your application. The middlewares registered in this array will be registered in the service container to be accessed easily by your Route facade:

-
{
"middlewares": [
"#src/http/middlewares/AppMiddleware",
"./src/http/interceptors/AppInterceptor.js"
]
}
+
.athennarc.json
{
"middlewares": [
"#src/http/middlewares/AppMiddleware",
"./src/http/interceptors/AppInterceptor.js"
]
}

Athenna expects that the middlewares set in this property to be annotated with @Middleware(), @Interceptor() or @Terminator() annotations, this is not mandatory, but you will only @@ -158,31 +161,31 @@

The property to register global middlewares.

tip

More information about middlewares could be found at http middlewares documentation section.

-

The namedMiddlewares property​

+

The namedMiddlewares property​

Map the named middlewares of your application. Named -middlewares could be configured using the @Middleware +middlewares could be configured using the @Middleware() annotation, but if you are not using TypeScript in your application, you can use this object to map your named middlewares. Named middlewares are very useful to be used in your route declaration:

-
{
"namedMiddlewares": {
"app": "#src/http/middlewares/AppMiddleware",
"intercept": "./src/http/interceptors/AppInterceptor.js"
}
}
-

The globalMiddlewares property​

+
.athennarc.json
{
"namedMiddlewares": {
"app": "#src/http/middlewares/AppMiddleware",
"intercept": "./src/http/interceptors/AppInterceptor.js"
}
}
+

The globalMiddlewares property​

An array with the global middlewares of your application. -Global middlewares could be configured using the @Middleware +Global middlewares could be configured using the @Middleware() annotation, but if you are not using TypeScript in your application, you can use this object to map your named middlewares. Global middlewares are executed every time in any request of your application:

-
{
"globalMiddlewares": [
"#src/http/middlewares/AppMiddleware",
"./src/http/interceptors/AppInterceptor.js"
]
}
-

The artisan property​

+
.athennarc.json
{
"globalMiddlewares": [
"#src/http/middlewares/AppMiddleware",
"./src/http/interceptors/AppInterceptor.js"
]
}
+

The artisan property​

An object with a variety of Artisan configurations. At this point the only configurations accepted are artisan.child.executor and artisan.child.path. Both configurations are used to define how the Artisan.callInChild() method will behave when no options are set to it as second argument:

-
{
"artisan": {
"child": {
"executor": "node --inspect",
"path": "./bin/artisan.ts"
}
}
}
+
.athennarc.json
{
"artisan": {
"child": {
"executor": "node --inspect",
"path": "./bin/artisan.ts"
}
}
}
tip

Athenna will automatically parse the artisan.child.path using the Path.ext() method, so you don't need to worry about if the -extension is .js or .ts.

+extension is .js or .ts.

\ No newline at end of file diff --git a/docs/getting-started/configuration.html b/docs/getting-started/configuration.html index e940ad4e..c5f6f1e2 100644 --- a/docs/getting-started/configuration.html +++ b/docs/getting-started/configuration.html @@ -18,11 +18,11 @@ - + -

Configuration

+ +configuration values to your application's end users.

\ No newline at end of file diff --git a/docs/getting-started/directory-structure.html b/docs/getting-started/directory-structure.html index 814f9e23..a85b5cf7 100644 --- a/docs/getting-started/directory-structure.html +++ b/docs/getting-started/directory-structure.html @@ -18,11 +18,11 @@ - + -

Directory Structure

+

Directory Structure

Understand the directory structure of your project.

important

We highly recommend you to use the node artisan make command to generate the files of your application. If diff --git a/docs/getting-started/installation.html b/docs/getting-started/installation.html index f51c0418..a724f1c9 100644 --- a/docs/getting-started/installation.html +++ b/docs/getting-started/installation.html @@ -18,11 +18,11 @@ - + -

Installation

+ +on schedulers.

\ No newline at end of file diff --git a/docs/orm/annotations.html b/docs/orm/annotations.html index d8bbd2ce..836d26f7 100644 --- a/docs/orm/annotations.html +++ b/docs/orm/annotations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/orm/extending-models.html b/docs/orm/extending-models.html index f31adb00..449f4e66 100644 --- a/docs/orm/extending-models.html +++ b/docs/orm/extending-models.html @@ -18,7 +18,7 @@ - + diff --git a/docs/orm/factories.html b/docs/orm/factories.html index 9b3e0391..0180d404 100644 --- a/docs/orm/factories.html +++ b/docs/orm/factories.html @@ -18,7 +18,7 @@ - + diff --git a/docs/orm/getting-started.html b/docs/orm/getting-started.html index f8026dd2..b7088e01 100644 --- a/docs/orm/getting-started.html +++ b/docs/orm/getting-started.html @@ -18,7 +18,7 @@ - + diff --git a/docs/orm/query-builder.html b/docs/orm/query-builder.html index 944556e6..797e98c9 100644 --- a/docs/orm/query-builder.html +++ b/docs/orm/query-builder.html @@ -18,7 +18,7 @@ - + diff --git a/docs/orm/relationships.html b/docs/orm/relationships.html index 49c77fe2..fad9a12d 100644 --- a/docs/orm/relationships.html +++ b/docs/orm/relationships.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/annotations.html b/docs/rest-api-application/annotations.html index b5cc15ad..edee3bed 100644 --- a/docs/rest-api-application/annotations.html +++ b/docs/rest-api-application/annotations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/controllers.html b/docs/rest-api-application/controllers.html index b4c0d615..1161d0ea 100644 --- a/docs/rest-api-application/controllers.html +++ b/docs/rest-api-application/controllers.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/error-handling.html b/docs/rest-api-application/error-handling.html index 760198c4..da7b3556 100644 --- a/docs/rest-api-application/error-handling.html +++ b/docs/rest-api-application/error-handling.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/middlewares.html b/docs/rest-api-application/middlewares.html index 761654a5..06ccb3cb 100644 --- a/docs/rest-api-application/middlewares.html +++ b/docs/rest-api-application/middlewares.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/rate-limiting.html b/docs/rest-api-application/rate-limiting.html index c6ec522a..17966840 100644 --- a/docs/rest-api-application/rate-limiting.html +++ b/docs/rest-api-application/rate-limiting.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/request-context.html b/docs/rest-api-application/request-context.html index 3f4b980f..5275bf99 100644 --- a/docs/rest-api-application/request-context.html +++ b/docs/rest-api-application/request-context.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/routing.html b/docs/rest-api-application/routing.html index 8fbcaf2c..9b155943 100644 --- a/docs/rest-api-application/routing.html +++ b/docs/rest-api-application/routing.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/security-with-helmet.html b/docs/rest-api-application/security-with-helmet.html index 9d9c7dea..ca8b9b6c 100644 --- a/docs/rest-api-application/security-with-helmet.html +++ b/docs/rest-api-application/security-with-helmet.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/static-files.html b/docs/rest-api-application/static-files.html index 7481ca71..5ad2ddbf 100644 --- a/docs/rest-api-application/static-files.html +++ b/docs/rest-api-application/static-files.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/swagger-documentation.html b/docs/rest-api-application/swagger-documentation.html index 9ab8a2ca..03b55633 100644 --- a/docs/rest-api-application/swagger-documentation.html +++ b/docs/rest-api-application/swagger-documentation.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/tracing-requests.html b/docs/rest-api-application/tracing-requests.html index eee2aa50..17ae80b4 100644 --- a/docs/rest-api-application/tracing-requests.html +++ b/docs/rest-api-application/tracing-requests.html @@ -18,7 +18,7 @@ - + diff --git a/docs/rest-api-application/views.html b/docs/rest-api-application/views.html index 27cea6ba..e2f06deb 100644 --- a/docs/rest-api-application/views.html +++ b/docs/rest-api-application/views.html @@ -18,7 +18,7 @@ - + diff --git a/docs/testing/annotations.html b/docs/testing/annotations.html index 3e494a7a..be194602 100644 --- a/docs/testing/annotations.html +++ b/docs/testing/annotations.html @@ -18,7 +18,7 @@ - + diff --git a/docs/testing/cli-tests.html b/docs/testing/cli-tests.html index da31f729..3250713c 100644 --- a/docs/testing/cli-tests.html +++ b/docs/testing/cli-tests.html @@ -18,7 +18,7 @@ - + diff --git a/docs/testing/getting-started.html b/docs/testing/getting-started.html index 0c968e14..ecb31dce 100644 --- a/docs/testing/getting-started.html +++ b/docs/testing/getting-started.html @@ -18,7 +18,7 @@ - + diff --git a/docs/testing/mocking.html b/docs/testing/mocking.html index 0eeffa1b..ea65dd8c 100644 --- a/docs/testing/mocking.html +++ b/docs/testing/mocking.html @@ -18,7 +18,7 @@ - + diff --git a/docs/testing/rest-api-testing.html b/docs/testing/rest-api-testing.html index b9fc9b9c..d00cb492 100644 --- a/docs/testing/rest-api-testing.html +++ b/docs/testing/rest-api-testing.html @@ -18,7 +18,7 @@ - + diff --git a/docs/the-basics/compilation.html b/docs/the-basics/compilation.html index 1f36bbb2..98e5671c 100644 --- a/docs/the-basics/compilation.html +++ b/docs/the-basics/compilation.html @@ -18,7 +18,7 @@ - + diff --git a/docs/the-basics/deployment.html b/docs/the-basics/deployment.html index bfc64206..5bb43c15 100644 --- a/docs/the-basics/deployment.html +++ b/docs/the-basics/deployment.html @@ -18,7 +18,7 @@ - + diff --git a/docs/the-basics/helpers.html b/docs/the-basics/helpers.html index e04cdbb4..c8fd020d 100644 --- a/docs/the-basics/helpers.html +++ b/docs/the-basics/helpers.html @@ -18,7 +18,7 @@ - + diff --git a/docs/the-basics/logging.html b/docs/the-basics/logging.html index 0f7efda3..112c48ac 100644 --- a/docs/the-basics/logging.html +++ b/docs/the-basics/logging.html @@ -18,7 +18,7 @@ - + diff --git a/index.html b/index.html index 5f20077b..06730167 100644 --- a/index.html +++ b/index.html @@ -18,7 +18,7 @@ - + diff --git a/sw.js b/sw.js index b2536974..a44edd46 100644 --- a/sw.js +++ b/sw.js @@ -4554,7 +4554,7 @@ function getPossibleURLs(url) { (async () => { const params = parseSwParams(); // eslint-disable-next-line no-underscore-dangle - const precacheManifest = [{"revision":"70e4e5a28c341e0616cc9fb3f43cf611","url":"404.html"},{"revision":"f4a4d2b1f3c75aa844da99e846418970","url":"assets/css/styles.48bfade9.css"},{"revision":"2e382f6be70f6a46db82cae3d9f8ed79","url":"assets/js/0058b4c6.3377be63.js"},{"revision":"ce7692ef4f3b147260f64c48f9dc75e2","url":"assets/js/08133570.7810342c.js"},{"revision":"1637502a88b36b2bfd295f3c190990eb","url":"assets/js/0b6406e9.a27a161b.js"},{"revision":"e19084f4840e284d62d0c142caef9330","url":"assets/js/1169.da74cb51.js"},{"revision":"9676bd72b42fdf013ca9c82a51d42561","url":"assets/js/1176.3fa1f1a0.js"},{"revision":"97de0e8aa367d33e8466421f60e966d5","url":"assets/js/1245.0093dbf5.js"},{"revision":"7d0a87bfe1471fb4e0b4c15b6a56666f","url":"assets/js/130.97c3c1fe.js"},{"revision":"f641f3ab97c93b5004a4dce717bbc142","url":"assets/js/1303.3d23202d.js"},{"revision":"6272b766d7d455d86156627389112e72","url":"assets/js/1331.b04ae620.js"},{"revision":"104c301a23975290127ea90aec1feb6d","url":"assets/js/1398.b34f2eb3.js"},{"revision":"511ce3c91731fd38943d367adb6e317b","url":"assets/js/162.f053b2a6.js"},{"revision":"b2f5fca232498f368ae01015aa276171","url":"assets/js/17896441.c1b96131.js"},{"revision":"da235e0ebf45c2b02ae2068f0cba14c0","url":"assets/js/1946.c1f898cd.js"},{"revision":"09006696bdd7024c0fc2001fded77004","url":"assets/js/1b2a18d5.3697ed96.js"},{"revision":"8f0811f11d0775291d9ff6da536a1e8c","url":"assets/js/1bfe5704.a89fa12e.js"},{"revision":"11ee0a784c33bd08e10b72c36fd70508","url":"assets/js/1cb4e3ed.4ef422e6.js"},{"revision":"b16db22358a337c0bd47fc5ac3ec88e9","url":"assets/js/1df93b7f.50f773da.js"},{"revision":"d27177b7fcee725ee2bd3421d9dd597c","url":"assets/js/1f391b9e.c2f91780.js"},{"revision":"361e6663f794cf23e05d1953dfc3f646","url":"assets/js/2130.e743b6f2.js"},{"revision":"e6a1fbd390c07c064d4e02fb32b8d8fb","url":"assets/js/21dc2778.d0235ee8.js"},{"revision":"7784d94fcfcac0cce8f086a9f6720948","url":"assets/js/2237.e35c7faa.js"},{"revision":"301783b3d03e40cb9438ff7515890727","url":"assets/js/2362.92df48f8.js"},{"revision":"d9021cfddd0a28239b257455099f0e74","url":"assets/js/2376.89729940.js"},{"revision":"2bd071d713f0d0616b03b807629374e8","url":"assets/js/2453.4efe8b38.js"},{"revision":"0be4901d545800984f1d5931131c4005","url":"assets/js/2548.b0f7735f.js"},{"revision":"151d3dd5067c781a83e091237c231136","url":"assets/js/25ef7947.1714955d.js"},{"revision":"1a9a9aed86896f2e95f240adc45b5c92","url":"assets/js/26455d6c.54502484.js"},{"revision":"3e1b3ba8cea7f71cf7450292c52da95f","url":"assets/js/2843.88f75fed.js"},{"revision":"24ac0bf5e9341c53d59e6279ad4a7550","url":"assets/js/28a593c7.2deffc28.js"},{"revision":"4594654b9a201cd2dc3208df9a9ea77b","url":"assets/js/2925.9d15cf56.js"},{"revision":"c933425263e5857bac88f47d8e40fea4","url":"assets/js/2983.44d58df4.js"},{"revision":"89bce76e57ad8d2cb5abcb0fecc65934","url":"assets/js/2ab4562b.80d506c1.js"},{"revision":"c66e99ecec45e367497c5cea326ee300","url":"assets/js/303.d3323750.js"},{"revision":"314d230a31c13d78509edf27404fc6ca","url":"assets/js/3068.a770ed84.js"},{"revision":"f6b831968ed3bae6202f28bc096971c7","url":"assets/js/30d27832.8c09266b.js"},{"revision":"910a3f90ed1d5bf104a6b3ae199f77bd","url":"assets/js/31058b07.0f0edd0e.js"},{"revision":"13ea34fb1d796e2fcda54866102b31f1","url":"assets/js/331.72938e66.js"},{"revision":"0948ba023ab737e06a37834d16a13e53","url":"assets/js/3626.5a9700a2.js"},{"revision":"aee193736b1d01c4b8bee1a180dd8f31","url":"assets/js/3706.59d3c775.js"},{"revision":"872f311d4e7d5b8cf5f0bec59a3a8b5b","url":"assets/js/376.396be82d.js"},{"revision":"2a9b9f46cf53610df832f8b425a144c2","url":"assets/js/391d9c20.80cef5a3.js"},{"revision":"0ea93c15ba6f6957735a426c6393cdbf","url":"assets/js/3aefb2a9.5e2f1293.js"},{"revision":"e3dfa748dfd86e375c823edc3d373523","url":"assets/js/3ebc3d67.710b7f21.js"},{"revision":"fbee006c3a5ad303d72a470be42fb79a","url":"assets/js/3f163355.68d2b803.js"},{"revision":"65f39975996a5546f6405f4aa61f94a4","url":"assets/js/3fb3fb7f.82cdc3c7.js"},{"revision":"f88d2e8046199f72af205705330df105","url":"assets/js/4132.f281ab14.js"},{"revision":"a3c92338c620521850c05b63412690d8","url":"assets/js/4162.d00bd1cc.js"},{"revision":"df91d6d96252b0a5283b2925c35fd292","url":"assets/js/420.d1a95da0.js"},{"revision":"d8b2f50e880b95812b4ce903d4b72895","url":"assets/js/426.207006fe.js"},{"revision":"9402f8182b4dcc1a3512eccd611e15a9","url":"assets/js/4334.66dbfd08.js"},{"revision":"7d3c733ccd99aed8d3e974aa2084d369","url":"assets/js/453.317c5703.js"},{"revision":"9c52d23b53494f4f9254edb49496e817","url":"assets/js/4741.8e6c530a.js"},{"revision":"68636cac918b9235cc7a8892b2cd76eb","url":"assets/js/478.de33117c.js"},{"revision":"5a96109ec6e88ad1d89351b2e66a9134","url":"assets/js/4943.60400b58.js"},{"revision":"9ce6aacefe090fc260a64e0f5564e144","url":"assets/js/4a3e27c5.dedb7468.js"},{"revision":"85f99c5efc076e9bf246dbefaf7fa020","url":"assets/js/4ca25bb5.db049c7f.js"},{"revision":"9b51d828eb6dc661951d9d18b29f9e75","url":"assets/js/50bfad63.bed9431a.js"},{"revision":"818fcd6957c4f66f7fd2579efc65b309","url":"assets/js/59190afd.a05b5ce2.js"},{"revision":"a0c0abfca02f5fc2205c03afbb7a337d","url":"assets/js/5e95c892.506f1936.js"},{"revision":"6155ddd2b7951d1a6e2343f0b505509a","url":"assets/js/60418129.0dee389f.js"},{"revision":"8e35bd9e5b1643de4840dfeaf0025c83","url":"assets/js/62e3c86b.35430c50.js"},{"revision":"5f016306ee66936fc7401e9312d4405c","url":"assets/js/635.0fc9bd4f.js"},{"revision":"503f7ee2577a64fbad78d444807da639","url":"assets/js/6420.f2879ce4.js"},{"revision":"3cee8f0242c7d98cc4a851b0ca7ae6dc","url":"assets/js/6459b84b.41354f80.js"},{"revision":"3d071628b68dca319c3e4b29e660d9cb","url":"assets/js/6788.80aec321.js"},{"revision":"008a2d54c59524f78c911d00265d0dac","url":"assets/js/68.7e6af8bd.js"},{"revision":"fb3c169e589d2b3c62f60450c85fdd78","url":"assets/js/6803.73ef982b.js"},{"revision":"80702cc5deb1f19fcbd53abea09e72a9","url":"assets/js/689.51613cea.js"},{"revision":"328fa6d6a14915ff2fe1af4e86d926b3","url":"assets/js/706.956d7499.js"},{"revision":"26fa4010f5d25f51dba1e3610047eb16","url":"assets/js/741.a0ad8db0.js"},{"revision":"0a1f199fe54ff7372217825bca28c80e","url":"assets/js/7426.280729df.js"},{"revision":"c296db481d58ac204258dcce959a0238","url":"assets/js/751a75f6.4bd9a970.js"},{"revision":"afd16cad1bab2a30e54f306ca0aebb11","url":"assets/js/76be5683.52c4503c.js"},{"revision":"e23c39e1f3d33a50d9d07f88cec7f3ed","url":"assets/js/77db9a17.cfdd980b.js"},{"revision":"56e82037fd201ab604c6b2623ba5e704","url":"assets/js/788.d55fe659.js"},{"revision":"5cf362339d506112dc97e1a2ac6c0bf3","url":"assets/js/801e3691.ae983ce3.js"},{"revision":"2735219e01aaa6d9b48dd51af32e6153","url":"assets/js/803.3585a98c.js"},{"revision":"57e93e8efa643d883f244b40a12c115b","url":"assets/js/8055.b792f2c6.js"},{"revision":"7dd848e9aca85810606fc7dae8abcbe5","url":"assets/js/807b6800.dca80451.js"},{"revision":"304cd96309fb1ee9ca04c68932470836","url":"assets/js/810.354b7339.js"},{"revision":"5e6dade83e13edd8a7ba47b6e2a05b91","url":"assets/js/8337.8759c63e.js"},{"revision":"52f4e0d8af9de48a15e57f63fc5e5765","url":"assets/js/8416df86.1f125844.js"},{"revision":"a6b5606fae934df93f0975f78387ffb4","url":"assets/js/843.b93daff8.js"},{"revision":"d3da917bbea37bd61b0afbd34a6fad75","url":"assets/js/8478.e7701138.js"},{"revision":"380625def308a5650c46a9c63a08ff9c","url":"assets/js/84dac133.690e12ba.js"},{"revision":"8dfe7e88a132eb3b998a48caecc59204","url":"assets/js/8635.75cbcfbe.js"},{"revision":"d3ce496745506dc9a9764ce84b904c46","url":"assets/js/8676.ee4f37b1.js"},{"revision":"878110b814c19daded985b82542ecab9","url":"assets/js/869.4d52d7e8.js"},{"revision":"45c4655f2ed9c325a05d732990a90586","url":"assets/js/8810.92902e39.js"},{"revision":"6ddd2f739474410e8c8108ec5d3eea97","url":"assets/js/8869.9590b926.js"},{"revision":"d5b45058bb78ceaf47c103142180426a","url":"assets/js/8f88d278.aea094c1.js"},{"revision":"4ccb76c39560f5657fa64ac523df7a08","url":"assets/js/902.d73a9e87.js"},{"revision":"0311d0a3dddf5468cd49d8d0c75ba42e","url":"assets/js/925.ecb05b21.js"},{"revision":"a1631cc7a89e7134d505d505d37c23d4","url":"assets/js/943.e677d0e4.js"},{"revision":"19ad11b257d2b8074ee6cf0aa1306d66","url":"assets/js/946.10626dca.js"},{"revision":"a7721ca061e0ae0a713c14933ff6f90f","url":"assets/js/94cba6ed.a30398e0.js"},{"revision":"172a69a6da0187def7abdaf5434d5404","url":"assets/js/94e1aa3d.4adec87c.js"},{"revision":"4141b3fbff38cfddc9c8e81408e19ce2","url":"assets/js/9689.74e05a4b.js"},{"revision":"28af91062b612aa3c3d466bd4cbe6cac","url":"assets/js/96c97f44.777fb791.js"},{"revision":"e12eeca0e99f204a095ffa0d0cf7df56","url":"assets/js/9730.aa51d1f6.js"},{"revision":"c331060c405380d2056a23bdfb397b50","url":"assets/js/98b8cc29.41262bc9.js"},{"revision":"4fb659fe8b9e903abe84b9d277bd9faf","url":"assets/js/a21dcd43.bf6ad62c.js"},{"revision":"0b476082a17adb6ee5c87bcf7afbdfdc","url":"assets/js/a2c501c8.3cf5e88c.js"},{"revision":"972fcea3eb9f6d3ead2429472ae00d9d","url":"assets/js/a4d3e054.763b2a62.js"},{"revision":"b6c9061fadabbe896a80be2b8177d539","url":"assets/js/a7456010.899d59d8.js"},{"revision":"fb16e500e22a794860ed50bef7b4486c","url":"assets/js/a7bd4aaa.288b6c02.js"},{"revision":"8d9eae61cc1539851146c6ef7c88c945","url":"assets/js/a94703ab.27dd5686.js"},{"revision":"3ea4ca2d392401fe4069db26fe8a274c","url":"assets/js/aba21aa0.9e876600.js"},{"revision":"ba5d640d504070c4a29d0de3f42df769","url":"assets/js/ad6f9ff5.a065e090.js"},{"revision":"609f464694b3d45758d54cb1cd1356f6","url":"assets/js/ba3d4959.1dc5260b.js"},{"revision":"a36c2328096808a592b1a70e4a3388ca","url":"assets/js/bfc34fa7.f99ddc31.js"},{"revision":"1daaaa3b98eaed032a0788fdedb0ad84","url":"assets/js/bfc576a4.5f25d85d.js"},{"revision":"a37dcebd7e162b38ec5961857f8434fb","url":"assets/js/c24fb5a2.e249ccb5.js"},{"revision":"e70e15e171366412e7e3d81897372bb5","url":"assets/js/cbe663fe.c30f3794.js"},{"revision":"324e5722bd1ed00aa996d5422da872a3","url":"assets/js/cc784980.8e829d8e.js"},{"revision":"ef45b512f0113812d7247cd3ef5273d0","url":"assets/js/d295d49b.7eea1fd8.js"},{"revision":"63e12462d525482045cb28db429f1941","url":"assets/js/d550161c.42e59631.js"},{"revision":"153f2577027e180d03b56abcff4555e6","url":"assets/js/e2eeca55.706e84f8.js"},{"revision":"c4b090b2fe37c6528e11051b0388c86c","url":"assets/js/e6388bba.9f60e687.js"},{"revision":"bd39f78f136bc8e1c4b26e1ffab067f6","url":"assets/js/e71332dd.083d306c.js"},{"revision":"29f54d6c43c6c26f220c154ed69b3be1","url":"assets/js/ead3bc46.2dc2ba37.js"},{"revision":"cd793c19d99a0387970e46291e6148a9","url":"assets/js/f210ba15.27977df4.js"},{"revision":"6bca8ff242de9bd6b29e475879ab4c91","url":"assets/js/f3e8f525.b365d540.js"},{"revision":"c562bbb0ebcc03d4a2c2aea45e32fee9","url":"assets/js/f991d430.a6763569.js"},{"revision":"a1f25a51db8ac0c550e91084b87986a1","url":"assets/js/main.41cc9ba1.js"},{"revision":"64fb6b9e7adc1207da6d2e844451c4a7","url":"assets/js/reactPlayerDailyMotion.bbf6379c.js"},{"revision":"e65fc68b3fdc6533cef3ffb92b4a5ed4","url":"assets/js/reactPlayerDailyMotion.ea480b77.js"},{"revision":"72051fb6e310f591288930d2e523d832","url":"assets/js/reactPlayerFacebook.1dd4863c.js"},{"revision":"4f794ec9df4fad7372b6e732450e8bc2","url":"assets/js/reactPlayerFacebook.582f7f1c.js"},{"revision":"77f6ce7877af18c507e65deba3415815","url":"assets/js/reactPlayerFilePlayer.bef56796.js"},{"revision":"8238ccb6b74d71b0d7876beb72ddd0a7","url":"assets/js/reactPlayerFilePlayer.fdcd1f51.js"},{"revision":"9e3aef44b18034d108b675ecd3ebf3f0","url":"assets/js/reactPlayerKaltura.065e45b1.js"},{"revision":"7d0b48f3535eb6ea36591253885ddf39","url":"assets/js/reactPlayerKaltura.079e6b00.js"},{"revision":"cc474f3399e102bcb6ced6d91fa7eb4f","url":"assets/js/reactPlayerMixcloud.a3fb461d.js"},{"revision":"287eb640cb324071ca486f02f9c96db3","url":"assets/js/reactPlayerMixcloud.c3e2db4a.js"},{"revision":"c357e73960498eacd122154aafbee4ec","url":"assets/js/reactPlayerMux.7e2e3f86.js"},{"revision":"b502841506cb5e660cb3322ca25eeca0","url":"assets/js/reactPlayerMux.93d5edbd.js"},{"revision":"69bf495b0b098442571d77d9ec0ea42e","url":"assets/js/reactPlayerPreview.5615f48a.js"},{"revision":"5b75706d2e33584ee907b3177327bc89","url":"assets/js/reactPlayerPreview.9d736f65.js"},{"revision":"86c37c22d0468260d0f1ed661ff5a2a0","url":"assets/js/reactPlayerSoundCloud.97cc8ff6.js"},{"revision":"ee70268c4f073d4329829299a68a6b27","url":"assets/js/reactPlayerSoundCloud.d89f0440.js"},{"revision":"0f866ba51a93b2a04cde9c1ab38ca42f","url":"assets/js/reactPlayerStreamable.ac5473ea.js"},{"revision":"11369fa54e11d259dc98e3565d5d6c3c","url":"assets/js/reactPlayerStreamable.d93f5569.js"},{"revision":"dc6e644046a7eb921a5ba34a1de1e3ee","url":"assets/js/reactPlayerTwitch.55051469.js"},{"revision":"59b9ebdedbda9512c62937c02c69fa25","url":"assets/js/reactPlayerTwitch.cc430dc4.js"},{"revision":"8c1d88b9b11a315baf6ccfba456d3075","url":"assets/js/reactPlayerVidyard.090f167f.js"},{"revision":"37521995b329b94a2a52f72d265b5bd3","url":"assets/js/reactPlayerVidyard.f5b36523.js"},{"revision":"d2f5dfb094aca90b6ca9426c7a78fcc6","url":"assets/js/reactPlayerVimeo.3dff7ecf.js"},{"revision":"91d5901ea9ca6658bcb602876a4ac8fa","url":"assets/js/reactPlayerVimeo.6ee50fc8.js"},{"revision":"d8874fdbc9d9d6ddd0ae49bbcf07c9a7","url":"assets/js/reactPlayerWistia.5498c2c2.js"},{"revision":"329d11437e736ab71a834280237e461b","url":"assets/js/reactPlayerWistia.de05d934.js"},{"revision":"6373aaa592361db789212dd58d4770f5","url":"assets/js/reactPlayerYouTube.6664bacb.js"},{"revision":"0ae73eccc2c4f6cffb240549dd7f5191","url":"assets/js/reactPlayerYouTube.ac7a68b5.js"},{"revision":"04ce851ee3d539ff758d02d692db0938","url":"assets/js/runtime~main.2bb103db.js"},{"revision":"f52adfe652a503d5574d814e61212c21","url":"benchmarks.html"},{"revision":"b797e91345d960a7545eede87b70fb9e","url":"css/alert.css"},{"revision":"8ff2bde63f90ec4d40c530b11ec6c4d6","url":"css/footer.css"},{"revision":"585116277f3360e08550f333b2ff177b","url":"css/icons.css"},{"revision":"779859016455ea4a6e386f3a43b1f436","url":"css/markdown.css"},{"revision":"44f3dd32bfc1b8f3456555ed3c87b16b","url":"css/navbar.css"},{"revision":"f5d35f3a2398cc72a32f4b668e34a11d","url":"css/root.css"},{"revision":"a392cff79a8a21773d4c2c71baa7d2e6","url":"css/sidebar.css"},{"revision":"6b7ab04959a7f321f3eb62175dcf18a1","url":"css/toc.css"},{"revision":"2954ca42cd6545af7821f825aa4f89e7","url":"css/video.css"},{"revision":"7d29938795e9dc720e0284ab3ce70f87","url":"docs/architecture-concepts/application-lifecycle.html"},{"revision":"4e0f6e81188518547471b03b67f2ba34","url":"docs/architecture-concepts/facades.html"},{"revision":"9ee7663d06af1c96b276d4abfe488b97","url":"docs/architecture-concepts/service-container.html"},{"revision":"e5f1f2f695821ff6916d003911c6cbc4","url":"docs/architecture-concepts/service-providers.html"},{"revision":"6efe96f06b954f095b288046d96a3d60","url":"docs/cli-application/annotations.html"},{"revision":"b9cca15f50439b633fb8b32fff8f630b","url":"docs/cli-application/commands.html"},{"revision":"cad8ff0663458af3504ba119acd2967a","url":"docs/cli-application/error-handling.html"},{"revision":"0b09afd1462c1f821b5a9a41b3be0c09","url":"docs/cli-application/publishing.html"},{"revision":"0008e5c43aef194eb9c42300efa7e8c0","url":"docs/cli-application/running.html"},{"revision":"6291192679b74cab07ef4cebbaf37ddc","url":"docs/cron-application/annotations.html"},{"revision":"f9ce0f4ad74baa270ab3bd5c2d5f63c0","url":"docs/cron-application/cron-context.html"},{"revision":"ac063bad1352161098d5eadf71fafd7e","url":"docs/cron-application/error-handling.html"},{"revision":"d9dbe1d7a82a5f6124f48e177a304a98","url":"docs/cron-application/schedulers.html"},{"revision":"de147cc31dd2f651797c42d30348bac6","url":"docs/cron-application/tracing-executions.html"},{"revision":"13b769f261d9ef5ec785e739d1551b61","url":"docs/database/getting-started.html"},{"revision":"f9c385c2c81094902a315c3d48f8c85f","url":"docs/database/migrations.html"},{"revision":"4f2eabfd70988426d6f328922e374e8c","url":"docs/database/query-builder.html"},{"revision":"a5357dfa1cb4131b6b54aff5df7be73a","url":"docs/database/seeding.html"},{"revision":"d58465d36c16400a542d43c9ba3525fd","url":"docs/digging-deeper/collections.html"},{"revision":"edd50573ea06c3616229be8193b8ddbb","url":"docs/digging-deeper/graceful-shutdown.html"},{"revision":"0732685bdbf8c7101682a42ee809e84c","url":"docs/digging-deeper/library-development.html"},{"revision":"aa9f5642fb9fa2dd6e6417a9a34e2107","url":"docs/digging-deeper/mail.html"},{"revision":"b8a8b254c2f479eddff20b325335cc58","url":"docs/digging-deeper/repl.html"},{"revision":"d5393471134837057a8d83897d1cd3fe","url":"docs/getting-started/athennarc-file.html"},{"revision":"981c14dad0479761248275488baee0e6","url":"docs/getting-started/configuration.html"},{"revision":"9544c9346befc506d396da34d406a0f3","url":"docs/getting-started/directory-structure.html"},{"revision":"cfae94fdc87330f8efa833f2344e7897","url":"docs/getting-started/installation.html"},{"revision":"be74acd28f0bb1604499a3df0669e34a","url":"docs/orm/annotations.html"},{"revision":"16deada47a792fe68e84e3e1114688d5","url":"docs/orm/extending-models.html"},{"revision":"a0f03800392d2ed5ff5ea94effd1a55a","url":"docs/orm/factories.html"},{"revision":"4db065bce33b55dd2a62571b7d16c1a1","url":"docs/orm/getting-started.html"},{"revision":"4f64dc01a223135365b3868323276eda","url":"docs/orm/query-builder.html"},{"revision":"4d0630ee7f10ed1ffffe75412bbdfaf9","url":"docs/orm/relationships.html"},{"revision":"1cf148f26c11584f9890677f344ea621","url":"docs/rest-api-application/annotations.html"},{"revision":"b53ab7e0f3cb5abeef99886898455b84","url":"docs/rest-api-application/controllers.html"},{"revision":"2a09d492f664fb51d9f3a5e75621d738","url":"docs/rest-api-application/error-handling.html"},{"revision":"6b7fd6428afce4fc84f740ce3b2355ce","url":"docs/rest-api-application/middlewares.html"},{"revision":"55a566641f498f2a42c467b7cb122ae5","url":"docs/rest-api-application/rate-limiting.html"},{"revision":"5455f0b9cb783d59d98f883bf34888ed","url":"docs/rest-api-application/request-context.html"},{"revision":"143862c9d7dfc03059c5a25794d0ce61","url":"docs/rest-api-application/routing.html"},{"revision":"7b96bc1de21d8ccccd58b34ed6c2b6f5","url":"docs/rest-api-application/security-with-helmet.html"},{"revision":"5b64067fc267af02e281862fd5dc576b","url":"docs/rest-api-application/static-files.html"},{"revision":"3860e14c615b8b1f8876ffd5a78e90b6","url":"docs/rest-api-application/swagger-documentation.html"},{"revision":"50a5991d25acbe986165c7d0c26d06d9","url":"docs/rest-api-application/tracing-requests.html"},{"revision":"b9357e82c5c973877e37d96e21db8c55","url":"docs/rest-api-application/views.html"},{"revision":"4f6b05401cfc91cec587bd9733165ebe","url":"docs/testing/annotations.html"},{"revision":"f2b4c21932501ee7505e56bf46724fda","url":"docs/testing/cli-tests.html"},{"revision":"fb336ec00a2e6dd5af3c2c11205772db","url":"docs/testing/getting-started.html"},{"revision":"74e082c8734f5cd6faae5fc89c5f9d52","url":"docs/testing/mocking.html"},{"revision":"7d6da2dd7f41bc4736cba5e574f21d8a","url":"docs/testing/rest-api-testing.html"},{"revision":"bce6163ab7bca53e9b87881e522a508e","url":"docs/the-basics/compilation.html"},{"revision":"b57fe09cc0ac268de7898c0abde6f450","url":"docs/the-basics/deployment.html"},{"revision":"1ec8d952183934479aeb1ee4ae2a6c6a","url":"docs/the-basics/helpers.html"},{"revision":"8fcfc5d7c792419f16bdb26961e16c5f","url":"docs/the-basics/logging.html"},{"revision":"517cafd5049da2a0707bcdf2bf841630","url":"index.html"},{"revision":"b9c2d96551764c633b0de8edf9e82b16","url":"manifest.json"},{"revision":"9b5f8e4ac9bd0649aab896c55a382ed9","url":"img/codes/http-route.png"},{"revision":"8b1a6b58c0fdfe999985be48568bcaec","url":"img/codes/ignite.png"},{"revision":"1dc71eacd5e43b82f89fce76d3c79f74","url":"img/codes/test-route.png"},{"revision":"8819ca59541f4814b6a55fcf66ecabbe","url":"img/examples/artisan-ui.png"},{"revision":"5a842a80da3f7052d79847910389ecf9","url":"img/favicons/favicon.ico"},{"revision":"6441cbb8861a71fcfd9cd468b225a998","url":"img/favicons/minerva.ico"},{"revision":"d72ef880b711475e5291c0fdfb1964b3","url":"img/logos/athena.png"},{"revision":"5a842a80da3f7052d79847910389ecf9","url":"img/logos/logo.png"},{"revision":"6441cbb8861a71fcfd9cd468b225a998","url":"img/logos/minerva.png"},{"revision":"7ea4a962be30750a0653ef693b519370","url":"img/pre-visualization/home.png"},{"revision":"38d2d7a80eba1f9ba7bbcafed917c96c","url":"fonts/AbrilText-Bold.woff"},{"revision":"9fc10a629a1d43baa01c1ec8188ea3b7","url":"fonts/AbrilText-Bold.woff2"}]; + const precacheManifest = [{"revision":"feb88cc0a5604fe66c120d055f582d84","url":"404.html"},{"revision":"f4a4d2b1f3c75aa844da99e846418970","url":"assets/css/styles.48bfade9.css"},{"revision":"97771d6fae6c0293e1ec25c04e77278a","url":"assets/js/0058b4c6.e7eea27f.js"},{"revision":"ce7692ef4f3b147260f64c48f9dc75e2","url":"assets/js/08133570.7810342c.js"},{"revision":"1637502a88b36b2bfd295f3c190990eb","url":"assets/js/0b6406e9.a27a161b.js"},{"revision":"e19084f4840e284d62d0c142caef9330","url":"assets/js/1169.da74cb51.js"},{"revision":"9676bd72b42fdf013ca9c82a51d42561","url":"assets/js/1176.3fa1f1a0.js"},{"revision":"97de0e8aa367d33e8466421f60e966d5","url":"assets/js/1245.0093dbf5.js"},{"revision":"7d0a87bfe1471fb4e0b4c15b6a56666f","url":"assets/js/130.97c3c1fe.js"},{"revision":"f641f3ab97c93b5004a4dce717bbc142","url":"assets/js/1303.3d23202d.js"},{"revision":"6272b766d7d455d86156627389112e72","url":"assets/js/1331.b04ae620.js"},{"revision":"104c301a23975290127ea90aec1feb6d","url":"assets/js/1398.b34f2eb3.js"},{"revision":"511ce3c91731fd38943d367adb6e317b","url":"assets/js/162.f053b2a6.js"},{"revision":"b2f5fca232498f368ae01015aa276171","url":"assets/js/17896441.c1b96131.js"},{"revision":"da235e0ebf45c2b02ae2068f0cba14c0","url":"assets/js/1946.c1f898cd.js"},{"revision":"09006696bdd7024c0fc2001fded77004","url":"assets/js/1b2a18d5.3697ed96.js"},{"revision":"8f0811f11d0775291d9ff6da536a1e8c","url":"assets/js/1bfe5704.a89fa12e.js"},{"revision":"11ee0a784c33bd08e10b72c36fd70508","url":"assets/js/1cb4e3ed.4ef422e6.js"},{"revision":"b16db22358a337c0bd47fc5ac3ec88e9","url":"assets/js/1df93b7f.50f773da.js"},{"revision":"d27177b7fcee725ee2bd3421d9dd597c","url":"assets/js/1f391b9e.c2f91780.js"},{"revision":"361e6663f794cf23e05d1953dfc3f646","url":"assets/js/2130.e743b6f2.js"},{"revision":"e6a1fbd390c07c064d4e02fb32b8d8fb","url":"assets/js/21dc2778.d0235ee8.js"},{"revision":"7784d94fcfcac0cce8f086a9f6720948","url":"assets/js/2237.e35c7faa.js"},{"revision":"301783b3d03e40cb9438ff7515890727","url":"assets/js/2362.92df48f8.js"},{"revision":"d9021cfddd0a28239b257455099f0e74","url":"assets/js/2376.89729940.js"},{"revision":"2bd071d713f0d0616b03b807629374e8","url":"assets/js/2453.4efe8b38.js"},{"revision":"0be4901d545800984f1d5931131c4005","url":"assets/js/2548.b0f7735f.js"},{"revision":"151d3dd5067c781a83e091237c231136","url":"assets/js/25ef7947.1714955d.js"},{"revision":"1a9a9aed86896f2e95f240adc45b5c92","url":"assets/js/26455d6c.54502484.js"},{"revision":"3e1b3ba8cea7f71cf7450292c52da95f","url":"assets/js/2843.88f75fed.js"},{"revision":"24ac0bf5e9341c53d59e6279ad4a7550","url":"assets/js/28a593c7.2deffc28.js"},{"revision":"4594654b9a201cd2dc3208df9a9ea77b","url":"assets/js/2925.9d15cf56.js"},{"revision":"c933425263e5857bac88f47d8e40fea4","url":"assets/js/2983.44d58df4.js"},{"revision":"89bce76e57ad8d2cb5abcb0fecc65934","url":"assets/js/2ab4562b.80d506c1.js"},{"revision":"c66e99ecec45e367497c5cea326ee300","url":"assets/js/303.d3323750.js"},{"revision":"314d230a31c13d78509edf27404fc6ca","url":"assets/js/3068.a770ed84.js"},{"revision":"d592f501aafd0782d7ec2cb835e8c7ad","url":"assets/js/30d27832.8b98abf6.js"},{"revision":"910a3f90ed1d5bf104a6b3ae199f77bd","url":"assets/js/31058b07.0f0edd0e.js"},{"revision":"13ea34fb1d796e2fcda54866102b31f1","url":"assets/js/331.72938e66.js"},{"revision":"0948ba023ab737e06a37834d16a13e53","url":"assets/js/3626.5a9700a2.js"},{"revision":"aee193736b1d01c4b8bee1a180dd8f31","url":"assets/js/3706.59d3c775.js"},{"revision":"872f311d4e7d5b8cf5f0bec59a3a8b5b","url":"assets/js/376.396be82d.js"},{"revision":"2a9b9f46cf53610df832f8b425a144c2","url":"assets/js/391d9c20.80cef5a3.js"},{"revision":"0ea93c15ba6f6957735a426c6393cdbf","url":"assets/js/3aefb2a9.5e2f1293.js"},{"revision":"e3dfa748dfd86e375c823edc3d373523","url":"assets/js/3ebc3d67.710b7f21.js"},{"revision":"fbee006c3a5ad303d72a470be42fb79a","url":"assets/js/3f163355.68d2b803.js"},{"revision":"65f39975996a5546f6405f4aa61f94a4","url":"assets/js/3fb3fb7f.82cdc3c7.js"},{"revision":"f88d2e8046199f72af205705330df105","url":"assets/js/4132.f281ab14.js"},{"revision":"a3c92338c620521850c05b63412690d8","url":"assets/js/4162.d00bd1cc.js"},{"revision":"df91d6d96252b0a5283b2925c35fd292","url":"assets/js/420.d1a95da0.js"},{"revision":"d8b2f50e880b95812b4ce903d4b72895","url":"assets/js/426.207006fe.js"},{"revision":"9402f8182b4dcc1a3512eccd611e15a9","url":"assets/js/4334.66dbfd08.js"},{"revision":"7d3c733ccd99aed8d3e974aa2084d369","url":"assets/js/453.317c5703.js"},{"revision":"9c52d23b53494f4f9254edb49496e817","url":"assets/js/4741.8e6c530a.js"},{"revision":"68636cac918b9235cc7a8892b2cd76eb","url":"assets/js/478.de33117c.js"},{"revision":"5a96109ec6e88ad1d89351b2e66a9134","url":"assets/js/4943.60400b58.js"},{"revision":"9ce6aacefe090fc260a64e0f5564e144","url":"assets/js/4a3e27c5.dedb7468.js"},{"revision":"85f99c5efc076e9bf246dbefaf7fa020","url":"assets/js/4ca25bb5.db049c7f.js"},{"revision":"9b51d828eb6dc661951d9d18b29f9e75","url":"assets/js/50bfad63.bed9431a.js"},{"revision":"818fcd6957c4f66f7fd2579efc65b309","url":"assets/js/59190afd.a05b5ce2.js"},{"revision":"a0c0abfca02f5fc2205c03afbb7a337d","url":"assets/js/5e95c892.506f1936.js"},{"revision":"6155ddd2b7951d1a6e2343f0b505509a","url":"assets/js/60418129.0dee389f.js"},{"revision":"8e35bd9e5b1643de4840dfeaf0025c83","url":"assets/js/62e3c86b.35430c50.js"},{"revision":"5f016306ee66936fc7401e9312d4405c","url":"assets/js/635.0fc9bd4f.js"},{"revision":"503f7ee2577a64fbad78d444807da639","url":"assets/js/6420.f2879ce4.js"},{"revision":"9402961acebaf4ab0fce4f9d9cca07f2","url":"assets/js/6459b84b.e29b05d3.js"},{"revision":"3d071628b68dca319c3e4b29e660d9cb","url":"assets/js/6788.80aec321.js"},{"revision":"008a2d54c59524f78c911d00265d0dac","url":"assets/js/68.7e6af8bd.js"},{"revision":"fb3c169e589d2b3c62f60450c85fdd78","url":"assets/js/6803.73ef982b.js"},{"revision":"80702cc5deb1f19fcbd53abea09e72a9","url":"assets/js/689.51613cea.js"},{"revision":"328fa6d6a14915ff2fe1af4e86d926b3","url":"assets/js/706.956d7499.js"},{"revision":"26fa4010f5d25f51dba1e3610047eb16","url":"assets/js/741.a0ad8db0.js"},{"revision":"0a1f199fe54ff7372217825bca28c80e","url":"assets/js/7426.280729df.js"},{"revision":"c296db481d58ac204258dcce959a0238","url":"assets/js/751a75f6.4bd9a970.js"},{"revision":"afd16cad1bab2a30e54f306ca0aebb11","url":"assets/js/76be5683.52c4503c.js"},{"revision":"e23c39e1f3d33a50d9d07f88cec7f3ed","url":"assets/js/77db9a17.cfdd980b.js"},{"revision":"56e82037fd201ab604c6b2623ba5e704","url":"assets/js/788.d55fe659.js"},{"revision":"5cf362339d506112dc97e1a2ac6c0bf3","url":"assets/js/801e3691.ae983ce3.js"},{"revision":"2735219e01aaa6d9b48dd51af32e6153","url":"assets/js/803.3585a98c.js"},{"revision":"57e93e8efa643d883f244b40a12c115b","url":"assets/js/8055.b792f2c6.js"},{"revision":"7dd848e9aca85810606fc7dae8abcbe5","url":"assets/js/807b6800.dca80451.js"},{"revision":"304cd96309fb1ee9ca04c68932470836","url":"assets/js/810.354b7339.js"},{"revision":"5e6dade83e13edd8a7ba47b6e2a05b91","url":"assets/js/8337.8759c63e.js"},{"revision":"52f4e0d8af9de48a15e57f63fc5e5765","url":"assets/js/8416df86.1f125844.js"},{"revision":"a6b5606fae934df93f0975f78387ffb4","url":"assets/js/843.b93daff8.js"},{"revision":"d3da917bbea37bd61b0afbd34a6fad75","url":"assets/js/8478.e7701138.js"},{"revision":"380625def308a5650c46a9c63a08ff9c","url":"assets/js/84dac133.690e12ba.js"},{"revision":"8dfe7e88a132eb3b998a48caecc59204","url":"assets/js/8635.75cbcfbe.js"},{"revision":"d3ce496745506dc9a9764ce84b904c46","url":"assets/js/8676.ee4f37b1.js"},{"revision":"878110b814c19daded985b82542ecab9","url":"assets/js/869.4d52d7e8.js"},{"revision":"45c4655f2ed9c325a05d732990a90586","url":"assets/js/8810.92902e39.js"},{"revision":"6ddd2f739474410e8c8108ec5d3eea97","url":"assets/js/8869.9590b926.js"},{"revision":"d5b45058bb78ceaf47c103142180426a","url":"assets/js/8f88d278.aea094c1.js"},{"revision":"4ccb76c39560f5657fa64ac523df7a08","url":"assets/js/902.d73a9e87.js"},{"revision":"0311d0a3dddf5468cd49d8d0c75ba42e","url":"assets/js/925.ecb05b21.js"},{"revision":"a1631cc7a89e7134d505d505d37c23d4","url":"assets/js/943.e677d0e4.js"},{"revision":"19ad11b257d2b8074ee6cf0aa1306d66","url":"assets/js/946.10626dca.js"},{"revision":"a7721ca061e0ae0a713c14933ff6f90f","url":"assets/js/94cba6ed.a30398e0.js"},{"revision":"172a69a6da0187def7abdaf5434d5404","url":"assets/js/94e1aa3d.4adec87c.js"},{"revision":"4141b3fbff38cfddc9c8e81408e19ce2","url":"assets/js/9689.74e05a4b.js"},{"revision":"28af91062b612aa3c3d466bd4cbe6cac","url":"assets/js/96c97f44.777fb791.js"},{"revision":"e12eeca0e99f204a095ffa0d0cf7df56","url":"assets/js/9730.aa51d1f6.js"},{"revision":"e94dc989dc13cf73bf59eac23c9233e8","url":"assets/js/98b8cc29.6c61093b.js"},{"revision":"4fb659fe8b9e903abe84b9d277bd9faf","url":"assets/js/a21dcd43.bf6ad62c.js"},{"revision":"0b476082a17adb6ee5c87bcf7afbdfdc","url":"assets/js/a2c501c8.3cf5e88c.js"},{"revision":"972fcea3eb9f6d3ead2429472ae00d9d","url":"assets/js/a4d3e054.763b2a62.js"},{"revision":"b6c9061fadabbe896a80be2b8177d539","url":"assets/js/a7456010.899d59d8.js"},{"revision":"fb16e500e22a794860ed50bef7b4486c","url":"assets/js/a7bd4aaa.288b6c02.js"},{"revision":"8d9eae61cc1539851146c6ef7c88c945","url":"assets/js/a94703ab.27dd5686.js"},{"revision":"3ea4ca2d392401fe4069db26fe8a274c","url":"assets/js/aba21aa0.9e876600.js"},{"revision":"ba5d640d504070c4a29d0de3f42df769","url":"assets/js/ad6f9ff5.a065e090.js"},{"revision":"609f464694b3d45758d54cb1cd1356f6","url":"assets/js/ba3d4959.1dc5260b.js"},{"revision":"a36c2328096808a592b1a70e4a3388ca","url":"assets/js/bfc34fa7.f99ddc31.js"},{"revision":"1daaaa3b98eaed032a0788fdedb0ad84","url":"assets/js/bfc576a4.5f25d85d.js"},{"revision":"a37dcebd7e162b38ec5961857f8434fb","url":"assets/js/c24fb5a2.e249ccb5.js"},{"revision":"e70e15e171366412e7e3d81897372bb5","url":"assets/js/cbe663fe.c30f3794.js"},{"revision":"324e5722bd1ed00aa996d5422da872a3","url":"assets/js/cc784980.8e829d8e.js"},{"revision":"ef45b512f0113812d7247cd3ef5273d0","url":"assets/js/d295d49b.7eea1fd8.js"},{"revision":"63e12462d525482045cb28db429f1941","url":"assets/js/d550161c.42e59631.js"},{"revision":"153f2577027e180d03b56abcff4555e6","url":"assets/js/e2eeca55.706e84f8.js"},{"revision":"c4b090b2fe37c6528e11051b0388c86c","url":"assets/js/e6388bba.9f60e687.js"},{"revision":"bd39f78f136bc8e1c4b26e1ffab067f6","url":"assets/js/e71332dd.083d306c.js"},{"revision":"29f54d6c43c6c26f220c154ed69b3be1","url":"assets/js/ead3bc46.2dc2ba37.js"},{"revision":"cd793c19d99a0387970e46291e6148a9","url":"assets/js/f210ba15.27977df4.js"},{"revision":"6bca8ff242de9bd6b29e475879ab4c91","url":"assets/js/f3e8f525.b365d540.js"},{"revision":"c562bbb0ebcc03d4a2c2aea45e32fee9","url":"assets/js/f991d430.a6763569.js"},{"revision":"a1f25a51db8ac0c550e91084b87986a1","url":"assets/js/main.41cc9ba1.js"},{"revision":"64fb6b9e7adc1207da6d2e844451c4a7","url":"assets/js/reactPlayerDailyMotion.bbf6379c.js"},{"revision":"e65fc68b3fdc6533cef3ffb92b4a5ed4","url":"assets/js/reactPlayerDailyMotion.ea480b77.js"},{"revision":"72051fb6e310f591288930d2e523d832","url":"assets/js/reactPlayerFacebook.1dd4863c.js"},{"revision":"4f794ec9df4fad7372b6e732450e8bc2","url":"assets/js/reactPlayerFacebook.582f7f1c.js"},{"revision":"77f6ce7877af18c507e65deba3415815","url":"assets/js/reactPlayerFilePlayer.bef56796.js"},{"revision":"8238ccb6b74d71b0d7876beb72ddd0a7","url":"assets/js/reactPlayerFilePlayer.fdcd1f51.js"},{"revision":"9e3aef44b18034d108b675ecd3ebf3f0","url":"assets/js/reactPlayerKaltura.065e45b1.js"},{"revision":"7d0b48f3535eb6ea36591253885ddf39","url":"assets/js/reactPlayerKaltura.079e6b00.js"},{"revision":"cc474f3399e102bcb6ced6d91fa7eb4f","url":"assets/js/reactPlayerMixcloud.a3fb461d.js"},{"revision":"287eb640cb324071ca486f02f9c96db3","url":"assets/js/reactPlayerMixcloud.c3e2db4a.js"},{"revision":"c357e73960498eacd122154aafbee4ec","url":"assets/js/reactPlayerMux.7e2e3f86.js"},{"revision":"b502841506cb5e660cb3322ca25eeca0","url":"assets/js/reactPlayerMux.93d5edbd.js"},{"revision":"69bf495b0b098442571d77d9ec0ea42e","url":"assets/js/reactPlayerPreview.5615f48a.js"},{"revision":"5b75706d2e33584ee907b3177327bc89","url":"assets/js/reactPlayerPreview.9d736f65.js"},{"revision":"86c37c22d0468260d0f1ed661ff5a2a0","url":"assets/js/reactPlayerSoundCloud.97cc8ff6.js"},{"revision":"ee70268c4f073d4329829299a68a6b27","url":"assets/js/reactPlayerSoundCloud.d89f0440.js"},{"revision":"0f866ba51a93b2a04cde9c1ab38ca42f","url":"assets/js/reactPlayerStreamable.ac5473ea.js"},{"revision":"11369fa54e11d259dc98e3565d5d6c3c","url":"assets/js/reactPlayerStreamable.d93f5569.js"},{"revision":"dc6e644046a7eb921a5ba34a1de1e3ee","url":"assets/js/reactPlayerTwitch.55051469.js"},{"revision":"59b9ebdedbda9512c62937c02c69fa25","url":"assets/js/reactPlayerTwitch.cc430dc4.js"},{"revision":"8c1d88b9b11a315baf6ccfba456d3075","url":"assets/js/reactPlayerVidyard.090f167f.js"},{"revision":"37521995b329b94a2a52f72d265b5bd3","url":"assets/js/reactPlayerVidyard.f5b36523.js"},{"revision":"d2f5dfb094aca90b6ca9426c7a78fcc6","url":"assets/js/reactPlayerVimeo.3dff7ecf.js"},{"revision":"91d5901ea9ca6658bcb602876a4ac8fa","url":"assets/js/reactPlayerVimeo.6ee50fc8.js"},{"revision":"d8874fdbc9d9d6ddd0ae49bbcf07c9a7","url":"assets/js/reactPlayerWistia.5498c2c2.js"},{"revision":"329d11437e736ab71a834280237e461b","url":"assets/js/reactPlayerWistia.de05d934.js"},{"revision":"6373aaa592361db789212dd58d4770f5","url":"assets/js/reactPlayerYouTube.6664bacb.js"},{"revision":"0ae73eccc2c4f6cffb240549dd7f5191","url":"assets/js/reactPlayerYouTube.ac7a68b5.js"},{"revision":"4212e3a28782b2d89b7b6ceeb46a94db","url":"assets/js/runtime~main.e4720fcb.js"},{"revision":"c52268c4921bbf4a1f5e0c0aa68c8012","url":"benchmarks.html"},{"revision":"b797e91345d960a7545eede87b70fb9e","url":"css/alert.css"},{"revision":"8ff2bde63f90ec4d40c530b11ec6c4d6","url":"css/footer.css"},{"revision":"585116277f3360e08550f333b2ff177b","url":"css/icons.css"},{"revision":"779859016455ea4a6e386f3a43b1f436","url":"css/markdown.css"},{"revision":"44f3dd32bfc1b8f3456555ed3c87b16b","url":"css/navbar.css"},{"revision":"f5d35f3a2398cc72a32f4b668e34a11d","url":"css/root.css"},{"revision":"a392cff79a8a21773d4c2c71baa7d2e6","url":"css/sidebar.css"},{"revision":"6b7ab04959a7f321f3eb62175dcf18a1","url":"css/toc.css"},{"revision":"2954ca42cd6545af7821f825aa4f89e7","url":"css/video.css"},{"revision":"ed0c54b1c2cb1f8ec3345588869ce3bc","url":"docs/architecture-concepts/application-lifecycle.html"},{"revision":"d61d264efb85c4fb9f242b7f94e8d311","url":"docs/architecture-concepts/facades.html"},{"revision":"d896f9c8ca398ab9212db2aa9c67a3e0","url":"docs/architecture-concepts/service-container.html"},{"revision":"d11d4cb5226f19fda957ac6e70b7c63f","url":"docs/architecture-concepts/service-providers.html"},{"revision":"3ed9a4543e0c4baab4e7c11702a850b1","url":"docs/cli-application/annotations.html"},{"revision":"50da770bf2053b655005eac9510f83ea","url":"docs/cli-application/commands.html"},{"revision":"ec33d6cf3409edff1634c9d8d3ddb5d4","url":"docs/cli-application/error-handling.html"},{"revision":"d431efe747e5c7f8bcea2f476a6aebb9","url":"docs/cli-application/publishing.html"},{"revision":"953ee0e361b15aa84227429eda3b8597","url":"docs/cli-application/running.html"},{"revision":"1868a69363d17d707003c35a48b5248a","url":"docs/cron-application/annotations.html"},{"revision":"66e33a941e78bd728c2d46bc01f8940f","url":"docs/cron-application/cron-context.html"},{"revision":"c0fb2283c3d0ee879b04e50bf791c7fd","url":"docs/cron-application/error-handling.html"},{"revision":"5b5b493ff12a4963d7eda73f7268d33e","url":"docs/cron-application/schedulers.html"},{"revision":"ec155fd3f9bd55aeb2c87b1f9dd311eb","url":"docs/cron-application/tracing-executions.html"},{"revision":"d917fe3808adb8dd5e4095006a7fbb79","url":"docs/database/getting-started.html"},{"revision":"b2b8e4238717e83c4fd6bbe7bd4e163d","url":"docs/database/migrations.html"},{"revision":"8f3c04a6c8f3f4746bd4e85ba07cc10b","url":"docs/database/query-builder.html"},{"revision":"314e73b9dca94effb0f092ab5565171d","url":"docs/database/seeding.html"},{"revision":"627e236746cccb31ba72ec1eb6eb3786","url":"docs/digging-deeper/collections.html"},{"revision":"996726bb4e5967f9aeec4df667a1a818","url":"docs/digging-deeper/graceful-shutdown.html"},{"revision":"829ce3c9fa0fe05f2721226bf64b53f7","url":"docs/digging-deeper/library-development.html"},{"revision":"c6a2ef74437a74e48355f559e3642d9a","url":"docs/digging-deeper/mail.html"},{"revision":"cbe300deb7f3025214a85b781d27a18d","url":"docs/digging-deeper/repl.html"},{"revision":"0b9d35a9079f137c869813ae8c2a57d5","url":"docs/getting-started/athennarc-file.html"},{"revision":"af5a9ac9f8ac7d03dacf5a993d9c6337","url":"docs/getting-started/configuration.html"},{"revision":"ed1b68e7e1f40908798cfa27d9da8bcf","url":"docs/getting-started/directory-structure.html"},{"revision":"2f9730b4f00c946caa815cf9640a334f","url":"docs/getting-started/installation.html"},{"revision":"523e657415887376c08701d893b42c41","url":"docs/orm/annotations.html"},{"revision":"b39b22b0f30237dec12eb0a0c631be0a","url":"docs/orm/extending-models.html"},{"revision":"2db29bcbe4692c58cef919460ea5e32b","url":"docs/orm/factories.html"},{"revision":"b4f0b6eedb56d177371306363e0f740b","url":"docs/orm/getting-started.html"},{"revision":"63cbb0d516eaf7c07c596666cb414ee9","url":"docs/orm/query-builder.html"},{"revision":"926ccb65a7448c43a5e422b2d03a22e0","url":"docs/orm/relationships.html"},{"revision":"1dacdfb94862c206b1cd75175dce7768","url":"docs/rest-api-application/annotations.html"},{"revision":"ed8bfab3ca0f1e3baa6f91ccd5b75df8","url":"docs/rest-api-application/controllers.html"},{"revision":"31ede8f900f9d71ae53acbfac6e71624","url":"docs/rest-api-application/error-handling.html"},{"revision":"b8524a7d8a0c00a6884f2b20f6e51644","url":"docs/rest-api-application/middlewares.html"},{"revision":"5b8645b3a4476ff2d96501dc2dfcb7b0","url":"docs/rest-api-application/rate-limiting.html"},{"revision":"464d7d66bac3383d3548e2c37d4b6644","url":"docs/rest-api-application/request-context.html"},{"revision":"f566311160d6560ba9875dcda2f8c8d1","url":"docs/rest-api-application/routing.html"},{"revision":"66df1dda9d826e7638bb562d3326cdc3","url":"docs/rest-api-application/security-with-helmet.html"},{"revision":"9f962938e208a2e4e8aeda3a8a0b6563","url":"docs/rest-api-application/static-files.html"},{"revision":"893902e413ad461c636c7a3266162a1a","url":"docs/rest-api-application/swagger-documentation.html"},{"revision":"b46a16e08eecd7f1aee9ff39deb00f3b","url":"docs/rest-api-application/tracing-requests.html"},{"revision":"77d03868b800cdbee0efa878d7c72b91","url":"docs/rest-api-application/views.html"},{"revision":"3f08ed3c8a1be24b88ad8589b5622dd2","url":"docs/testing/annotations.html"},{"revision":"dd9107c778b7b753928f8a5a13131d96","url":"docs/testing/cli-tests.html"},{"revision":"97831566e3305cb8f905395bbdf108cb","url":"docs/testing/getting-started.html"},{"revision":"a495fe6c1233a210375c0226d7c68fef","url":"docs/testing/mocking.html"},{"revision":"5b51e437beeb8c13f8f33cdec96ca179","url":"docs/testing/rest-api-testing.html"},{"revision":"75b80bd990da25f3861440f3d66cf103","url":"docs/the-basics/compilation.html"},{"revision":"872d283f84740a47a3127e7c55801b02","url":"docs/the-basics/deployment.html"},{"revision":"8aaf6279a0dd257ee5d0f874fd28c0a9","url":"docs/the-basics/helpers.html"},{"revision":"fb15457067984a7e2947052bb62d69bd","url":"docs/the-basics/logging.html"},{"revision":"aa01aeb6ad2f430e0efe637c686791e8","url":"index.html"},{"revision":"b9c2d96551764c633b0de8edf9e82b16","url":"manifest.json"},{"revision":"9b5f8e4ac9bd0649aab896c55a382ed9","url":"img/codes/http-route.png"},{"revision":"8b1a6b58c0fdfe999985be48568bcaec","url":"img/codes/ignite.png"},{"revision":"1dc71eacd5e43b82f89fce76d3c79f74","url":"img/codes/test-route.png"},{"revision":"8819ca59541f4814b6a55fcf66ecabbe","url":"img/examples/artisan-ui.png"},{"revision":"5a842a80da3f7052d79847910389ecf9","url":"img/favicons/favicon.ico"},{"revision":"6441cbb8861a71fcfd9cd468b225a998","url":"img/favicons/minerva.ico"},{"revision":"d72ef880b711475e5291c0fdfb1964b3","url":"img/logos/athena.png"},{"revision":"5a842a80da3f7052d79847910389ecf9","url":"img/logos/logo.png"},{"revision":"6441cbb8861a71fcfd9cd468b225a998","url":"img/logos/minerva.png"},{"revision":"7ea4a962be30750a0653ef693b519370","url":"img/pre-visualization/home.png"},{"revision":"38d2d7a80eba1f9ba7bbcafed917c96c","url":"fonts/AbrilText-Bold.woff"},{"revision":"9fc10a629a1d43baa01c1ec8188ea3b7","url":"fonts/AbrilText-Bold.woff2"}]; const controller = new workbox_precaching__WEBPACK_IMPORTED_MODULE_0__.PrecacheController({ // Safer to turn this true? fallbackToNetwork: true,