mirror of
https://github.com/coollabsio/coolify-examples.git
synced 2026-02-18 13:28:57 +00:00
new examples
This commit is contained in:
242
CLAUDE.md
Normal file
242
CLAUDE.md
Normal file
@@ -0,0 +1,242 @@
|
|||||||
|
# Claude Guidelines for Coolify Examples
|
||||||
|
|
||||||
|
## Project Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
coolify-examples/
|
||||||
|
├── node/ # Node.js runtime examples
|
||||||
|
│ ├── <backend-framework>/ # Backend frameworks (single directory)
|
||||||
|
│ └── <frontend-framework>/
|
||||||
|
│ ├── ssr/ # Server-side rendering variant
|
||||||
|
│ └── static/ # Static export variant
|
||||||
|
└── bun/ # (future) Bun runtime examples
|
||||||
|
```
|
||||||
|
|
||||||
|
## Framework Classification
|
||||||
|
|
||||||
|
### Backend Frameworks
|
||||||
|
- Single directory, no ssr/static variants
|
||||||
|
- Examples: expressjs, fastify, nestjs, adonisjs, simple-webserver
|
||||||
|
|
||||||
|
### Frontend Frameworks with SSR + Static
|
||||||
|
- Create `ssr/` and `static/` subdirectories
|
||||||
|
- Examples: nextjs, nuxtjs, remix, astro, sveltekit, tanstack-start
|
||||||
|
|
||||||
|
### Static-Only Frameworks
|
||||||
|
- Single directory (no `/static` subdirectory needed)
|
||||||
|
- Examples: react, vite, eleventy, gatsby
|
||||||
|
|
||||||
|
## Determining SSR vs Static Support
|
||||||
|
|
||||||
|
When adding a new framework, check:
|
||||||
|
|
||||||
|
1. **SSR support**: Does it have a Node.js adapter/server mode?
|
||||||
|
2. **Static support**: Can it pre-render/export to static HTML files?
|
||||||
|
|
||||||
|
| Framework | SSR | Static | Notes |
|
||||||
|
|-----------|-----|--------|-------|
|
||||||
|
| Next.js | Yes | Yes | `output: 'export'` for static |
|
||||||
|
| Nuxt.js | Yes | Yes | `ssr: false` + `nuxt generate` for static |
|
||||||
|
| Remix/React Router | Yes | Yes | `ssr: false` for SPA mode |
|
||||||
|
| Astro | Yes | Yes | Needs `@astrojs/node` adapter for SSR |
|
||||||
|
| SvelteKit | Yes | Yes | `adapter-node` for SSR, `adapter-static` for static |
|
||||||
|
| TanStack Start | Yes | Yes | `server.preset: 'static'` for static |
|
||||||
|
| React | No | Yes | Library only, use with Vite |
|
||||||
|
| Vite | No | Yes | Build tool only, no built-in SSR |
|
||||||
|
| Eleventy | No | Yes | Static site generator only |
|
||||||
|
| Gatsby | No | Yes | Static site generator only |
|
||||||
|
| NestJS | N/A | N/A | Backend only, no rendering |
|
||||||
|
| Express | N/A | N/A | Backend only, no rendering |
|
||||||
|
| Fastify | N/A | N/A | Backend only, no rendering |
|
||||||
|
| AdonisJS | N/A | N/A | Backend only, no rendering |
|
||||||
|
|
||||||
|
## Standard Package.json Scripts
|
||||||
|
|
||||||
|
### Backend frameworks
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"scripts": {
|
||||||
|
"start": "node index.js"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### SSR variants
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"scripts": {
|
||||||
|
"dev": "<framework-dev-cmd>",
|
||||||
|
"build": "<framework-build-cmd>",
|
||||||
|
"start": "<framework-start-cmd>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Static variants
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"scripts": {
|
||||||
|
"dev": "<framework-dev-cmd>",
|
||||||
|
"build": "<framework-build-cmd>",
|
||||||
|
"start": "npx serve@latest <output-dir>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Standard Endpoints for Backend Examples
|
||||||
|
|
||||||
|
All backend examples should have:
|
||||||
|
- `GET /` - Returns `{"message": "Hello from <Framework>!"}`
|
||||||
|
- `GET /health` - Returns `{"status": "ok"}`
|
||||||
|
|
||||||
|
## README Template
|
||||||
|
|
||||||
|
Each framework should have a README.md with:
|
||||||
|
1. Framework name as title
|
||||||
|
2. Brief description
|
||||||
|
3. Getting started commands
|
||||||
|
4. Note about SSR/static if applicable
|
||||||
|
|
||||||
|
## Environment Variable Testing
|
||||||
|
|
||||||
|
Each framework example should include environment variables to verify integration works correctly.
|
||||||
|
|
||||||
|
### Environment Variable Types
|
||||||
|
|
||||||
|
| Type | Base Name | When Set | Where Accessible |
|
||||||
|
|------|-----------|----------|------------------|
|
||||||
|
| Build-time Public | `<PREFIX>_BUILD_PUBLIC_VAR` | During `npm run build` | Client + Server |
|
||||||
|
| Runtime Private | `RUNTIME_PRIVATE_VAR` | At server startup | Server only |
|
||||||
|
| Runtime Public | `RUNTIME_PUBLIC_VAR` | At server startup | Client + Server |
|
||||||
|
|
||||||
|
**Build-time** = baked into bundle, cannot change without rebuild
|
||||||
|
**Runtime** = read when server starts, can change by restarting
|
||||||
|
|
||||||
|
Note: Build-time private vars are omitted because secrets should use runtime vars (to avoid rebuilding per environment).
|
||||||
|
|
||||||
|
### SSR Frameworks (3 env var types)
|
||||||
|
|
||||||
|
SSR frameworks implement all 3 env var types. Runtime vars are served via an `/api/env` endpoint that the client fetches.
|
||||||
|
|
||||||
|
| Framework | Build Public Var | Runtime Private | Runtime Public | Access Pattern |
|
||||||
|
|-----------|------------------|-----------------|----------------|----------------|
|
||||||
|
| Next.js | `NEXT_PUBLIC_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
| Nuxt.js | `NUXT_PUBLIC_BUILD_PUBLIC_VAR` | `NUXT_RUNTIME_PRIVATE_VAR` | `NUXT_PUBLIC_RUNTIME_PUBLIC_VAR` | `runtimeConfig` |
|
||||||
|
| Remix | `VITE_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
| Astro | `PUBLIC_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `process.env` in frontmatter |
|
||||||
|
| SvelteKit | `PUBLIC_BUILD_VAR` | `RUNTIME_PRIVATE_VAR` | `PUBLIC_RUNTIME_PUBLIC_VAR` | `$env/dynamic/*` |
|
||||||
|
| TanStack Start | `VITE_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
| Angular | `NG_APP_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
| Solid-Start | `VITE_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
| Vue | `VITE_BUILD_PUBLIC_VAR` | `RUNTIME_PRIVATE_VAR` | `RUNTIME_PUBLIC_VAR` | `/api/env` route |
|
||||||
|
|
||||||
|
### Static Frameworks (build-time only)
|
||||||
|
|
||||||
|
Static frameworks only support build-time env vars since there's no server at runtime:
|
||||||
|
|
||||||
|
| Framework | Build Public Var | Access Pattern |
|
||||||
|
|-----------|------------------|----------------|
|
||||||
|
| Next.js | `NEXT_PUBLIC_BUILD_PUBLIC_VAR` | `process.env.NEXT_PUBLIC_*` |
|
||||||
|
| Remix | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
| Astro | `PUBLIC_BUILD_PUBLIC_VAR` | `import.meta.env.PUBLIC_*` |
|
||||||
|
| SvelteKit | `PUBLIC_BUILD_PUBLIC_VAR` | `$env/static/public` |
|
||||||
|
| TanStack Start | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
| React | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
| Vite | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
| Eleventy | `BUILD_PUBLIC_VAR` | Via `addGlobalData()` in config |
|
||||||
|
| Gatsby | `GATSBY_BUILD_PUBLIC_VAR` | `process.env.GATSBY_*` |
|
||||||
|
| Angular | `NG_APP_BUILD_PUBLIC_VAR` | `process.env['NG_APP_*']` |
|
||||||
|
| Solid-Start | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
| Vue | `VITE_BUILD_PUBLIC_VAR` | `import.meta.env.VITE_*` |
|
||||||
|
|
||||||
|
### Backend Frameworks (runtime only)
|
||||||
|
|
||||||
|
Backend frameworks use runtime env vars (read at startup). No build-time vars needed.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// Runtime env vars (read at server startup)
|
||||||
|
const RUNTIME_PRIVATE_VAR = process.env.RUNTIME_PRIVATE_VAR || 'default-value';
|
||||||
|
const RUNTIME_PUBLIC_VAR = process.env.RUNTIME_PUBLIC_VAR || 'default-value';
|
||||||
|
|
||||||
|
console.log('=== Runtime Variables ===');
|
||||||
|
console.log('RUNTIME_PRIVATE_VAR:', RUNTIME_PRIVATE_VAR);
|
||||||
|
console.log('RUNTIME_PUBLIC_VAR:', RUNTIME_PUBLIC_VAR);
|
||||||
|
|
||||||
|
// In route handler:
|
||||||
|
return {
|
||||||
|
message: 'Hello from <Framework>!',
|
||||||
|
runtimePrivateVar: RUNTIME_PRIVATE_VAR,
|
||||||
|
runtimePublicVar: RUNTIME_PUBLIC_VAR,
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### API Endpoint for SSR Runtime Vars
|
||||||
|
|
||||||
|
SSR frameworks need an API endpoint to serve runtime vars to the client:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /api/env endpoint
|
||||||
|
app.get('/api/env', (req, res) => {
|
||||||
|
res.json({
|
||||||
|
runtimePrivateVar: RUNTIME_PRIVATE_VAR,
|
||||||
|
runtimePublicVar: RUNTIME_PUBLIC_VAR,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### UI Display Pattern
|
||||||
|
|
||||||
|
Show env vars in a styled box on the home page:
|
||||||
|
|
||||||
|
**For SSR frameworks:**
|
||||||
|
```html
|
||||||
|
<div style="padding: 20px; background: #f0f0f0; margin: 20px; border-radius: 8px;">
|
||||||
|
<h2>Environment Variable Test</h2>
|
||||||
|
|
||||||
|
<h3>Build-time (baked into bundle)</h3>
|
||||||
|
<p><strong><PREFIX>_BUILD_PUBLIC_VAR:</strong> {value}</p>
|
||||||
|
|
||||||
|
<h3>Runtime (read at server startup)</h3>
|
||||||
|
<p><strong>RUNTIME_PRIVATE_VAR:</strong> {value}</p>
|
||||||
|
<p><strong>RUNTIME_PUBLIC_VAR:</strong> {value}</p>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
**For static frameworks:**
|
||||||
|
```html
|
||||||
|
<div style="padding: 20px; background: #f0f0f0; margin: 20px; border-radius: 8px;">
|
||||||
|
<h2>Environment Variable Test</h2>
|
||||||
|
|
||||||
|
<h3>Build-time (baked into bundle)</h3>
|
||||||
|
<p><strong><PREFIX>_BUILD_PUBLIC_VAR:</strong> {value}</p>
|
||||||
|
|
||||||
|
<p style="color: #666; font-size: 14px;">
|
||||||
|
Note: Static sites only support build-time env vars (no server at runtime)
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Testing Commands
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# SSR: Build with build-time vars
|
||||||
|
NEXT_PUBLIC_BUILD_PUBLIC_VAR=build-value npm run build
|
||||||
|
|
||||||
|
# SSR: Start with runtime vars
|
||||||
|
RUNTIME_PRIVATE_VAR=secret RUNTIME_PUBLIC_VAR=public npm run start
|
||||||
|
|
||||||
|
# Static: Build with build-time vars only
|
||||||
|
VITE_BUILD_PUBLIC_VAR=build-value npm run build
|
||||||
|
```
|
||||||
|
|
||||||
|
## Adding a New Framework
|
||||||
|
|
||||||
|
1. Ask/determine if it supports SSR, static, or both
|
||||||
|
2. Create appropriate directory structure
|
||||||
|
3. Initialize project (CLI or manual)
|
||||||
|
4. Configure for SSR or static as needed
|
||||||
|
5. Add standard scripts to package.json
|
||||||
|
6. Add environment variable test (see above)
|
||||||
|
7. Add .gitignore
|
||||||
|
8. Add README.md
|
||||||
|
9. Update parent README.md tables
|
||||||
201
LICENSE
201
LICENSE
@@ -1,201 +0,0 @@
|
|||||||
Apache License
|
|
||||||
Version 2.0, January 2004
|
|
||||||
http://www.apache.org/licenses/
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
||||||
|
|
||||||
1. Definitions.
|
|
||||||
|
|
||||||
"License" shall mean the terms and conditions for use, reproduction,
|
|
||||||
and distribution as defined by Sections 1 through 9 of this document.
|
|
||||||
|
|
||||||
"Licensor" shall mean the copyright owner or entity authorized by
|
|
||||||
the copyright owner that is granting the License.
|
|
||||||
|
|
||||||
"Legal Entity" shall mean the union of the acting entity and all
|
|
||||||
other entities that control, are controlled by, or are under common
|
|
||||||
control with that entity. For the purposes of this definition,
|
|
||||||
"control" means (i) the power, direct or indirect, to cause the
|
|
||||||
direction or management of such entity, whether by contract or
|
|
||||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
||||||
|
|
||||||
"You" (or "Your") shall mean an individual or Legal Entity
|
|
||||||
exercising permissions granted by this License.
|
|
||||||
|
|
||||||
"Source" form shall mean the preferred form for making modifications,
|
|
||||||
including but not limited to software source code, documentation
|
|
||||||
source, and configuration files.
|
|
||||||
|
|
||||||
"Object" form shall mean any form resulting from mechanical
|
|
||||||
transformation or translation of a Source form, including but
|
|
||||||
not limited to compiled object code, generated documentation,
|
|
||||||
and conversions to other media types.
|
|
||||||
|
|
||||||
"Work" shall mean the work of authorship, whether in Source or
|
|
||||||
Object form, made available under the License, as indicated by a
|
|
||||||
copyright notice that is included in or attached to the work
|
|
||||||
(an example is provided in the Appendix below).
|
|
||||||
|
|
||||||
"Derivative Works" shall mean any work, whether in Source or Object
|
|
||||||
form, that is based on (or derived from) the Work and for which the
|
|
||||||
editorial revisions, annotations, elaborations, or other modifications
|
|
||||||
represent, as a whole, an original work of authorship. For the purposes
|
|
||||||
of this License, Derivative Works shall not include works that remain
|
|
||||||
separable from, or merely link (or bind by name) to the interfaces of,
|
|
||||||
the Work and Derivative Works thereof.
|
|
||||||
|
|
||||||
"Contribution" shall mean any work of authorship, including
|
|
||||||
the original version of the Work and any modifications or additions
|
|
||||||
to that Work or Derivative Works thereof, that is intentionally
|
|
||||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
|
||||||
or by an individual or Legal Entity authorized to submit on behalf of
|
|
||||||
the copyright owner. For the purposes of this definition, "submitted"
|
|
||||||
means any form of electronic, verbal, or written communication sent
|
|
||||||
to the Licensor or its representatives, including but not limited to
|
|
||||||
communication on electronic mailing lists, source code control systems,
|
|
||||||
and issue tracking systems that are managed by, or on behalf of, the
|
|
||||||
Licensor for the purpose of discussing and improving the Work, but
|
|
||||||
excluding communication that is conspicuously marked or otherwise
|
|
||||||
designated in writing by the copyright owner as "Not a Contribution."
|
|
||||||
|
|
||||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
|
||||||
on behalf of whom a Contribution has been received by Licensor and
|
|
||||||
subsequently incorporated within the Work.
|
|
||||||
|
|
||||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
copyright license to reproduce, prepare Derivative Works of,
|
|
||||||
publicly display, publicly perform, sublicense, and distribute the
|
|
||||||
Work and such Derivative Works in Source or Object form.
|
|
||||||
|
|
||||||
3. Grant of Patent License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
(except as stated in this section) patent license to make, have made,
|
|
||||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
|
||||||
where such license applies only to those patent claims licensable
|
|
||||||
by such Contributor that are necessarily infringed by their
|
|
||||||
Contribution(s) alone or by combination of their Contribution(s)
|
|
||||||
with the Work to which such Contribution(s) was submitted. If You
|
|
||||||
institute patent litigation against any entity (including a
|
|
||||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
|
||||||
or a Contribution incorporated within the Work constitutes direct
|
|
||||||
or contributory patent infringement, then any patent licenses
|
|
||||||
granted to You under this License for that Work shall terminate
|
|
||||||
as of the date such litigation is filed.
|
|
||||||
|
|
||||||
4. Redistribution. You may reproduce and distribute copies of the
|
|
||||||
Work or Derivative Works thereof in any medium, with or without
|
|
||||||
modifications, and in Source or Object form, provided that You
|
|
||||||
meet the following conditions:
|
|
||||||
|
|
||||||
(a) You must give any other recipients of the Work or
|
|
||||||
Derivative Works a copy of this License; and
|
|
||||||
|
|
||||||
(b) You must cause any modified files to carry prominent notices
|
|
||||||
stating that You changed the files; and
|
|
||||||
|
|
||||||
(c) You must retain, in the Source form of any Derivative Works
|
|
||||||
that You distribute, all copyright, patent, trademark, and
|
|
||||||
attribution notices from the Source form of the Work,
|
|
||||||
excluding those notices that do not pertain to any part of
|
|
||||||
the Derivative Works; and
|
|
||||||
|
|
||||||
(d) If the Work includes a "NOTICE" text file as part of its
|
|
||||||
distribution, then any Derivative Works that You distribute must
|
|
||||||
include a readable copy of the attribution notices contained
|
|
||||||
within such NOTICE file, excluding those notices that do not
|
|
||||||
pertain to any part of the Derivative Works, in at least one
|
|
||||||
of the following places: within a NOTICE text file distributed
|
|
||||||
as part of the Derivative Works; within the Source form or
|
|
||||||
documentation, if provided along with the Derivative Works; or,
|
|
||||||
within a display generated by the Derivative Works, if and
|
|
||||||
wherever such third-party notices normally appear. The contents
|
|
||||||
of the NOTICE file are for informational purposes only and
|
|
||||||
do not modify the License. You may add Your own attribution
|
|
||||||
notices within Derivative Works that You distribute, alongside
|
|
||||||
or as an addendum to the NOTICE text from the Work, provided
|
|
||||||
that such additional attribution notices cannot be construed
|
|
||||||
as modifying the License.
|
|
||||||
|
|
||||||
You may add Your own copyright statement to Your modifications and
|
|
||||||
may provide additional or different license terms and conditions
|
|
||||||
for use, reproduction, or distribution of Your modifications, or
|
|
||||||
for any such Derivative Works as a whole, provided Your use,
|
|
||||||
reproduction, and distribution of the Work otherwise complies with
|
|
||||||
the conditions stated in this License.
|
|
||||||
|
|
||||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
|
||||||
any Contribution intentionally submitted for inclusion in the Work
|
|
||||||
by You to the Licensor shall be under the terms and conditions of
|
|
||||||
this License, without any additional terms or conditions.
|
|
||||||
Notwithstanding the above, nothing herein shall supersede or modify
|
|
||||||
the terms of any separate license agreement you may have executed
|
|
||||||
with Licensor regarding such Contributions.
|
|
||||||
|
|
||||||
6. Trademarks. This License does not grant permission to use the trade
|
|
||||||
names, trademarks, service marks, or product names of the Licensor,
|
|
||||||
except as required for reasonable and customary use in describing the
|
|
||||||
origin of the Work and reproducing the content of the NOTICE file.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
|
||||||
agreed to in writing, Licensor provides the Work (and each
|
|
||||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
||||||
implied, including, without limitation, any warranties or conditions
|
|
||||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
|
||||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
|
||||||
appropriateness of using or redistributing the Work and assume any
|
|
||||||
risks associated with Your exercise of permissions under this License.
|
|
||||||
|
|
||||||
8. Limitation of Liability. In no event and under no legal theory,
|
|
||||||
whether in tort (including negligence), contract, or otherwise,
|
|
||||||
unless required by applicable law (such as deliberate and grossly
|
|
||||||
negligent acts) or agreed to in writing, shall any Contributor be
|
|
||||||
liable to You for damages, including any direct, indirect, special,
|
|
||||||
incidental, or consequential damages of any character arising as a
|
|
||||||
result of this License or out of the use or inability to use the
|
|
||||||
Work (including but not limited to damages for loss of goodwill,
|
|
||||||
work stoppage, computer failure or malfunction, or any and all
|
|
||||||
other commercial damages or losses), even if such Contributor
|
|
||||||
has been advised of the possibility of such damages.
|
|
||||||
|
|
||||||
9. Accepting Warranty or Additional Liability. While redistributing
|
|
||||||
the Work or Derivative Works thereof, You may choose to offer,
|
|
||||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
|
||||||
or other liability obligations and/or rights consistent with this
|
|
||||||
License. However, in accepting such obligations, You may act only
|
|
||||||
on Your own behalf and on Your sole responsibility, not on behalf
|
|
||||||
of any other Contributor, and only if You agree to indemnify,
|
|
||||||
defend, and hold each Contributor harmless for any liability
|
|
||||||
incurred by, or claims asserted against, such Contributor by reason
|
|
||||||
of your accepting any such warranty or additional liability.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work.
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following
|
|
||||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
|
||||||
replaced with your own identifying information. (Don't include
|
|
||||||
the brackets!) The text should be enclosed in the appropriate
|
|
||||||
comment syntax for the file format. We also recommend that a
|
|
||||||
file or class name and description of purpose be included on the
|
|
||||||
same "printed page" as the copyright notice for easier
|
|
||||||
identification within third-party archives.
|
|
||||||
|
|
||||||
Copyright [2024] [Andras Bacsai]
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
73
README.md
73
README.md
@@ -1,26 +1,53 @@
|
|||||||
# Coolify Examples
|
# Coolify Examples
|
||||||
This repository contains examples of how to deploy applications using Coolify.
|
|
||||||
|
|
||||||
- [x] Static
|
Example applications for deploying to [Coolify](https://coolify.io).
|
||||||
- [x] Node.js
|
|
||||||
- [x] Dockerfile
|
|
||||||
- [x] Docker Compose
|
|
||||||
- [x] Strapi
|
|
||||||
- [x] Nuxt
|
|
||||||
- [x] Vite
|
|
||||||
- [x] Bun
|
|
||||||
- [x] Astro
|
|
||||||
- [x] Rust
|
|
||||||
- [x] Laravel
|
|
||||||
- [x] Nuxt
|
|
||||||
- [x] Vue
|
|
||||||
- [ ] SvelteKit-node
|
|
||||||
- [ ] SvelteKit-static
|
|
||||||
- [x] Next.js
|
|
||||||
- [x] Turbo Next.js
|
|
||||||
- [ ] Rails
|
|
||||||
- [ ] Django
|
|
||||||
- [ ] Turborepo
|
|
||||||
- [ ] PHP
|
|
||||||
- [ ] React/Preact/Vue/Vite/Svelte
|
|
||||||
|
|
||||||
|
## Runtimes
|
||||||
|
|
||||||
|
| Runtime | Description |
|
||||||
|
|---------|-------------|
|
||||||
|
| [node](./node) | Node.js applications |
|
||||||
|
|
||||||
|
## Node.js Examples
|
||||||
|
|
||||||
|
### Backend Frameworks
|
||||||
|
|
||||||
|
| Framework | Description |
|
||||||
|
|-----------|-------------|
|
||||||
|
| [simple-webserver](./node/simple-webserver) | Pure Node.js HTTP server |
|
||||||
|
| [expressjs](./node/expressjs) | Express.js web framework |
|
||||||
|
| [fastify](./node/fastify) | Fastify web framework |
|
||||||
|
| [nestjs](./node/nestjs) | NestJS framework |
|
||||||
|
| [adonisjs](./node/adonisjs) | AdonisJS framework |
|
||||||
|
|
||||||
|
### Frontend Frameworks
|
||||||
|
|
||||||
|
| Framework | SSR | Static |
|
||||||
|
|-----------|-----|--------|
|
||||||
|
| [nextjs](./node/nextjs) | [ssr](./node/nextjs/ssr) | [static](./node/nextjs/static) |
|
||||||
|
| [nuxtjs](./node/nuxtjs) | [ssr](./node/nuxtjs/ssr) | [static](./node/nuxtjs/static) |
|
||||||
|
| [remix](./node/remix) | [ssr](./node/remix/ssr) | [static](./node/remix/static) |
|
||||||
|
| [astro](./node/astro) | [ssr](./node/astro/ssr) | [static](./node/astro/static) |
|
||||||
|
| [sveltekit](./node/sveltekit) | [ssr](./node/sveltekit/ssr) | [static](./node/sveltekit/static) |
|
||||||
|
| [tanstack-start](./node/tanstack-start) | [ssr](./node/tanstack-start/ssr) | [static](./node/tanstack-start/static) |
|
||||||
|
| [angular](./node/angular) | [ssr](./node/angular/ssr) | [static](./node/angular/static) |
|
||||||
|
| [vite](./node/vite) | - | [static](./node/vite) |
|
||||||
|
| [eleventy](./node/eleventy) | - | [static](./node/eleventy) |
|
||||||
|
| [gatsby](./node/gatsby) | - | [static](./node/gatsby) |
|
||||||
|
|
||||||
|
## SSR vs Static
|
||||||
|
|
||||||
|
- **SSR (Server-Side Rendering)**: Pages rendered on each request. Requires a running server.
|
||||||
|
- **Static**: Pages pre-rendered at build time. Output is plain HTML/CSS/JS files served via a static file server.
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
Each example includes a README with instructions. Generally:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd <example-directory>
|
||||||
|
npm install
|
||||||
|
npm run dev # development
|
||||||
|
npm run build # production build
|
||||||
|
npm start # production server
|
||||||
|
```
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
# Coolify Configuration
|
|
||||||
|
|
||||||
Check [astro.config.mjs](./astro.config.mjs).
|
|
||||||
|
|
||||||
1. Use `Nixpacks`.
|
|
||||||
2. Set `Ports Exposed` to `4321` (or any port you set in your `astro.config.mjs` file)
|
|
||||||
3. Set `Start Command` to `HOST=0.0.0.0 node dist/server/entry.mjs`
|
|
||||||
- Alternatively, you can set the `start` script inside `package.json` to `HOST=0.0.0.0 node dist/server/entry.mjs`. Then Nixpacks will automatically use it as the start command.
|
|
||||||
6636
astro/server/package-lock.json
generated
6636
astro/server/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,18 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "",
|
|
||||||
"type": "module",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "astro dev",
|
|
||||||
"start": "astro dev",
|
|
||||||
"build": "astro check && astro build",
|
|
||||||
"preview": "astro preview",
|
|
||||||
"astro": "astro"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@astrojs/check": "^0.5.9",
|
|
||||||
"@astrojs/node": "^8.2.3",
|
|
||||||
"astro": "^4.5.3",
|
|
||||||
"typescript": "^5.4.2"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
---
|
|
||||||
interface Props {
|
|
||||||
title: string;
|
|
||||||
body: string;
|
|
||||||
href: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { href, title, body } = Astro.props;
|
|
||||||
---
|
|
||||||
|
|
||||||
<li class="link-card">
|
|
||||||
<a href={href}>
|
|
||||||
<h2>
|
|
||||||
{title}
|
|
||||||
<span>→</span>
|
|
||||||
</h2>
|
|
||||||
<p>
|
|
||||||
{body}
|
|
||||||
</p>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
<style>
|
|
||||||
.link-card {
|
|
||||||
list-style: none;
|
|
||||||
display: flex;
|
|
||||||
padding: 1px;
|
|
||||||
background-color: #23262d;
|
|
||||||
background-image: none;
|
|
||||||
background-size: 400%;
|
|
||||||
border-radius: 7px;
|
|
||||||
background-position: 100%;
|
|
||||||
transition: background-position 0.6s cubic-bezier(0.22, 1, 0.36, 1);
|
|
||||||
box-shadow: inset 0 0 0 1px rgba(255, 255, 255, 0.1);
|
|
||||||
}
|
|
||||||
.link-card > a {
|
|
||||||
width: 100%;
|
|
||||||
text-decoration: none;
|
|
||||||
line-height: 1.4;
|
|
||||||
padding: calc(1.5rem - 1px);
|
|
||||||
border-radius: 8px;
|
|
||||||
color: white;
|
|
||||||
background-color: #23262d;
|
|
||||||
opacity: 0.8;
|
|
||||||
}
|
|
||||||
h2 {
|
|
||||||
margin: 0;
|
|
||||||
font-size: 1.25rem;
|
|
||||||
transition: color 0.6s cubic-bezier(0.22, 1, 0.36, 1);
|
|
||||||
}
|
|
||||||
p {
|
|
||||||
margin-top: 0.5rem;
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
.link-card:is(:hover, :focus-within) {
|
|
||||||
background-position: 0;
|
|
||||||
background-image: var(--accent-gradient);
|
|
||||||
}
|
|
||||||
.link-card:is(:hover, :focus-within) h2 {
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
1
astro/server/src/env.d.ts
vendored
1
astro/server/src/env.d.ts
vendored
@@ -1 +0,0 @@
|
|||||||
/// <reference types="astro/client" />
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
---
|
|
||||||
interface Props {
|
|
||||||
title: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { title } = Astro.props;
|
|
||||||
---
|
|
||||||
|
|
||||||
<!doctype html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8" />
|
|
||||||
<meta name="description" content="Astro description" />
|
|
||||||
<meta name="viewport" content="width=device-width" />
|
|
||||||
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
|
||||||
<meta name="generator" content={Astro.generator} />
|
|
||||||
<title>{title}</title>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<slot />
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
<style is:global>
|
|
||||||
:root {
|
|
||||||
--accent: 136, 58, 234;
|
|
||||||
--accent-light: 224, 204, 250;
|
|
||||||
--accent-dark: 49, 10, 101;
|
|
||||||
--accent-gradient: linear-gradient(
|
|
||||||
45deg,
|
|
||||||
rgb(var(--accent)),
|
|
||||||
rgb(var(--accent-light)) 30%,
|
|
||||||
white 60%
|
|
||||||
);
|
|
||||||
}
|
|
||||||
html {
|
|
||||||
font-family: system-ui, sans-serif;
|
|
||||||
background: #13151a;
|
|
||||||
background-size: 224px;
|
|
||||||
}
|
|
||||||
code {
|
|
||||||
font-family:
|
|
||||||
Menlo,
|
|
||||||
Monaco,
|
|
||||||
Lucida Console,
|
|
||||||
Liberation Mono,
|
|
||||||
DejaVu Sans Mono,
|
|
||||||
Bitstream Vera Sans Mono,
|
|
||||||
Courier New,
|
|
||||||
monospace;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,123 +0,0 @@
|
|||||||
---
|
|
||||||
import Layout from '../layouts/Layout.astro';
|
|
||||||
import Card from '../components/Card.astro';
|
|
||||||
---
|
|
||||||
|
|
||||||
<Layout title="Welcome to Astro.">
|
|
||||||
<main>
|
|
||||||
<svg
|
|
||||||
class="astro-a"
|
|
||||||
width="495"
|
|
||||||
height="623"
|
|
||||||
viewBox="0 0 495 623"
|
|
||||||
fill="none"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
aria-hidden="true"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M167.19 364.254C83.4786 364.254 0 404.819 0 404.819C0 404.819 141.781 19.4876 142.087 18.7291C146.434 7.33701 153.027 0 162.289 0H332.441C341.703 0 348.574 7.33701 352.643 18.7291C352.92 19.5022 494.716 404.819 494.716 404.819C494.716 404.819 426.67 364.254 327.525 364.254L264.41 169.408C262.047 159.985 255.147 153.581 247.358 153.581C239.569 153.581 232.669 159.985 230.306 169.408L167.19 364.254ZM160.869 530.172C160.877 530.18 160.885 530.187 160.894 530.195L160.867 530.181C160.868 530.178 160.868 530.175 160.869 530.172ZM136.218 411.348C124.476 450.467 132.698 504.458 160.869 530.172C160.997 529.696 161.125 529.242 161.248 528.804C161.502 527.907 161.737 527.073 161.917 526.233C165.446 509.895 178.754 499.52 195.577 500.01C211.969 500.487 220.67 508.765 223.202 527.254C224.141 534.12 224.23 541.131 224.319 548.105C224.328 548.834 224.337 549.563 224.347 550.291C224.563 566.098 228.657 580.707 237.264 593.914C245.413 606.426 256.108 615.943 270.749 622.478C270.593 621.952 270.463 621.508 270.35 621.126C270.045 620.086 269.872 619.499 269.685 618.911C258.909 585.935 266.668 563.266 295.344 543.933C298.254 541.971 301.187 540.041 304.12 538.112C310.591 533.854 317.059 529.599 323.279 525.007C345.88 508.329 360.09 486.327 363.431 457.844C364.805 446.148 363.781 434.657 359.848 423.275C358.176 424.287 356.587 425.295 355.042 426.275C351.744 428.366 348.647 430.33 345.382 431.934C303.466 452.507 259.152 455.053 214.03 448.245C184.802 443.834 156.584 436.019 136.218 411.348Z"
|
|
||||||
fill="url(#paint0_linear_1805_24383)"></path>
|
|
||||||
<defs>
|
|
||||||
<linearGradient
|
|
||||||
id="paint0_linear_1805_24383"
|
|
||||||
x1="247.358"
|
|
||||||
y1="0"
|
|
||||||
x2="247.358"
|
|
||||||
y2="622.479"
|
|
||||||
gradientUnits="userSpaceOnUse"
|
|
||||||
>
|
|
||||||
<stop stop-opacity="0.9"></stop>
|
|
||||||
<stop offset="1" stop-opacity="0.2"></stop>
|
|
||||||
</linearGradient>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
||||||
<h1>Welcome to <span class="text-gradient">Astro</span></h1>
|
|
||||||
<p class="instructions">
|
|
||||||
To get started, open the directory <code>src/pages</code> in your project.<br />
|
|
||||||
<strong>Code Challenge:</strong> Tweak the "Welcome to Astro" message above.
|
|
||||||
</p>
|
|
||||||
<ul role="list" class="link-card-grid">
|
|
||||||
<Card
|
|
||||||
href="https://docs.astro.build/"
|
|
||||||
title="Documentation"
|
|
||||||
body="Learn how Astro works and explore the official API docs."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/integrations/"
|
|
||||||
title="Integrations"
|
|
||||||
body="Supercharge your project with new frameworks and libraries."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/themes/"
|
|
||||||
title="Themes"
|
|
||||||
body="Explore a galaxy of community-built starter themes."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/chat/"
|
|
||||||
title="Community"
|
|
||||||
body="Come say hi to our amazing Discord community. ❤️"
|
|
||||||
/>
|
|
||||||
</ul>
|
|
||||||
</main>
|
|
||||||
</Layout>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
main {
|
|
||||||
margin: auto;
|
|
||||||
padding: 1rem;
|
|
||||||
width: 800px;
|
|
||||||
max-width: calc(100% - 2rem);
|
|
||||||
color: white;
|
|
||||||
font-size: 20px;
|
|
||||||
line-height: 1.6;
|
|
||||||
}
|
|
||||||
.astro-a {
|
|
||||||
position: absolute;
|
|
||||||
top: -32px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translatex(-50%);
|
|
||||||
width: 220px;
|
|
||||||
height: auto;
|
|
||||||
z-index: -1;
|
|
||||||
}
|
|
||||||
h1 {
|
|
||||||
font-size: 4rem;
|
|
||||||
font-weight: 700;
|
|
||||||
line-height: 1;
|
|
||||||
text-align: center;
|
|
||||||
margin-bottom: 1em;
|
|
||||||
}
|
|
||||||
.text-gradient {
|
|
||||||
background-image: var(--accent-gradient);
|
|
||||||
-webkit-background-clip: text;
|
|
||||||
-webkit-text-fill-color: transparent;
|
|
||||||
background-size: 400%;
|
|
||||||
background-position: 0%;
|
|
||||||
}
|
|
||||||
.instructions {
|
|
||||||
margin-bottom: 2rem;
|
|
||||||
border: 1px solid rgba(var(--accent-light), 25%);
|
|
||||||
background: linear-gradient(rgba(var(--accent-dark), 66%), rgba(var(--accent-dark), 33%));
|
|
||||||
padding: 1.5rem;
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
.instructions code {
|
|
||||||
font-size: 0.8em;
|
|
||||||
font-weight: bold;
|
|
||||||
background: rgba(var(--accent-light), 12%);
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
border-radius: 4px;
|
|
||||||
padding: 0.3em 0.4em;
|
|
||||||
}
|
|
||||||
.instructions strong {
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
}
|
|
||||||
.link-card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(24ch, 1fr));
|
|
||||||
gap: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
{
|
|
||||||
"extends": "astro/tsconfigs/strict"
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
# Coolify Configuration
|
|
||||||
|
|
||||||
Check [astro.config.mjs](./astro.config.mjs).
|
|
||||||
|
|
||||||
1. Use `Nixpacks`.
|
|
||||||
2. Turn on `Is it a static site?`.
|
|
||||||
3. Set `Publish Directory` to `/dist`.
|
|
||||||
6460
astro/static/package-lock.json
generated
6460
astro/static/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,61 +0,0 @@
|
|||||||
---
|
|
||||||
interface Props {
|
|
||||||
title: string;
|
|
||||||
body: string;
|
|
||||||
href: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { href, title, body } = Astro.props;
|
|
||||||
---
|
|
||||||
|
|
||||||
<li class="link-card">
|
|
||||||
<a href={href}>
|
|
||||||
<h2>
|
|
||||||
{title}
|
|
||||||
<span>→</span>
|
|
||||||
</h2>
|
|
||||||
<p>
|
|
||||||
{body}
|
|
||||||
</p>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
<style>
|
|
||||||
.link-card {
|
|
||||||
list-style: none;
|
|
||||||
display: flex;
|
|
||||||
padding: 1px;
|
|
||||||
background-color: #23262d;
|
|
||||||
background-image: none;
|
|
||||||
background-size: 400%;
|
|
||||||
border-radius: 7px;
|
|
||||||
background-position: 100%;
|
|
||||||
transition: background-position 0.6s cubic-bezier(0.22, 1, 0.36, 1);
|
|
||||||
box-shadow: inset 0 0 0 1px rgba(255, 255, 255, 0.1);
|
|
||||||
}
|
|
||||||
.link-card > a {
|
|
||||||
width: 100%;
|
|
||||||
text-decoration: none;
|
|
||||||
line-height: 1.4;
|
|
||||||
padding: calc(1.5rem - 1px);
|
|
||||||
border-radius: 8px;
|
|
||||||
color: white;
|
|
||||||
background-color: #23262d;
|
|
||||||
opacity: 0.8;
|
|
||||||
}
|
|
||||||
h2 {
|
|
||||||
margin: 0;
|
|
||||||
font-size: 1.25rem;
|
|
||||||
transition: color 0.6s cubic-bezier(0.22, 1, 0.36, 1);
|
|
||||||
}
|
|
||||||
p {
|
|
||||||
margin-top: 0.5rem;
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
.link-card:is(:hover, :focus-within) {
|
|
||||||
background-position: 0;
|
|
||||||
background-image: var(--accent-gradient);
|
|
||||||
}
|
|
||||||
.link-card:is(:hover, :focus-within) h2 {
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
1
astro/static/src/env.d.ts
vendored
1
astro/static/src/env.d.ts
vendored
@@ -1 +0,0 @@
|
|||||||
/// <reference types="astro/client" />
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
---
|
|
||||||
interface Props {
|
|
||||||
title: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { title } = Astro.props;
|
|
||||||
---
|
|
||||||
|
|
||||||
<!doctype html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8" />
|
|
||||||
<meta name="description" content="Astro description" />
|
|
||||||
<meta name="viewport" content="width=device-width" />
|
|
||||||
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
|
||||||
<meta name="generator" content={Astro.generator} />
|
|
||||||
<title>{title}</title>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<slot />
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
<style is:global>
|
|
||||||
:root {
|
|
||||||
--accent: 136, 58, 234;
|
|
||||||
--accent-light: 224, 204, 250;
|
|
||||||
--accent-dark: 49, 10, 101;
|
|
||||||
--accent-gradient: linear-gradient(
|
|
||||||
45deg,
|
|
||||||
rgb(var(--accent)),
|
|
||||||
rgb(var(--accent-light)) 30%,
|
|
||||||
white 60%
|
|
||||||
);
|
|
||||||
}
|
|
||||||
html {
|
|
||||||
font-family: system-ui, sans-serif;
|
|
||||||
background: #13151a;
|
|
||||||
background-size: 224px;
|
|
||||||
}
|
|
||||||
code {
|
|
||||||
font-family:
|
|
||||||
Menlo,
|
|
||||||
Monaco,
|
|
||||||
Lucida Console,
|
|
||||||
Liberation Mono,
|
|
||||||
DejaVu Sans Mono,
|
|
||||||
Bitstream Vera Sans Mono,
|
|
||||||
Courier New,
|
|
||||||
monospace;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
Custom 404 here hello
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
error 500
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
502 is a lot
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
---
|
|
||||||
import Layout from '../layouts/Layout.astro';
|
|
||||||
---
|
|
||||||
|
|
||||||
<Layout title="Dashboard">
|
|
||||||
<main>
|
|
||||||
<h1>Welcome to Dashboard</h1>
|
|
||||||
<p class="instructions">
|
|
||||||
To get started, open the directory <code>src/pages</code> in your project.<br />
|
|
||||||
<strong>Code Challenge:</strong> Tweak the "Welcome to Astro" message above.
|
|
||||||
</p>
|
|
||||||
</main>
|
|
||||||
</Layout>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
main {
|
|
||||||
margin: auto;
|
|
||||||
padding: 1rem;
|
|
||||||
width: 800px;
|
|
||||||
max-width: calc(100% - 2rem);
|
|
||||||
color: white;
|
|
||||||
font-size: 20px;
|
|
||||||
line-height: 1.6;
|
|
||||||
}
|
|
||||||
.astro-a {
|
|
||||||
position: absolute;
|
|
||||||
top: -32px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translatex(-50%);
|
|
||||||
width: 220px;
|
|
||||||
height: auto;
|
|
||||||
z-index: -1;
|
|
||||||
}
|
|
||||||
h1 {
|
|
||||||
font-size: 4rem;
|
|
||||||
font-weight: 700;
|
|
||||||
line-height: 1;
|
|
||||||
text-align: center;
|
|
||||||
margin-bottom: 1em;
|
|
||||||
}
|
|
||||||
.text-gradient {
|
|
||||||
background-image: var(--accent-gradient);
|
|
||||||
-webkit-background-clip: text;
|
|
||||||
-webkit-text-fill-color: transparent;
|
|
||||||
background-size: 400%;
|
|
||||||
background-position: 0%;
|
|
||||||
}
|
|
||||||
.instructions {
|
|
||||||
margin-bottom: 2rem;
|
|
||||||
border: 1px solid rgba(var(--accent-light), 25%);
|
|
||||||
background: linear-gradient(rgba(var(--accent-dark), 66%), rgba(var(--accent-dark), 33%));
|
|
||||||
padding: 1.5rem;
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
.instructions code {
|
|
||||||
font-size: 0.8em;
|
|
||||||
font-weight: bold;
|
|
||||||
background: rgba(var(--accent-light), 12%);
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
border-radius: 4px;
|
|
||||||
padding: 0.3em 0.4em;
|
|
||||||
}
|
|
||||||
.instructions strong {
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
}
|
|
||||||
.link-card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(24ch, 1fr));
|
|
||||||
gap: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,123 +0,0 @@
|
|||||||
---
|
|
||||||
import Layout from '../layouts/Layout.astro';
|
|
||||||
import Card from '../components/Card.astro';
|
|
||||||
---
|
|
||||||
|
|
||||||
<Layout title="Welcome to Astro.">
|
|
||||||
<main>
|
|
||||||
<svg
|
|
||||||
class="astro-a"
|
|
||||||
width="495"
|
|
||||||
height="623"
|
|
||||||
viewBox="0 0 495 623"
|
|
||||||
fill="none"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
aria-hidden="true"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M167.19 364.254C83.4786 364.254 0 404.819 0 404.819C0 404.819 141.781 19.4876 142.087 18.7291C146.434 7.33701 153.027 0 162.289 0H332.441C341.703 0 348.574 7.33701 352.643 18.7291C352.92 19.5022 494.716 404.819 494.716 404.819C494.716 404.819 426.67 364.254 327.525 364.254L264.41 169.408C262.047 159.985 255.147 153.581 247.358 153.581C239.569 153.581 232.669 159.985 230.306 169.408L167.19 364.254ZM160.869 530.172C160.877 530.18 160.885 530.187 160.894 530.195L160.867 530.181C160.868 530.178 160.868 530.175 160.869 530.172ZM136.218 411.348C124.476 450.467 132.698 504.458 160.869 530.172C160.997 529.696 161.125 529.242 161.248 528.804C161.502 527.907 161.737 527.073 161.917 526.233C165.446 509.895 178.754 499.52 195.577 500.01C211.969 500.487 220.67 508.765 223.202 527.254C224.141 534.12 224.23 541.131 224.319 548.105C224.328 548.834 224.337 549.563 224.347 550.291C224.563 566.098 228.657 580.707 237.264 593.914C245.413 606.426 256.108 615.943 270.749 622.478C270.593 621.952 270.463 621.508 270.35 621.126C270.045 620.086 269.872 619.499 269.685 618.911C258.909 585.935 266.668 563.266 295.344 543.933C298.254 541.971 301.187 540.041 304.12 538.112C310.591 533.854 317.059 529.599 323.279 525.007C345.88 508.329 360.09 486.327 363.431 457.844C364.805 446.148 363.781 434.657 359.848 423.275C358.176 424.287 356.587 425.295 355.042 426.275C351.744 428.366 348.647 430.33 345.382 431.934C303.466 452.507 259.152 455.053 214.03 448.245C184.802 443.834 156.584 436.019 136.218 411.348Z"
|
|
||||||
fill="url(#paint0_linear_1805_24383)"></path>
|
|
||||||
<defs>
|
|
||||||
<linearGradient
|
|
||||||
id="paint0_linear_1805_24383"
|
|
||||||
x1="247.358"
|
|
||||||
y1="0"
|
|
||||||
x2="247.358"
|
|
||||||
y2="622.479"
|
|
||||||
gradientUnits="userSpaceOnUse"
|
|
||||||
>
|
|
||||||
<stop stop-opacity="0.9"></stop>
|
|
||||||
<stop offset="1" stop-opacity="0.2"></stop>
|
|
||||||
</linearGradient>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
||||||
<h1>Welcome to <span class="text-gradient">Astro</span></h1>
|
|
||||||
<p class="instructions">
|
|
||||||
To get started, open the directory <code>src/pages</code> in your project.<br />
|
|
||||||
<strong>Code Challenge:</strong> Tweak the "Welcome to Astro" message above.
|
|
||||||
</p>
|
|
||||||
<ul role="list" class="link-card-grid">
|
|
||||||
<Card
|
|
||||||
href="https://docs.astro.build/"
|
|
||||||
title="Documentation"
|
|
||||||
body="Learn how Astro works and explore the official API docs."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/integrations/"
|
|
||||||
title="Integrations"
|
|
||||||
body="Supercharge your project with new frameworks and libraries."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/themes/"
|
|
||||||
title="Themes"
|
|
||||||
body="Explore a galaxy of community-built starter themes."
|
|
||||||
/>
|
|
||||||
<Card
|
|
||||||
href="https://astro.build/chat/"
|
|
||||||
title="Community"
|
|
||||||
body="Come say hi to our amazing Discord community. ❤️"
|
|
||||||
/>
|
|
||||||
</ul>
|
|
||||||
</main>
|
|
||||||
</Layout>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
main {
|
|
||||||
margin: auto;
|
|
||||||
padding: 1rem;
|
|
||||||
width: 800px;
|
|
||||||
max-width: calc(100% - 2rem);
|
|
||||||
color: white;
|
|
||||||
font-size: 20px;
|
|
||||||
line-height: 1.6;
|
|
||||||
}
|
|
||||||
.astro-a {
|
|
||||||
position: absolute;
|
|
||||||
top: -32px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translatex(-50%);
|
|
||||||
width: 220px;
|
|
||||||
height: auto;
|
|
||||||
z-index: -1;
|
|
||||||
}
|
|
||||||
h1 {
|
|
||||||
font-size: 4rem;
|
|
||||||
font-weight: 700;
|
|
||||||
line-height: 1;
|
|
||||||
text-align: center;
|
|
||||||
margin-bottom: 1em;
|
|
||||||
}
|
|
||||||
.text-gradient {
|
|
||||||
background-image: var(--accent-gradient);
|
|
||||||
-webkit-background-clip: text;
|
|
||||||
-webkit-text-fill-color: transparent;
|
|
||||||
background-size: 400%;
|
|
||||||
background-position: 0%;
|
|
||||||
}
|
|
||||||
.instructions {
|
|
||||||
margin-bottom: 2rem;
|
|
||||||
border: 1px solid rgba(var(--accent-light), 25%);
|
|
||||||
background: linear-gradient(rgba(var(--accent-dark), 66%), rgba(var(--accent-dark), 33%));
|
|
||||||
padding: 1.5rem;
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
.instructions code {
|
|
||||||
font-size: 0.8em;
|
|
||||||
font-weight: bold;
|
|
||||||
background: rgba(var(--accent-light), 12%);
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
border-radius: 4px;
|
|
||||||
padding: 0.3em 0.4em;
|
|
||||||
}
|
|
||||||
.instructions strong {
|
|
||||||
color: rgb(var(--accent-light));
|
|
||||||
}
|
|
||||||
.link-card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(24ch, 1fr));
|
|
||||||
gap: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
{
|
|
||||||
"extends": "astro/tsconfigs/strict"
|
|
||||||
}
|
|
||||||
175
bun/.gitignore
vendored
175
bun/.gitignore
vendored
@@ -1,175 +0,0 @@
|
|||||||
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
|
|
||||||
|
|
||||||
# Logs
|
|
||||||
|
|
||||||
logs
|
|
||||||
_.log
|
|
||||||
npm-debug.log_
|
|
||||||
yarn-debug.log*
|
|
||||||
yarn-error.log*
|
|
||||||
lerna-debug.log*
|
|
||||||
.pnpm-debug.log*
|
|
||||||
|
|
||||||
# Caches
|
|
||||||
|
|
||||||
.cache
|
|
||||||
|
|
||||||
# Diagnostic reports (https://nodejs.org/api/report.html)
|
|
||||||
|
|
||||||
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
|
|
||||||
|
|
||||||
# Runtime data
|
|
||||||
|
|
||||||
pids
|
|
||||||
_.pid
|
|
||||||
_.seed
|
|
||||||
*.pid.lock
|
|
||||||
|
|
||||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
|
||||||
|
|
||||||
lib-cov
|
|
||||||
|
|
||||||
# Coverage directory used by tools like istanbul
|
|
||||||
|
|
||||||
coverage
|
|
||||||
*.lcov
|
|
||||||
|
|
||||||
# nyc test coverage
|
|
||||||
|
|
||||||
.nyc_output
|
|
||||||
|
|
||||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
|
||||||
|
|
||||||
.grunt
|
|
||||||
|
|
||||||
# Bower dependency directory (https://bower.io/)
|
|
||||||
|
|
||||||
bower_components
|
|
||||||
|
|
||||||
# node-waf configuration
|
|
||||||
|
|
||||||
.lock-wscript
|
|
||||||
|
|
||||||
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
|
||||||
|
|
||||||
build/Release
|
|
||||||
|
|
||||||
# Dependency directories
|
|
||||||
|
|
||||||
node_modules/
|
|
||||||
jspm_packages/
|
|
||||||
|
|
||||||
# Snowpack dependency directory (https://snowpack.dev/)
|
|
||||||
|
|
||||||
web_modules/
|
|
||||||
|
|
||||||
# TypeScript cache
|
|
||||||
|
|
||||||
*.tsbuildinfo
|
|
||||||
|
|
||||||
# Optional npm cache directory
|
|
||||||
|
|
||||||
.npm
|
|
||||||
|
|
||||||
# Optional eslint cache
|
|
||||||
|
|
||||||
.eslintcache
|
|
||||||
|
|
||||||
# Optional stylelint cache
|
|
||||||
|
|
||||||
.stylelintcache
|
|
||||||
|
|
||||||
# Microbundle cache
|
|
||||||
|
|
||||||
.rpt2_cache/
|
|
||||||
.rts2_cache_cjs/
|
|
||||||
.rts2_cache_es/
|
|
||||||
.rts2_cache_umd/
|
|
||||||
|
|
||||||
# Optional REPL history
|
|
||||||
|
|
||||||
.node_repl_history
|
|
||||||
|
|
||||||
# Output of 'npm pack'
|
|
||||||
|
|
||||||
*.tgz
|
|
||||||
|
|
||||||
# Yarn Integrity file
|
|
||||||
|
|
||||||
.yarn-integrity
|
|
||||||
|
|
||||||
# dotenv environment variable files
|
|
||||||
|
|
||||||
.env
|
|
||||||
.env.development.local
|
|
||||||
.env.test.local
|
|
||||||
.env.production.local
|
|
||||||
.env.local
|
|
||||||
|
|
||||||
# parcel-bundler cache (https://parceljs.org/)
|
|
||||||
|
|
||||||
.parcel-cache
|
|
||||||
|
|
||||||
# Next.js build output
|
|
||||||
|
|
||||||
.next
|
|
||||||
out
|
|
||||||
|
|
||||||
# Nuxt.js build / generate output
|
|
||||||
|
|
||||||
.nuxt
|
|
||||||
dist
|
|
||||||
|
|
||||||
# Gatsby files
|
|
||||||
|
|
||||||
# Comment in the public line in if your project uses Gatsby and not Next.js
|
|
||||||
|
|
||||||
# https://nextjs.org/blog/next-9-1#public-directory-support
|
|
||||||
|
|
||||||
# public
|
|
||||||
|
|
||||||
# vuepress build output
|
|
||||||
|
|
||||||
.vuepress/dist
|
|
||||||
|
|
||||||
# vuepress v2.x temp and cache directory
|
|
||||||
|
|
||||||
.temp
|
|
||||||
|
|
||||||
# Docusaurus cache and generated files
|
|
||||||
|
|
||||||
.docusaurus
|
|
||||||
|
|
||||||
# Serverless directories
|
|
||||||
|
|
||||||
.serverless/
|
|
||||||
|
|
||||||
# FuseBox cache
|
|
||||||
|
|
||||||
.fusebox/
|
|
||||||
|
|
||||||
# DynamoDB Local files
|
|
||||||
|
|
||||||
.dynamodb/
|
|
||||||
|
|
||||||
# TernJS port file
|
|
||||||
|
|
||||||
.tern-port
|
|
||||||
|
|
||||||
# Stores VSCode versions used for testing VSCode extensions
|
|
||||||
|
|
||||||
.vscode-test
|
|
||||||
|
|
||||||
# yarn v2
|
|
||||||
|
|
||||||
.yarn/cache
|
|
||||||
.yarn/unplugged
|
|
||||||
.yarn/build-state.yml
|
|
||||||
.yarn/install-state.gz
|
|
||||||
.pnp.*
|
|
||||||
|
|
||||||
# IntelliJ based IDEs
|
|
||||||
.idea
|
|
||||||
|
|
||||||
# Finder (MacOS) folder config
|
|
||||||
.DS_Store
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
FROM oven/bun:1.1.45-alpine AS base
|
|
||||||
RUN apk add --no-cache git wget
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
COPY package.json bun.lockb ./
|
|
||||||
# Make sure these files are present before installing
|
|
||||||
RUN bun install --frozen-lockfile
|
|
||||||
|
|
||||||
# Now copy the rest of your source code
|
|
||||||
COPY . .
|
|
||||||
|
|
||||||
FROM base AS build
|
|
||||||
|
|
||||||
# Declare the build argument
|
|
||||||
ARG SOURCE_COMMIT
|
|
||||||
# Set the environment variable using the build argument
|
|
||||||
ENV SOURCE_COMMIT=${SOURCE_COMMIT}
|
|
||||||
|
|
||||||
ENV NODE_ENV=production
|
|
||||||
|
|
||||||
# Run build script
|
|
||||||
# RUN bun run build
|
|
||||||
|
|
||||||
FROM build AS release
|
|
||||||
USER bun
|
|
||||||
EXPOSE 3000/tcp
|
|
||||||
ENTRYPOINT [ "bun", "run", "index.ts" ]
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
# bun
|
|
||||||
|
|
||||||
To install dependencies:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
bun install
|
|
||||||
```
|
|
||||||
|
|
||||||
To run:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
bun run index.ts
|
|
||||||
```
|
|
||||||
|
|
||||||
This project was created using `bun init` in bun v1.0.25. [Bun](https://bun.sh) is a fast all-in-one JavaScript runtime.
|
|
||||||
BIN
bun/bun.lockb
BIN
bun/bun.lockb
Binary file not shown.
14
bun/index.ts
14
bun/index.ts
@@ -1,14 +0,0 @@
|
|||||||
const PORT = process.env.PORT || 3000;
|
|
||||||
Bun.serve({
|
|
||||||
port: PORT,
|
|
||||||
fetch(req) {
|
|
||||||
const url = new URL(req.url);
|
|
||||||
if (url.pathname === "/") return new Response("Home page!");
|
|
||||||
if (url.pathname === "/blog") return new Response("Blog!");
|
|
||||||
if (url.pathname === "/about") return new Response("About!");
|
|
||||||
if (url.pathname === "/x") return new Response("a");
|
|
||||||
if (url.pathname === "/201") return new Response("201", { status: 201 });
|
|
||||||
return new Response("404!");
|
|
||||||
},
|
|
||||||
})
|
|
||||||
console.log(`Server running at http://localhost:${PORT}`);
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "bun",
|
|
||||||
"module": "index.ts",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "bun --watch index.ts",
|
|
||||||
"start": "bun index.ts"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/bun": "latest"
|
|
||||||
},
|
|
||||||
"peerDependencies": {
|
|
||||||
"typescript": "^5.3.3"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
16
coolify.json
16
coolify.json
@@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
"$schema": "https://cdn.coollabs.io/coolify/coolify.schema.json",
|
|
||||||
"build": {
|
|
||||||
"build_pack": "dockerfile",
|
|
||||||
"base_directory": "/dockerfile"
|
|
||||||
},
|
|
||||||
"network": {
|
|
||||||
"domains": {
|
|
||||||
"fqdn": "http://hellotwitter.127.0.0.1.sslip.io"
|
|
||||||
},
|
|
||||||
"ports": {
|
|
||||||
"expose": "80",
|
|
||||||
"mappings": "8888:80"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
:2015
|
|
||||||
respond "Hello, world!"
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
services:
|
|
||||||
caddy:
|
|
||||||
image: caddy
|
|
||||||
ports:
|
|
||||||
- 2015:2015
|
|
||||||
volumes:
|
|
||||||
- ./Caddyfile:/etc/caddy/Caddyfile
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
hello
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
version: '3.8'
|
|
||||||
services:
|
|
||||||
broker:
|
|
||||||
image: 'docker.io/library/redis:7'
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- 'redisdata:/data'
|
|
||||||
db:
|
|
||||||
image: 'docker.io/library/postgres:15'
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- 'pgdata:/var/lib/postgresql/data'
|
|
||||||
environment:
|
|
||||||
POSTGRES_DB: paperless
|
|
||||||
POSTGRES_USER: paperless
|
|
||||||
POSTGRES_PASSWORD: paperless
|
|
||||||
webserver:
|
|
||||||
image: 'ghcr.io/paperless-ngx/paperless-ngx:latest'
|
|
||||||
restart: unless-stopped
|
|
||||||
depends_on:
|
|
||||||
- db
|
|
||||||
- broker
|
|
||||||
ports:
|
|
||||||
- '9001:8000'
|
|
||||||
volumes:
|
|
||||||
- 'smb_data:/usr/src/paperless/data'
|
|
||||||
- 'smb_media:/usr/src/paperless/media'
|
|
||||||
- 'smb_export:/usr/src/paperless/export'
|
|
||||||
- 'smb_consume:/usr/src/paperless/consume'
|
|
||||||
environment:
|
|
||||||
PAPERLESS_REDIS: 'redis://broker:6379'
|
|
||||||
PAPERLESS_DBHOST: db
|
|
||||||
PAPERLESS_CONSUMER_RECURSIVE: 'true'
|
|
||||||
PAPERLESS_URL: 'https://paperless.paperless.io'
|
|
||||||
PAPERLESS_FILENAME_FORMAT: '{created_year}{created_month}/{correspondent}{title}_{created_year}{created_month}{created_day}'
|
|
||||||
PAPERLESS_SECRET_KEY: 'sdfhaskjdfhasjdfhasljk'
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_UPSCALE: 1.5
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_DPI: 600
|
|
||||||
PAPERLESS_CONSUMER_ENABLE_BARCODES: 'true'
|
|
||||||
PAPERLESS_CONSUMER_ENABLE_TAG_BARCODE: 'true'
|
|
||||||
PAPERLESS_CONSUMER_TAG_BARCODE_MAPPING: '{"[a-fA-F0-9]{64}": "\g<0>"}'
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_SCANNER: ZXING
|
|
||||||
PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS: true
|
|
||||||
user: '0:0'
|
|
||||||
cap_add:
|
|
||||||
- SYS_ADMIN
|
|
||||||
- DAC_READ_SEARCH
|
|
||||||
privileged: true
|
|
||||||
volumes:
|
|
||||||
smb_data:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-data
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_media:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-media
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_export:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-export
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_consume:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-consume
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
pgdata: null
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
services:
|
|
||||||
app:
|
|
||||||
image: "nginx"
|
|
||||||
volumes:
|
|
||||||
- ./db/migration.sql:/docker-entrypoint-initdb.d/create_tables.sql
|
|
||||||
- ./empty:/empty
|
|
||||||
- ./404:/404
|
|
||||||
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
version: '3.8'
|
|
||||||
services:
|
|
||||||
app:
|
|
||||||
image: 'nginx'
|
|
||||||
environment:
|
|
||||||
DB_URL: 'postgres://$POSTGRES_USER:$POSTGRES_PASSWORD@db/postgres'
|
|
||||||
APP_KEY: base64
|
|
||||||
APP_DEBUG: '${APP_DEBUG:-false}'
|
|
||||||
volumes:
|
|
||||||
- './index.html:/usr/share/nginx/html/index.html'
|
|
||||||
- 'test-html:/html'
|
|
||||||
depends_on:
|
|
||||||
- db
|
|
||||||
db:
|
|
||||||
image: postgres
|
|
||||||
environment:
|
|
||||||
POSTGRES_USER: ${POSTGRES_USER:-postgres}
|
|
||||||
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:-postgres}
|
|
||||||
volumes:
|
|
||||||
- 'data:/var/lib/postgresql/data'
|
|
||||||
healthcheck:
|
|
||||||
test:
|
|
||||||
- CMD
|
|
||||||
- pg_isready
|
|
||||||
- '-U'
|
|
||||||
- postgres
|
|
||||||
interval: 2s
|
|
||||||
timeout: 10s
|
|
||||||
retries: 10
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
services:
|
|
||||||
uptimekuma:
|
|
||||||
image: louislam/uptime-kuma:1.23.13@sha256:96510915e6be539b76bcba2e6873591c67aca8a6075ff09f5b4723ae47f333fc
|
|
||||||
volumes:
|
|
||||||
- uptime-kuma:/app/data
|
|
||||||
labels:
|
|
||||||
- caddy=status.127.0.0.1.sslip.io
|
|
||||||
- caddy.reverse_proxy="* {{upstreams 3001}}"
|
|
||||||
networks:
|
|
||||||
- coolify
|
|
||||||
healthcheck:
|
|
||||||
test:
|
|
||||||
- CMD-SHELL
|
|
||||||
- extra/healthcheck
|
|
||||||
interval: 2s
|
|
||||||
timeout: 10s
|
|
||||||
retries: 15
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
uptime-kuma:
|
|
||||||
name: uptime-kuma
|
|
||||||
|
|
||||||
networks:
|
|
||||||
coolify:
|
|
||||||
external: true
|
|
||||||
@@ -1,82 +0,0 @@
|
|||||||
ervices:
|
|
||||||
db:
|
|
||||||
image: ${POSTGRES_IMAGE:-ghcr.io/getzep/postgres}:${POSTGRES_TAG:-latest}
|
|
||||||
restart: always
|
|
||||||
shm_size: "${POSTGRES_SHM_SIZE:-128mb}"
|
|
||||||
environment:
|
|
||||||
- POSTGRES_USER=${POSTGRES_USER:-postgres}
|
|
||||||
- POSTGRES_PASSWORD=${POSTGRES_PASSWORD:-postgres}
|
|
||||||
- POSTGRES_DB=${POSTGRES_DB:-postgres}
|
|
||||||
volumes:
|
|
||||||
- zep_postgres_data:/var/lib/postgresql/data
|
|
||||||
healthcheck:
|
|
||||||
test: ["CMD", "pg_isready", "-q", "-d", "${POSTGRES_DB:-postgres}", "-U", "${POSTGRES_USER:-postgres}"]
|
|
||||||
interval: 5s
|
|
||||||
timeout: 15s
|
|
||||||
retries: 3
|
|
||||||
ports:
|
|
||||||
- ":5432"
|
|
||||||
- ":10000"
|
|
||||||
|
|
||||||
nlp:
|
|
||||||
image: ${NLP_IMAGE:-ghcr.io/getzep/zep-nlp-server}:${NLP_TAG:-latest}
|
|
||||||
restart: always
|
|
||||||
environment:
|
|
||||||
- ZEP_NLP_SERVER_PORT=${ZEP_NLP_SERVER_PORT:-5557}
|
|
||||||
- ZEP_LOG_LEVEL=${ZEP_LOG_LEVEL:-info}
|
|
||||||
- ZEP_EMBEDDINGS_DEVICE=${ZEP_EMBEDDINGS_DEVICE:-cpu}
|
|
||||||
- ZEP_EMBEDDINGS_MESSAGES_MODEL=${ZEP_EMBEDDINGS_MESSAGES_MODEL:-all-MiniLM-L6-v2}
|
|
||||||
- ZEP_EMBEDDINGS_DOCUMENTS_MODEL=${ZEP_EMBEDDINGS_DOCUMENTS_MODEL:-all-MiniLM-L6-v2}
|
|
||||||
healthcheck:
|
|
||||||
test: timeout 10s bash -c ':> /dev/tcp/127.0.0.1/${ZEP_NLP_SERVER_PORT:-5557}' || exit 1
|
|
||||||
interval: 10s
|
|
||||||
timeout: 5s
|
|
||||||
retries: 5
|
|
||||||
start_period: 45s
|
|
||||||
ports:
|
|
||||||
- ":5557"
|
|
||||||
|
|
||||||
zep:
|
|
||||||
image: ${ZEP_IMAGE:-ghcr.io/getzep/zep}:${ZEP_TAG:-latest}
|
|
||||||
restart: always
|
|
||||||
#depends_on:
|
|
||||||
#db:
|
|
||||||
#condition: service_healthy
|
|
||||||
#nlp:
|
|
||||||
#condition: service_healthy
|
|
||||||
ports:
|
|
||||||
- ":8000"
|
|
||||||
volumes:
|
|
||||||
- zep_config:/app/config
|
|
||||||
build:
|
|
||||||
context: .
|
|
||||||
dockerfile: Dockerfile-mine
|
|
||||||
environment:
|
|
||||||
- ZEP_STORE_POSTGRES_DSN=postgres://${POSTGRES_USER:-postgres}:${POSTGRES_PASSWORD:-postgres}@db:5432/${POSTGRES_DB:-postgres}?sslmode=disable
|
|
||||||
- ZEP_NLP_SERVER_URL=http://nlp:${ZEP_NLP_SERVER_PORT:-5557}
|
|
||||||
- ZEP_OPENAI_API_KEY=${ZEP_OPENAI_API_KEY}
|
|
||||||
- ZEP_AUTH_REQUIRED=${ZEP_AUTH_REQUIRED:-false}
|
|
||||||
- ZEP_AUTH_SECRET=${ZEP_AUTH_SECRET}
|
|
||||||
- ZEP_LLM_SERVICE=${ZEP_LLM_SERVICE:-openai}
|
|
||||||
- ZEP_LLM_MODEL=${ZEP_LLM_MODEL:-gpt-3.5-turbo}
|
|
||||||
- ZEP_MEMORY_MESSAGE_WINDOW=${ZEP_MEMORY_MESSAGE_WINDOW:-12}
|
|
||||||
- ZEP_LOG_LEVEL=${ZEP_LOG_LEVEL:-info}
|
|
||||||
- ZEP_SERVER_MAX_REQUEST_SIZE=${ZEP_SERVER_MAX_REQUEST_SIZE:-5242880}
|
|
||||||
- ZEP_EXTRACTORS_DOCUMENTS_EMBEDDINGS_SERVICE=${ZEP_EXTRACTORS_DOCUMENTS_EMBEDDINGS_SERVICE:-openai}
|
|
||||||
- ZEP_EXTRACTORS_DOCUMENTS_EMBEDDINGS_DIMENSIONS=${ZEP_EXTRACTORS_DOCUMENTS_EMBEDDINGS_DIMENSIONS:-1536}
|
|
||||||
- ZEP_EXTRACTORS_MESSAGES_EMBEDDINGS_SERVICE=${ZEP_EXTRACTORS_MESSAGES_EMBEDDINGS_SERVICE:-openai}
|
|
||||||
- ZEP_EXTRACTORS_MESSAGES_EMBEDDINGS_DIMENSIONS=${ZEP_EXTRACTORS_MESSAGES_EMBEDDINGS_DIMENSIONS:-1536}
|
|
||||||
- ZEP_EXTRACTORS_MESSAGES_SUMMARIZER_EMBEDDINGS_SERVICE=${ZEP_EXTRACTORS_MESSAGES_SUMMARIZER_EMBEDDINGS_SERVICE:-openai}
|
|
||||||
- ZEP_EXTRACTORS_MESSAGES_SUMMARIZER_EMBEDDINGS_DIMENSIONS=${ZEP_EXTRACTORS_MESSAGES_SUMMARIZER_EMBEDDINGS_DIMENSIONS:-1536}
|
|
||||||
- ZEP_STORE_TYPE=postgres
|
|
||||||
- ZEP_SERVER_WEB_ENABLED=true
|
|
||||||
#healthcheck:
|
|
||||||
#test: timeout 10s bash -c ':> /dev/tcp/127.0.0.1/8000' || exit 1
|
|
||||||
#interval: 5s
|
|
||||||
#timeout: 10s
|
|
||||||
#retries: 3
|
|
||||||
#start_period: 40s
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
zep_postgres_data:
|
|
||||||
zep_config:
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
services:
|
|
||||||
traccar-db:
|
|
||||||
image: mariadb
|
|
||||||
environment:
|
|
||||||
MYSQL_ROOT_PASSWORD: ${MYSQL_ROOT_PASS}
|
|
||||||
MYSQL_DATABASE: traccar
|
|
||||||
MYSQL_USER: traccar
|
|
||||||
MYSQL_PASSWORD: ${MYSQL_PASS}
|
|
||||||
volumes:
|
|
||||||
- traccar_db:/var/lib/mysql
|
|
||||||
healthcheck:
|
|
||||||
test: mariadb-admin -p$$MYSQL_ROOT_PASSWORD ping -h localhost
|
|
||||||
interval: 20s
|
|
||||||
start_period: 10s
|
|
||||||
timeout: 10s
|
|
||||||
retries: 3
|
|
||||||
traccar:
|
|
||||||
restart: unless-stopped
|
|
||||||
depends_on:
|
|
||||||
traccar-db:
|
|
||||||
condition: service_healthy
|
|
||||||
volumes:
|
|
||||||
- traccar_logs:/opt/traccar/logs
|
|
||||||
- traccar_config:/opt/traccar/conf/test.xml
|
|
||||||
- test_config:/opt/traccar/logs
|
|
||||||
image: traccar/traccar:latest
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
version: '3.8'
|
|
||||||
services:
|
|
||||||
broker:
|
|
||||||
image: 'docker.io/library/redis:7'
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- 'redisdata:/data'
|
|
||||||
db:
|
|
||||||
image: 'docker.io/library/postgres:15'
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- 'pgdata:/var/lib/postgresql/data'
|
|
||||||
environment:
|
|
||||||
POSTGRES_DB: paperless
|
|
||||||
POSTGRES_USER: paperless
|
|
||||||
POSTGRES_PASSWORD: paperless
|
|
||||||
webserver:
|
|
||||||
image: 'ghcr.io/paperless-ngx/paperless-ngx:latest'
|
|
||||||
restart: unless-stopped
|
|
||||||
depends_on:
|
|
||||||
- db
|
|
||||||
- broker
|
|
||||||
ports:
|
|
||||||
- '9001:8000'
|
|
||||||
volumes:
|
|
||||||
- 'smb_data:/usr/src/paperless/data'
|
|
||||||
- 'smb_media:/usr/src/paperless/media'
|
|
||||||
- 'smb_export:/usr/src/paperless/export'
|
|
||||||
- 'smb_consume:/usr/src/paperless/consume'
|
|
||||||
environment:
|
|
||||||
PAPERLESS_REDIS: 'redis://broker:6379'
|
|
||||||
PAPERLESS_DBHOST: db
|
|
||||||
PAPERLESS_CONSUMER_RECURSIVE: 'true'
|
|
||||||
PAPERLESS_URL: 'https://paperless.paperless.io'
|
|
||||||
PAPERLESS_FILENAME_FORMAT: '{created_year}{created_month}/{correspondent}{title}_{created_year}{created_month}{created_day}'
|
|
||||||
PAPERLESS_SECRET_KEY: 'sdfhaskjdfhasjdfhasljk'
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_UPSCALE: 1.5
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_DPI: 600
|
|
||||||
PAPERLESS_CONSUMER_ENABLE_BARCODES: 'true'
|
|
||||||
PAPERLESS_CONSUMER_ENABLE_TAG_BARCODE: 'true'
|
|
||||||
PAPERLESS_CONSUMER_TAG_BARCODE_MAPPING: '{"[a-fA-F0-9]{64}": "\g<0>"}'
|
|
||||||
PAPERLESS_CONSUMER_BARCODE_SCANNER: ZXING
|
|
||||||
PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS: true
|
|
||||||
user: '0:0'
|
|
||||||
cap_add:
|
|
||||||
- SYS_ADMIN
|
|
||||||
- DAC_READ_SEARCH
|
|
||||||
privileged: true
|
|
||||||
volumes:
|
|
||||||
smb_data:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-data
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_media:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-media
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_export:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-export
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
smb_consume:
|
|
||||||
driver: local
|
|
||||||
driver_opts:
|
|
||||||
type: cifs
|
|
||||||
device: //10.10.10.11/paperless-consume
|
|
||||||
o: 'username=smbtest,password=smbtest,vers=3.0,uid=0,gid=0,dir_mode=0777,file_mode=0777'
|
|
||||||
pgdata: null
|
|
||||||
redisdata: null
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
name: stream
|
|
||||||
services:
|
|
||||||
foo:
|
|
||||||
image: busybox
|
|
||||||
command: echo "I'm running ${COMPOSE_PROJECT_NAME}"
|
|
||||||
restart: no
|
|
||||||
web:
|
|
||||||
image: nginx:latest
|
|
||||||
cap_add:
|
|
||||||
- ALL
|
|
||||||
ports:
|
|
||||||
- "8888:80"
|
|
||||||
annotations:
|
|
||||||
com.example.foo: bar
|
|
||||||
db:
|
|
||||||
image: postgres:16
|
|
||||||
environment:
|
|
||||||
POSTGRES_USER: example
|
|
||||||
POSTGRES_DB: exampledb
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
services:
|
|
||||||
app:
|
|
||||||
image: 'nginx'
|
|
||||||
tty: true
|
|
||||||
ports:
|
|
||||||
- '3333:80'
|
|
||||||
labels:
|
|
||||||
- traefik.http.middlewares.custom-auth.basicauth.users=test:$2y$12$ci.4U63YX83CwkyUrjqxAucnmi2xXOIlEF6T/KdP9824f1Rf1iyNG
|
|
||||||
app2:
|
|
||||||
image: 'nginx'
|
|
||||||
labels:
|
|
||||||
- traefik.http.middlewares.custom-auth2.basicauth.users=test:$2y$05$ok6tRzMf9LwTHibBjtwcJeDEPEr6f/gTNRlHRNthzVbF77PWPu2hK
|
|
||||||
app3:
|
|
||||||
image: 'nginx'
|
|
||||||
labels:
|
|
||||||
- traefik.http.middlewares.custom-auth3.basicauth.users=test:$2y$05$OpnNJgwdtK/PLkP.JL7Rzeikv2PB2skTZC1B3x15Zka.Vevo5ux6m
|
|
||||||
db:
|
|
||||||
image: 'postgres:alpine'
|
|
||||||
volumes:
|
|
||||||
- 'store:/store'
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
store:
|
|
||||||
driver: local
|
|
||||||
175
docker-compose/.gitignore
vendored
175
docker-compose/.gitignore
vendored
@@ -1,175 +0,0 @@
|
|||||||
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
|
|
||||||
|
|
||||||
# Logs
|
|
||||||
|
|
||||||
logs
|
|
||||||
_.log
|
|
||||||
npm-debug.log_
|
|
||||||
yarn-debug.log*
|
|
||||||
yarn-error.log*
|
|
||||||
lerna-debug.log*
|
|
||||||
.pnpm-debug.log*
|
|
||||||
|
|
||||||
# Caches
|
|
||||||
|
|
||||||
.cache
|
|
||||||
|
|
||||||
# Diagnostic reports (https://nodejs.org/api/report.html)
|
|
||||||
|
|
||||||
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
|
|
||||||
|
|
||||||
# Runtime data
|
|
||||||
|
|
||||||
pids
|
|
||||||
_.pid
|
|
||||||
_.seed
|
|
||||||
*.pid.lock
|
|
||||||
|
|
||||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
|
||||||
|
|
||||||
lib-cov
|
|
||||||
|
|
||||||
# Coverage directory used by tools like istanbul
|
|
||||||
|
|
||||||
coverage
|
|
||||||
*.lcov
|
|
||||||
|
|
||||||
# nyc test coverage
|
|
||||||
|
|
||||||
.nyc_output
|
|
||||||
|
|
||||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
|
||||||
|
|
||||||
.grunt
|
|
||||||
|
|
||||||
# Bower dependency directory (https://bower.io/)
|
|
||||||
|
|
||||||
bower_components
|
|
||||||
|
|
||||||
# node-waf configuration
|
|
||||||
|
|
||||||
.lock-wscript
|
|
||||||
|
|
||||||
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
|
||||||
|
|
||||||
build/Release
|
|
||||||
|
|
||||||
# Dependency directories
|
|
||||||
|
|
||||||
node_modules/
|
|
||||||
jspm_packages/
|
|
||||||
|
|
||||||
# Snowpack dependency directory (https://snowpack.dev/)
|
|
||||||
|
|
||||||
web_modules/
|
|
||||||
|
|
||||||
# TypeScript cache
|
|
||||||
|
|
||||||
*.tsbuildinfo
|
|
||||||
|
|
||||||
# Optional npm cache directory
|
|
||||||
|
|
||||||
.npm
|
|
||||||
|
|
||||||
# Optional eslint cache
|
|
||||||
|
|
||||||
.eslintcache
|
|
||||||
|
|
||||||
# Optional stylelint cache
|
|
||||||
|
|
||||||
.stylelintcache
|
|
||||||
|
|
||||||
# Microbundle cache
|
|
||||||
|
|
||||||
.rpt2_cache/
|
|
||||||
.rts2_cache_cjs/
|
|
||||||
.rts2_cache_es/
|
|
||||||
.rts2_cache_umd/
|
|
||||||
|
|
||||||
# Optional REPL history
|
|
||||||
|
|
||||||
.node_repl_history
|
|
||||||
|
|
||||||
# Output of 'npm pack'
|
|
||||||
|
|
||||||
*.tgz
|
|
||||||
|
|
||||||
# Yarn Integrity file
|
|
||||||
|
|
||||||
.yarn-integrity
|
|
||||||
|
|
||||||
# dotenv environment variable files
|
|
||||||
|
|
||||||
.env
|
|
||||||
.env.development.local
|
|
||||||
.env.test.local
|
|
||||||
.env.production.local
|
|
||||||
.env.local
|
|
||||||
|
|
||||||
# parcel-bundler cache (https://parceljs.org/)
|
|
||||||
|
|
||||||
.parcel-cache
|
|
||||||
|
|
||||||
# Next.js build output
|
|
||||||
|
|
||||||
.next
|
|
||||||
out
|
|
||||||
|
|
||||||
# Nuxt.js build / generate output
|
|
||||||
|
|
||||||
.nuxt
|
|
||||||
dist
|
|
||||||
|
|
||||||
# Gatsby files
|
|
||||||
|
|
||||||
# Comment in the public line in if your project uses Gatsby and not Next.js
|
|
||||||
|
|
||||||
# https://nextjs.org/blog/next-9-1#public-directory-support
|
|
||||||
|
|
||||||
# public
|
|
||||||
|
|
||||||
# vuepress build output
|
|
||||||
|
|
||||||
.vuepress/dist
|
|
||||||
|
|
||||||
# vuepress v2.x temp and cache directory
|
|
||||||
|
|
||||||
.temp
|
|
||||||
|
|
||||||
# Docusaurus cache and generated files
|
|
||||||
|
|
||||||
.docusaurus
|
|
||||||
|
|
||||||
# Serverless directories
|
|
||||||
|
|
||||||
.serverless/
|
|
||||||
|
|
||||||
# FuseBox cache
|
|
||||||
|
|
||||||
.fusebox/
|
|
||||||
|
|
||||||
# DynamoDB Local files
|
|
||||||
|
|
||||||
.dynamodb/
|
|
||||||
|
|
||||||
# TernJS port file
|
|
||||||
|
|
||||||
.tern-port
|
|
||||||
|
|
||||||
# Stores VSCode versions used for testing VSCode extensions
|
|
||||||
|
|
||||||
.vscode-test
|
|
||||||
|
|
||||||
# yarn v2
|
|
||||||
|
|
||||||
.yarn/cache
|
|
||||||
.yarn/unplugged
|
|
||||||
.yarn/build-state.yml
|
|
||||||
.yarn/install-state.gz
|
|
||||||
.pnp.*
|
|
||||||
|
|
||||||
# IntelliJ based IDEs
|
|
||||||
.idea
|
|
||||||
|
|
||||||
# Finder (MacOS) folder config
|
|
||||||
.DS_Store
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
FROM oven/bun
|
|
||||||
WORKDIR /app
|
|
||||||
COPY . /app
|
|
||||||
RUN env
|
|
||||||
RUN bun install
|
|
||||||
CMD bun start
|
|
||||||
Binary file not shown.
@@ -1,39 +0,0 @@
|
|||||||
services:
|
|
||||||
api:
|
|
||||||
build:
|
|
||||||
context: .
|
|
||||||
dockerfile: Dockerfile
|
|
||||||
ports:
|
|
||||||
- "3000"
|
|
||||||
environment:
|
|
||||||
- FQDN=${SERVICE_FQDN_API}
|
|
||||||
- PORT=3000
|
|
||||||
- DB_CONNECTION=postgres
|
|
||||||
- TEST=${TEST}
|
|
||||||
- TEST_PASSWORD=${SERVICE_PASSWORD_TEST}
|
|
||||||
database:
|
|
||||||
image: postgres:14-alpine
|
|
||||||
volumes:
|
|
||||||
- db-data:/var/lib/postgresql/data
|
|
||||||
- ./asd/asd:/asd:ro
|
|
||||||
- type: bind
|
|
||||||
source: ./app/pg.sql
|
|
||||||
target: /app/pg.sql
|
|
||||||
content: |
|
|
||||||
-- NOTE: change to your own passwords for production environments
|
|
||||||
\set pgpass `echo "$POSTGRES_PASSWORD"`
|
|
||||||
|
|
||||||
ALTER USER authenticator WITH PASSWORD :'pgpass';
|
|
||||||
ALTER USER pgbouncer WITH PASSWORD :'pgpass';
|
|
||||||
- /asdf:/asdf
|
|
||||||
- type: bind
|
|
||||||
source: ./app
|
|
||||||
target: /app
|
|
||||||
read_only: true
|
|
||||||
- type: bind
|
|
||||||
source: /1234
|
|
||||||
target: /123
|
|
||||||
read_only: false
|
|
||||||
environment:
|
|
||||||
- POSTGRES_DB=database
|
|
||||||
- POSTGRES_PASSWORD=password
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
services:
|
|
||||||
api:
|
|
||||||
build:
|
|
||||||
context: .
|
|
||||||
dockerfile: Dockerfile
|
|
||||||
args:
|
|
||||||
NPM_TOKEN: ${NPM_TOKEN:?NPM_TOKEN not set}
|
|
||||||
ports:
|
|
||||||
- "3000"
|
|
||||||
environment:
|
|
||||||
- PORT=3000
|
|
||||||
- DB_CONNECTION=postgres
|
|
||||||
- TEST=${TEST}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
const PORT = process.env.PORT || 3000;
|
|
||||||
const HOSTNAME = process.env.HOST || "0.0.0.0"
|
|
||||||
Bun.serve({
|
|
||||||
hostname: HOSTNAME,
|
|
||||||
port: PORT,
|
|
||||||
fetch(req) {
|
|
||||||
const url = new URL(req.url);
|
|
||||||
if (url.pathname === "/") return new Response("Home page!");
|
|
||||||
if (url.pathname === "/envs") {
|
|
||||||
return new Response(JSON.stringify(process.env, null, 2), {
|
|
||||||
headers: { "content-type": "application/json" },
|
|
||||||
});
|
|
||||||
|
|
||||||
};
|
|
||||||
return new Response("404!");
|
|
||||||
},
|
|
||||||
})
|
|
||||||
console.log(`Server running at http://${HOSTNAME}:${PORT}`);
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "bun",
|
|
||||||
"module": "index.ts",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "bun --watch index.ts",
|
|
||||||
"start": "bun index.ts"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/bun": "latest"
|
|
||||||
},
|
|
||||||
"peerDependencies": {
|
|
||||||
"typescript": "^5.3.3"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
FROM nginx:alpine as base
|
|
||||||
ENV DATABASE_URL=$DATABASE_URL
|
|
||||||
RUN echo -e "\e[1;33mENVIRONMENT VARIABLES\e[0m"
|
|
||||||
RUN env
|
|
||||||
RUN echo "Hello, World!" > /usr/share/nginx/html/index.html
|
|
||||||
RUN echo "Hello, World about!" > /usr/share/nginx/html/about.html
|
|
||||||
#HEALTHCHECK --interval=5s --timeout=3s --start-period=30s --retries=3 CMD wget -qO- http://localhost:80 || exit 1
|
|
||||||
|
|
||||||
FROM base AS builder
|
|
||||||
RUN echo -e "\e[1;33mENVIRONMENT VARIABLES\e[0m"
|
|
||||||
RUN env
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "Git Based Dockerfile",
|
|
||||||
"build": {
|
|
||||||
"build_pack": "dockerfile"
|
|
||||||
},
|
|
||||||
"network": {
|
|
||||||
"ports": {
|
|
||||||
"expose": "80"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
[
|
|
||||||
import_deps: [:ecto, :ecto_sql, :phoenix],
|
|
||||||
subdirectories: ["priv/*/migrations"],
|
|
||||||
plugins: [Phoenix.LiveView.HTMLFormatter],
|
|
||||||
inputs: ["*.{heex,ex,exs}", "{config,lib,test}/**/*.{heex,ex,exs}", "priv/*/seeds.exs"]
|
|
||||||
]
|
|
||||||
37
elixir-phoenix/.gitignore
vendored
37
elixir-phoenix/.gitignore
vendored
@@ -1,37 +0,0 @@
|
|||||||
# The directory Mix will write compiled artifacts to.
|
|
||||||
/_build/
|
|
||||||
|
|
||||||
# If you run "mix test --cover", coverage assets end up here.
|
|
||||||
/cover/
|
|
||||||
|
|
||||||
# The directory Mix downloads your dependencies sources to.
|
|
||||||
/deps/
|
|
||||||
|
|
||||||
# Where 3rd-party dependencies like ExDoc output generated docs.
|
|
||||||
/doc/
|
|
||||||
|
|
||||||
# Ignore .fetch files in case you like to edit your project deps locally.
|
|
||||||
/.fetch
|
|
||||||
|
|
||||||
# If the VM crashes, it generates a dump, let's ignore it too.
|
|
||||||
erl_crash.dump
|
|
||||||
|
|
||||||
# Also ignore archive artifacts (built via "mix archive.build").
|
|
||||||
*.ez
|
|
||||||
|
|
||||||
# Temporary files, for example, from tests.
|
|
||||||
/tmp/
|
|
||||||
|
|
||||||
# Ignore package tarball (built via "mix hex.build").
|
|
||||||
hello-*.tar
|
|
||||||
|
|
||||||
# Ignore assets that are produced by build tools.
|
|
||||||
/priv/static/assets/
|
|
||||||
|
|
||||||
# Ignore digested assets cache.
|
|
||||||
/priv/static/cache_manifest.json
|
|
||||||
|
|
||||||
# In case you use Node.js/npm, you want to ignore these.
|
|
||||||
npm-debug.log
|
|
||||||
/assets/node_modules/
|
|
||||||
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
# Elixir Phoenix Configuration
|
|
||||||
|
|
||||||
- Set `Build Pack` to `nixpacks`
|
|
||||||
- Set `MIX_ENV` to `prod`
|
|
||||||
- It should be a `Build time` environment variable
|
|
||||||
- Set `SECRET_KEY_BASE` to a random string (https://hexdocs.pm/phoenix/deployment.html#handling-of-your-application-secrets)
|
|
||||||
- It should be a `Build time` environment variable
|
|
||||||
- Set `DATABASE_URL` to your database connection string
|
|
||||||
- It should be a `Build time` environment variable
|
|
||||||
- Set `Ports Exposes` to `4000` (default)
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
@import "tailwindcss/base";
|
|
||||||
@import "tailwindcss/components";
|
|
||||||
@import "tailwindcss/utilities";
|
|
||||||
|
|
||||||
/* This file is for your main application CSS */
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
// If you want to use Phoenix channels, run `mix help phx.gen.channel`
|
|
||||||
// to get started and then uncomment the line below.
|
|
||||||
// import "./user_socket.js"
|
|
||||||
|
|
||||||
// You can include dependencies in two ways.
|
|
||||||
//
|
|
||||||
// The simplest option is to put them in assets/vendor and
|
|
||||||
// import them using relative paths:
|
|
||||||
//
|
|
||||||
// import "../vendor/some-package.js"
|
|
||||||
//
|
|
||||||
// Alternatively, you can `npm install some-package --prefix assets` and import
|
|
||||||
// them using a path starting with the package name:
|
|
||||||
//
|
|
||||||
// import "some-package"
|
|
||||||
//
|
|
||||||
|
|
||||||
// Include phoenix_html to handle method=PUT/DELETE in forms and buttons.
|
|
||||||
import "phoenix_html"
|
|
||||||
// Establish Phoenix Socket and LiveView configuration.
|
|
||||||
import {Socket} from "phoenix"
|
|
||||||
import {LiveSocket} from "phoenix_live_view"
|
|
||||||
import topbar from "../vendor/topbar"
|
|
||||||
|
|
||||||
let csrfToken = document.querySelector("meta[name='csrf-token']").getAttribute("content")
|
|
||||||
let liveSocket = new LiveSocket("/live", Socket, {
|
|
||||||
longPollFallbackMs: 2500,
|
|
||||||
params: {_csrf_token: csrfToken}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Show progress bar on live navigation and form submits
|
|
||||||
topbar.config({barColors: {0: "#29d"}, shadowColor: "rgba(0, 0, 0, .3)"})
|
|
||||||
window.addEventListener("phx:page-loading-start", _info => topbar.show(300))
|
|
||||||
window.addEventListener("phx:page-loading-stop", _info => topbar.hide())
|
|
||||||
|
|
||||||
// connect if there are any LiveViews on the page
|
|
||||||
liveSocket.connect()
|
|
||||||
|
|
||||||
// expose liveSocket on window for web console debug logs and latency simulation:
|
|
||||||
// >> liveSocket.enableDebug()
|
|
||||||
// >> liveSocket.enableLatencySim(1000) // enabled for duration of browser session
|
|
||||||
// >> liveSocket.disableLatencySim()
|
|
||||||
window.liveSocket = liveSocket
|
|
||||||
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
// See the Tailwind configuration guide for advanced usage
|
|
||||||
// https://tailwindcss.com/docs/configuration
|
|
||||||
|
|
||||||
const plugin = require("tailwindcss/plugin")
|
|
||||||
const fs = require("fs")
|
|
||||||
const path = require("path")
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
content: [
|
|
||||||
"./js/**/*.js",
|
|
||||||
"../lib/hello_web.ex",
|
|
||||||
"../lib/hello_web/**/*.*ex"
|
|
||||||
],
|
|
||||||
theme: {
|
|
||||||
extend: {
|
|
||||||
colors: {
|
|
||||||
brand: "#FD4F00",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
plugins: [
|
|
||||||
require("@tailwindcss/forms"),
|
|
||||||
// Allows prefixing tailwind classes with LiveView classes to add rules
|
|
||||||
// only when LiveView classes are applied, for example:
|
|
||||||
//
|
|
||||||
// <div class="phx-click-loading:animate-ping">
|
|
||||||
//
|
|
||||||
plugin(({addVariant}) => addVariant("phx-click-loading", [".phx-click-loading&", ".phx-click-loading &"])),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-submit-loading", [".phx-submit-loading&", ".phx-submit-loading &"])),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-change-loading", [".phx-change-loading&", ".phx-change-loading &"])),
|
|
||||||
|
|
||||||
// Embeds Heroicons (https://heroicons.com) into your app.css bundle
|
|
||||||
// See your `CoreComponents.icon/1` for more information.
|
|
||||||
//
|
|
||||||
plugin(function({matchComponents, theme}) {
|
|
||||||
let iconsDir = path.join(__dirname, "../deps/heroicons/optimized")
|
|
||||||
let values = {}
|
|
||||||
let icons = [
|
|
||||||
["", "/24/outline"],
|
|
||||||
["-solid", "/24/solid"],
|
|
||||||
["-mini", "/20/solid"],
|
|
||||||
["-micro", "/16/solid"]
|
|
||||||
]
|
|
||||||
icons.forEach(([suffix, dir]) => {
|
|
||||||
fs.readdirSync(path.join(iconsDir, dir)).forEach(file => {
|
|
||||||
let name = path.basename(file, ".svg") + suffix
|
|
||||||
values[name] = {name, fullPath: path.join(iconsDir, dir, file)}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
matchComponents({
|
|
||||||
"hero": ({name, fullPath}) => {
|
|
||||||
let content = fs.readFileSync(fullPath).toString().replace(/\r?\n|\r/g, "")
|
|
||||||
let size = theme("spacing.6")
|
|
||||||
if (name.endsWith("-mini")) {
|
|
||||||
size = theme("spacing.5")
|
|
||||||
} else if (name.endsWith("-micro")) {
|
|
||||||
size = theme("spacing.4")
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
[`--hero-${name}`]: `url('data:image/svg+xml;utf8,${content}')`,
|
|
||||||
"-webkit-mask": `var(--hero-${name})`,
|
|
||||||
"mask": `var(--hero-${name})`,
|
|
||||||
"mask-repeat": "no-repeat",
|
|
||||||
"background-color": "currentColor",
|
|
||||||
"vertical-align": "middle",
|
|
||||||
"display": "inline-block",
|
|
||||||
"width": size,
|
|
||||||
"height": size
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, {values})
|
|
||||||
})
|
|
||||||
]
|
|
||||||
}
|
|
||||||
165
elixir-phoenix/assets/vendor/topbar.js
vendored
165
elixir-phoenix/assets/vendor/topbar.js
vendored
@@ -1,165 +0,0 @@
|
|||||||
/**
|
|
||||||
* @license MIT
|
|
||||||
* topbar 2.0.0, 2023-02-04
|
|
||||||
* https://buunguyen.github.io/topbar
|
|
||||||
* Copyright (c) 2021 Buu Nguyen
|
|
||||||
*/
|
|
||||||
(function (window, document) {
|
|
||||||
"use strict";
|
|
||||||
|
|
||||||
// https://gist.github.com/paulirish/1579671
|
|
||||||
(function () {
|
|
||||||
var lastTime = 0;
|
|
||||||
var vendors = ["ms", "moz", "webkit", "o"];
|
|
||||||
for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
|
|
||||||
window.requestAnimationFrame =
|
|
||||||
window[vendors[x] + "RequestAnimationFrame"];
|
|
||||||
window.cancelAnimationFrame =
|
|
||||||
window[vendors[x] + "CancelAnimationFrame"] ||
|
|
||||||
window[vendors[x] + "CancelRequestAnimationFrame"];
|
|
||||||
}
|
|
||||||
if (!window.requestAnimationFrame)
|
|
||||||
window.requestAnimationFrame = function (callback, element) {
|
|
||||||
var currTime = new Date().getTime();
|
|
||||||
var timeToCall = Math.max(0, 16 - (currTime - lastTime));
|
|
||||||
var id = window.setTimeout(function () {
|
|
||||||
callback(currTime + timeToCall);
|
|
||||||
}, timeToCall);
|
|
||||||
lastTime = currTime + timeToCall;
|
|
||||||
return id;
|
|
||||||
};
|
|
||||||
if (!window.cancelAnimationFrame)
|
|
||||||
window.cancelAnimationFrame = function (id) {
|
|
||||||
clearTimeout(id);
|
|
||||||
};
|
|
||||||
})();
|
|
||||||
|
|
||||||
var canvas,
|
|
||||||
currentProgress,
|
|
||||||
showing,
|
|
||||||
progressTimerId = null,
|
|
||||||
fadeTimerId = null,
|
|
||||||
delayTimerId = null,
|
|
||||||
addEvent = function (elem, type, handler) {
|
|
||||||
if (elem.addEventListener) elem.addEventListener(type, handler, false);
|
|
||||||
else if (elem.attachEvent) elem.attachEvent("on" + type, handler);
|
|
||||||
else elem["on" + type] = handler;
|
|
||||||
},
|
|
||||||
options = {
|
|
||||||
autoRun: true,
|
|
||||||
barThickness: 3,
|
|
||||||
barColors: {
|
|
||||||
0: "rgba(26, 188, 156, .9)",
|
|
||||||
".25": "rgba(52, 152, 219, .9)",
|
|
||||||
".50": "rgba(241, 196, 15, .9)",
|
|
||||||
".75": "rgba(230, 126, 34, .9)",
|
|
||||||
"1.0": "rgba(211, 84, 0, .9)",
|
|
||||||
},
|
|
||||||
shadowBlur: 10,
|
|
||||||
shadowColor: "rgba(0, 0, 0, .6)",
|
|
||||||
className: null,
|
|
||||||
},
|
|
||||||
repaint = function () {
|
|
||||||
canvas.width = window.innerWidth;
|
|
||||||
canvas.height = options.barThickness * 5; // need space for shadow
|
|
||||||
|
|
||||||
var ctx = canvas.getContext("2d");
|
|
||||||
ctx.shadowBlur = options.shadowBlur;
|
|
||||||
ctx.shadowColor = options.shadowColor;
|
|
||||||
|
|
||||||
var lineGradient = ctx.createLinearGradient(0, 0, canvas.width, 0);
|
|
||||||
for (var stop in options.barColors)
|
|
||||||
lineGradient.addColorStop(stop, options.barColors[stop]);
|
|
||||||
ctx.lineWidth = options.barThickness;
|
|
||||||
ctx.beginPath();
|
|
||||||
ctx.moveTo(0, options.barThickness / 2);
|
|
||||||
ctx.lineTo(
|
|
||||||
Math.ceil(currentProgress * canvas.width),
|
|
||||||
options.barThickness / 2
|
|
||||||
);
|
|
||||||
ctx.strokeStyle = lineGradient;
|
|
||||||
ctx.stroke();
|
|
||||||
},
|
|
||||||
createCanvas = function () {
|
|
||||||
canvas = document.createElement("canvas");
|
|
||||||
var style = canvas.style;
|
|
||||||
style.position = "fixed";
|
|
||||||
style.top = style.left = style.right = style.margin = style.padding = 0;
|
|
||||||
style.zIndex = 100001;
|
|
||||||
style.display = "none";
|
|
||||||
if (options.className) canvas.classList.add(options.className);
|
|
||||||
document.body.appendChild(canvas);
|
|
||||||
addEvent(window, "resize", repaint);
|
|
||||||
},
|
|
||||||
topbar = {
|
|
||||||
config: function (opts) {
|
|
||||||
for (var key in opts)
|
|
||||||
if (options.hasOwnProperty(key)) options[key] = opts[key];
|
|
||||||
},
|
|
||||||
show: function (delay) {
|
|
||||||
if (showing) return;
|
|
||||||
if (delay) {
|
|
||||||
if (delayTimerId) return;
|
|
||||||
delayTimerId = setTimeout(() => topbar.show(), delay);
|
|
||||||
} else {
|
|
||||||
showing = true;
|
|
||||||
if (fadeTimerId !== null) window.cancelAnimationFrame(fadeTimerId);
|
|
||||||
if (!canvas) createCanvas();
|
|
||||||
canvas.style.opacity = 1;
|
|
||||||
canvas.style.display = "block";
|
|
||||||
topbar.progress(0);
|
|
||||||
if (options.autoRun) {
|
|
||||||
(function loop() {
|
|
||||||
progressTimerId = window.requestAnimationFrame(loop);
|
|
||||||
topbar.progress(
|
|
||||||
"+" + 0.05 * Math.pow(1 - Math.sqrt(currentProgress), 2)
|
|
||||||
);
|
|
||||||
})();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
progress: function (to) {
|
|
||||||
if (typeof to === "undefined") return currentProgress;
|
|
||||||
if (typeof to === "string") {
|
|
||||||
to =
|
|
||||||
(to.indexOf("+") >= 0 || to.indexOf("-") >= 0
|
|
||||||
? currentProgress
|
|
||||||
: 0) + parseFloat(to);
|
|
||||||
}
|
|
||||||
currentProgress = to > 1 ? 1 : to;
|
|
||||||
repaint();
|
|
||||||
return currentProgress;
|
|
||||||
},
|
|
||||||
hide: function () {
|
|
||||||
clearTimeout(delayTimerId);
|
|
||||||
delayTimerId = null;
|
|
||||||
if (!showing) return;
|
|
||||||
showing = false;
|
|
||||||
if (progressTimerId != null) {
|
|
||||||
window.cancelAnimationFrame(progressTimerId);
|
|
||||||
progressTimerId = null;
|
|
||||||
}
|
|
||||||
(function loop() {
|
|
||||||
if (topbar.progress("+.1") >= 1) {
|
|
||||||
canvas.style.opacity -= 0.05;
|
|
||||||
if (canvas.style.opacity <= 0.05) {
|
|
||||||
canvas.style.display = "none";
|
|
||||||
fadeTimerId = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fadeTimerId = window.requestAnimationFrame(loop);
|
|
||||||
})();
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
if (typeof module === "object" && typeof module.exports === "object") {
|
|
||||||
module.exports = topbar;
|
|
||||||
} else if (typeof define === "function" && define.amd) {
|
|
||||||
define(function () {
|
|
||||||
return topbar;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
this.topbar = topbar;
|
|
||||||
}
|
|
||||||
}.call(this, window, document));
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
# This file is responsible for configuring your application
|
|
||||||
# and its dependencies with the aid of the Config module.
|
|
||||||
#
|
|
||||||
# This configuration file is loaded before any dependency and
|
|
||||||
# is restricted to this project.
|
|
||||||
|
|
||||||
# General application configuration
|
|
||||||
import Config
|
|
||||||
|
|
||||||
config :hello,
|
|
||||||
ecto_repos: [Hello.Repo],
|
|
||||||
generators: [timestamp_type: :utc_datetime]
|
|
||||||
|
|
||||||
# Configures the endpoint
|
|
||||||
config :hello, HelloWeb.Endpoint,
|
|
||||||
url: [host: "localhost"],
|
|
||||||
adapter: Bandit.PhoenixAdapter,
|
|
||||||
render_errors: [
|
|
||||||
formats: [html: HelloWeb.ErrorHTML, json: HelloWeb.ErrorJSON],
|
|
||||||
layout: false
|
|
||||||
],
|
|
||||||
pubsub_server: Hello.PubSub,
|
|
||||||
live_view: [signing_salt: "iosHDknx"]
|
|
||||||
|
|
||||||
# Configures the mailer
|
|
||||||
#
|
|
||||||
# By default it uses the "Local" adapter which stores the emails
|
|
||||||
# locally. You can see the emails in your browser, at "/dev/mailbox".
|
|
||||||
#
|
|
||||||
# For production it's recommended to configure a different adapter
|
|
||||||
# at the `config/runtime.exs`.
|
|
||||||
config :hello, Hello.Mailer, adapter: Swoosh.Adapters.Local
|
|
||||||
|
|
||||||
# Configure esbuild (the version is required)
|
|
||||||
config :esbuild,
|
|
||||||
version: "0.17.11",
|
|
||||||
hello: [
|
|
||||||
args:
|
|
||||||
~w(js/app.js --bundle --target=es2017 --outdir=../priv/static/assets --external:/fonts/* --external:/images/*),
|
|
||||||
cd: Path.expand("../assets", __DIR__),
|
|
||||||
env: %{"NODE_PATH" => Path.expand("../deps", __DIR__)}
|
|
||||||
]
|
|
||||||
|
|
||||||
# Configure tailwind (the version is required)
|
|
||||||
config :tailwind,
|
|
||||||
version: "3.4.3",
|
|
||||||
hello: [
|
|
||||||
args: ~w(
|
|
||||||
--config=tailwind.config.js
|
|
||||||
--input=css/app.css
|
|
||||||
--output=../priv/static/assets/app.css
|
|
||||||
),
|
|
||||||
cd: Path.expand("../assets", __DIR__)
|
|
||||||
]
|
|
||||||
|
|
||||||
# Configures Elixir's Logger
|
|
||||||
config :logger, :console,
|
|
||||||
format: "$time $metadata[$level] $message\n",
|
|
||||||
metadata: [:request_id]
|
|
||||||
|
|
||||||
# Use Jason for JSON parsing in Phoenix
|
|
||||||
config :phoenix, :json_library, Jason
|
|
||||||
|
|
||||||
# Import environment specific config. This must remain at the bottom
|
|
||||||
# of this file so it overrides the configuration defined above.
|
|
||||||
import_config "#{config_env()}.exs"
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# Configure your database
|
|
||||||
config :hello, Hello.Repo,
|
|
||||||
username: "postgres",
|
|
||||||
password: "hello",
|
|
||||||
hostname: "localhost",
|
|
||||||
port: 8888,
|
|
||||||
database: "hello_dev",
|
|
||||||
stacktrace: true,
|
|
||||||
show_sensitive_data_on_connection_error: true,
|
|
||||||
pool_size: 10
|
|
||||||
|
|
||||||
# For development, we disable any cache and enable
|
|
||||||
# debugging and code reloading.
|
|
||||||
#
|
|
||||||
# The watchers configuration can be used to run external
|
|
||||||
# watchers to your application. For example, we can use it
|
|
||||||
# to bundle .js and .css sources.
|
|
||||||
config :hello, HelloWeb.Endpoint,
|
|
||||||
# Binding to loopback ipv4 address prevents access from other machines.
|
|
||||||
# Change to `ip: {0, 0, 0, 0}` to allow access from other machines.
|
|
||||||
http: [ip: {127, 0, 0, 1}, port: 4000],
|
|
||||||
check_origin: false,
|
|
||||||
code_reloader: true,
|
|
||||||
debug_errors: true,
|
|
||||||
secret_key_base: "bL9x+aexFk9wlXM9rBvHX+jjF4xCthi0+GzepROu93bEpoyz6wNz79MyNW8pvpd6",
|
|
||||||
watchers: [
|
|
||||||
esbuild: {Esbuild, :install_and_run, [:hello, ~w(--sourcemap=inline --watch)]},
|
|
||||||
tailwind: {Tailwind, :install_and_run, [:hello, ~w(--watch)]}
|
|
||||||
]
|
|
||||||
|
|
||||||
# ## SSL Support
|
|
||||||
#
|
|
||||||
# In order to use HTTPS in development, a self-signed
|
|
||||||
# certificate can be generated by running the following
|
|
||||||
# Mix task:
|
|
||||||
#
|
|
||||||
# mix phx.gen.cert
|
|
||||||
#
|
|
||||||
# Run `mix help phx.gen.cert` for more information.
|
|
||||||
#
|
|
||||||
# The `http:` config above can be replaced with:
|
|
||||||
#
|
|
||||||
# https: [
|
|
||||||
# port: 4001,
|
|
||||||
# cipher_suite: :strong,
|
|
||||||
# keyfile: "priv/cert/selfsigned_key.pem",
|
|
||||||
# certfile: "priv/cert/selfsigned.pem"
|
|
||||||
# ],
|
|
||||||
#
|
|
||||||
# If desired, both `http:` and `https:` keys can be
|
|
||||||
# configured to run both http and https servers on
|
|
||||||
# different ports.
|
|
||||||
|
|
||||||
# Watch static and templates for browser reloading.
|
|
||||||
config :hello, HelloWeb.Endpoint,
|
|
||||||
live_reload: [
|
|
||||||
patterns: [
|
|
||||||
~r"priv/static/(?!uploads/).*(js|css|png|jpeg|jpg|gif|svg)$",
|
|
||||||
~r"priv/gettext/.*(po)$",
|
|
||||||
~r"lib/hello_web/(controllers|live|components)/.*(ex|heex)$"
|
|
||||||
]
|
|
||||||
]
|
|
||||||
|
|
||||||
# Enable dev routes for dashboard and mailbox
|
|
||||||
config :hello, dev_routes: true
|
|
||||||
|
|
||||||
# Do not include metadata nor timestamps in development logs
|
|
||||||
config :logger, :console, format: "[$level] $message\n"
|
|
||||||
|
|
||||||
# Set a higher stacktrace during development. Avoid configuring such
|
|
||||||
# in production as building large stacktraces may be expensive.
|
|
||||||
config :phoenix, :stacktrace_depth, 20
|
|
||||||
|
|
||||||
# Initialize plugs at runtime for faster development compilation
|
|
||||||
config :phoenix, :plug_init_mode, :runtime
|
|
||||||
|
|
||||||
config :phoenix_live_view,
|
|
||||||
# Include HEEx debug annotations as HTML comments in rendered markup
|
|
||||||
debug_heex_annotations: true,
|
|
||||||
# Enable helpful, but potentially expensive runtime checks
|
|
||||||
enable_expensive_runtime_checks: true
|
|
||||||
|
|
||||||
# Disable swoosh api client as it is only required for production adapters.
|
|
||||||
config :swoosh, :api_client, false
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# Note we also include the path to a cache manifest
|
|
||||||
# containing the digested version of static files. This
|
|
||||||
# manifest is generated by the `mix assets.deploy` task,
|
|
||||||
# which you should run after static files are built and
|
|
||||||
# before starting your production server.
|
|
||||||
config :hello, HelloWeb.Endpoint, cache_static_manifest: "priv/static/cache_manifest.json"
|
|
||||||
|
|
||||||
# Configures Swoosh API Client
|
|
||||||
config :swoosh, api_client: Swoosh.ApiClient.Finch, finch_name: Hello.Finch
|
|
||||||
|
|
||||||
# Disable Swoosh Local Memory Storage
|
|
||||||
config :swoosh, local: false
|
|
||||||
|
|
||||||
# Do not print debug messages in production
|
|
||||||
config :logger, level: :info
|
|
||||||
|
|
||||||
# Runtime production configuration, including reading
|
|
||||||
# of environment variables, is done on config/runtime.exs.
|
|
||||||
@@ -1,117 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# config/runtime.exs is executed for all environments, including
|
|
||||||
# during releases. It is executed after compilation and before the
|
|
||||||
# system starts, so it is typically used to load production configuration
|
|
||||||
# and secrets from environment variables or elsewhere. Do not define
|
|
||||||
# any compile-time configuration in here, as it won't be applied.
|
|
||||||
# The block below contains prod specific runtime configuration.
|
|
||||||
|
|
||||||
# ## Using releases
|
|
||||||
#
|
|
||||||
# If you use `mix release`, you need to explicitly enable the server
|
|
||||||
# by passing the PHX_SERVER=true when you start it:
|
|
||||||
#
|
|
||||||
# PHX_SERVER=true bin/hello start
|
|
||||||
#
|
|
||||||
# Alternatively, you can use `mix phx.gen.release` to generate a `bin/server`
|
|
||||||
# script that automatically sets the env var above.
|
|
||||||
if System.get_env("PHX_SERVER") do
|
|
||||||
config :hello, HelloWeb.Endpoint, server: true
|
|
||||||
end
|
|
||||||
|
|
||||||
if config_env() == :prod do
|
|
||||||
database_url =
|
|
||||||
System.get_env("DATABASE_URL") ||
|
|
||||||
raise """
|
|
||||||
environment variable DATABASE_URL is missing.
|
|
||||||
For example: ecto://USER:PASS@HOST/DATABASE
|
|
||||||
"""
|
|
||||||
|
|
||||||
maybe_ipv6 = if System.get_env("ECTO_IPV6") in ~w(true 1), do: [:inet6], else: []
|
|
||||||
|
|
||||||
config :hello, Hello.Repo,
|
|
||||||
# ssl: true,
|
|
||||||
url: database_url,
|
|
||||||
pool_size: String.to_integer(System.get_env("POOL_SIZE") || "10"),
|
|
||||||
socket_options: maybe_ipv6
|
|
||||||
|
|
||||||
# The secret key base is used to sign/encrypt cookies and other secrets.
|
|
||||||
# A default value is used in config/dev.exs and config/test.exs but you
|
|
||||||
# want to use a different value for prod and you most likely don't want
|
|
||||||
# to check this value into version control, so we use an environment
|
|
||||||
# variable instead.
|
|
||||||
secret_key_base =
|
|
||||||
System.get_env("SECRET_KEY_BASE") ||
|
|
||||||
raise """
|
|
||||||
environment variable SECRET_KEY_BASE is missing.
|
|
||||||
You can generate one by calling: mix phx.gen.secret
|
|
||||||
"""
|
|
||||||
|
|
||||||
host = System.get_env("PHX_HOST") || "example.com"
|
|
||||||
port = String.to_integer(System.get_env("PORT") || "4000")
|
|
||||||
|
|
||||||
config :hello, :dns_cluster_query, System.get_env("DNS_CLUSTER_QUERY")
|
|
||||||
|
|
||||||
config :hello, HelloWeb.Endpoint,
|
|
||||||
url: [host: host, port: 443, scheme: "https"],
|
|
||||||
http: [
|
|
||||||
# Enable IPv6 and bind on all interfaces.
|
|
||||||
# Set it to {0, 0, 0, 0, 0, 0, 0, 1} for local network only access.
|
|
||||||
# See the documentation on https://hexdocs.pm/bandit/Bandit.html#t:options/0
|
|
||||||
# for details about using IPv6 vs IPv4 and loopback vs public addresses.
|
|
||||||
ip: {0, 0, 0, 0, 0, 0, 0, 0},
|
|
||||||
port: port
|
|
||||||
],
|
|
||||||
secret_key_base: secret_key_base
|
|
||||||
|
|
||||||
# ## SSL Support
|
|
||||||
#
|
|
||||||
# To get SSL working, you will need to add the `https` key
|
|
||||||
# to your endpoint configuration:
|
|
||||||
#
|
|
||||||
# config :hello, HelloWeb.Endpoint,
|
|
||||||
# https: [
|
|
||||||
# ...,
|
|
||||||
# port: 443,
|
|
||||||
# cipher_suite: :strong,
|
|
||||||
# keyfile: System.get_env("SOME_APP_SSL_KEY_PATH"),
|
|
||||||
# certfile: System.get_env("SOME_APP_SSL_CERT_PATH")
|
|
||||||
# ]
|
|
||||||
#
|
|
||||||
# The `cipher_suite` is set to `:strong` to support only the
|
|
||||||
# latest and more secure SSL ciphers. This means old browsers
|
|
||||||
# and clients may not be supported. You can set it to
|
|
||||||
# `:compatible` for wider support.
|
|
||||||
#
|
|
||||||
# `:keyfile` and `:certfile` expect an absolute path to the key
|
|
||||||
# and cert in disk or a relative path inside priv, for example
|
|
||||||
# "priv/ssl/server.key". For all supported SSL configuration
|
|
||||||
# options, see https://hexdocs.pm/plug/Plug.SSL.html#configure/1
|
|
||||||
#
|
|
||||||
# We also recommend setting `force_ssl` in your config/prod.exs,
|
|
||||||
# ensuring no data is ever sent via http, always redirecting to https:
|
|
||||||
#
|
|
||||||
# config :hello, HelloWeb.Endpoint,
|
|
||||||
# force_ssl: [hsts: true]
|
|
||||||
#
|
|
||||||
# Check `Plug.SSL` for all available options in `force_ssl`.
|
|
||||||
|
|
||||||
# ## Configuring the mailer
|
|
||||||
#
|
|
||||||
# In production you need to configure the mailer to use a different adapter.
|
|
||||||
# Also, you may need to configure the Swoosh API client of your choice if you
|
|
||||||
# are not using SMTP. Here is an example of the configuration:
|
|
||||||
#
|
|
||||||
# config :hello, Hello.Mailer,
|
|
||||||
# adapter: Swoosh.Adapters.Mailgun,
|
|
||||||
# api_key: System.get_env("MAILGUN_API_KEY"),
|
|
||||||
# domain: System.get_env("MAILGUN_DOMAIN")
|
|
||||||
#
|
|
||||||
# For this example you need include a HTTP client required by Swoosh API client.
|
|
||||||
# Swoosh supports Hackney and Finch out of the box:
|
|
||||||
#
|
|
||||||
# config :swoosh, :api_client, Swoosh.ApiClient.Hackney
|
|
||||||
#
|
|
||||||
# See https://hexdocs.pm/swoosh/Swoosh.html#module-installation for details.
|
|
||||||
end
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# Configure your database
|
|
||||||
#
|
|
||||||
# The MIX_TEST_PARTITION environment variable can be used
|
|
||||||
# to provide built-in test partitioning in CI environment.
|
|
||||||
# Run `mix help test` for more information.
|
|
||||||
config :hello, Hello.Repo,
|
|
||||||
username: "postgres",
|
|
||||||
password: "postgres",
|
|
||||||
hostname: "localhost",
|
|
||||||
database: "hello_test#{System.get_env("MIX_TEST_PARTITION")}",
|
|
||||||
pool: Ecto.Adapters.SQL.Sandbox,
|
|
||||||
pool_size: System.schedulers_online() * 2
|
|
||||||
|
|
||||||
# We don't run a server during test. If one is required,
|
|
||||||
# you can enable the server option below.
|
|
||||||
config :hello, HelloWeb.Endpoint,
|
|
||||||
http: [ip: {127, 0, 0, 1}, port: 4002],
|
|
||||||
secret_key_base: "97cbBfJQ4RNSN9Lg3UpGrsgrj8z0a7fn2LIgnqxgdFYdv1PfUDwkbFLDZvW5g82T",
|
|
||||||
server: false
|
|
||||||
|
|
||||||
# In test we don't send emails
|
|
||||||
config :hello, Hello.Mailer, adapter: Swoosh.Adapters.Test
|
|
||||||
|
|
||||||
# Disable swoosh api client as it is only required for production adapters
|
|
||||||
config :swoosh, :api_client, false
|
|
||||||
|
|
||||||
# Print only warnings and errors during test
|
|
||||||
config :logger, level: :warning
|
|
||||||
|
|
||||||
# Initialize plugs at runtime for faster test compilation
|
|
||||||
config :phoenix, :plug_init_mode, :runtime
|
|
||||||
|
|
||||||
# Enable helpful, but potentially expensive runtime checks
|
|
||||||
config :phoenix_live_view,
|
|
||||||
enable_expensive_runtime_checks: true
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
defmodule Hello do
|
|
||||||
@moduledoc """
|
|
||||||
Hello keeps the contexts that define your domain
|
|
||||||
and business logic.
|
|
||||||
|
|
||||||
Contexts are also responsible for managing your data, regardless
|
|
||||||
if it comes from the database, an external API or others.
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
defmodule Hello.Application do
|
|
||||||
# See https://hexdocs.pm/elixir/Application.html
|
|
||||||
# for more information on OTP Applications
|
|
||||||
@moduledoc false
|
|
||||||
|
|
||||||
use Application
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def start(_type, _args) do
|
|
||||||
children = [
|
|
||||||
HelloWeb.Telemetry,
|
|
||||||
Hello.Repo,
|
|
||||||
{DNSCluster, query: Application.get_env(:hello, :dns_cluster_query) || :ignore},
|
|
||||||
{Phoenix.PubSub, name: Hello.PubSub},
|
|
||||||
# Start the Finch HTTP client for sending emails
|
|
||||||
{Finch, name: Hello.Finch},
|
|
||||||
# Start a worker by calling: Hello.Worker.start_link(arg)
|
|
||||||
# {Hello.Worker, arg},
|
|
||||||
# Start to serve requests, typically the last entry
|
|
||||||
HelloWeb.Endpoint
|
|
||||||
]
|
|
||||||
|
|
||||||
# See https://hexdocs.pm/elixir/Supervisor.html
|
|
||||||
# for other strategies and supported options
|
|
||||||
opts = [strategy: :one_for_one, name: Hello.Supervisor]
|
|
||||||
Supervisor.start_link(children, opts)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Tell Phoenix to update the endpoint configuration
|
|
||||||
# whenever the application is updated.
|
|
||||||
@impl true
|
|
||||||
def config_change(changed, _new, removed) do
|
|
||||||
HelloWeb.Endpoint.config_change(changed, removed)
|
|
||||||
:ok
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
defmodule Hello.Mailer do
|
|
||||||
use Swoosh.Mailer, otp_app: :hello
|
|
||||||
end
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
defmodule Hello.Repo do
|
|
||||||
use Ecto.Repo,
|
|
||||||
otp_app: :hello,
|
|
||||||
adapter: Ecto.Adapters.Postgres
|
|
||||||
end
|
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
defmodule HelloWeb do
|
|
||||||
@moduledoc """
|
|
||||||
The entrypoint for defining your web interface, such
|
|
||||||
as controllers, components, channels, and so on.
|
|
||||||
|
|
||||||
This can be used in your application as:
|
|
||||||
|
|
||||||
use HelloWeb, :controller
|
|
||||||
use HelloWeb, :html
|
|
||||||
|
|
||||||
The definitions below will be executed for every controller,
|
|
||||||
component, etc, so keep them short and clean, focused
|
|
||||||
on imports, uses and aliases.
|
|
||||||
|
|
||||||
Do NOT define functions inside the quoted expressions
|
|
||||||
below. Instead, define additional modules and import
|
|
||||||
those modules here.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def static_paths, do: ~w(assets fonts images favicon.ico robots.txt)
|
|
||||||
|
|
||||||
def router do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Router, helpers: false
|
|
||||||
|
|
||||||
# Import common connection and controller functions to use in pipelines
|
|
||||||
import Plug.Conn
|
|
||||||
import Phoenix.Controller
|
|
||||||
import Phoenix.LiveView.Router
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def channel do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Channel
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def controller do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Controller,
|
|
||||||
formats: [:html, :json],
|
|
||||||
layouts: [html: HelloWeb.Layouts]
|
|
||||||
|
|
||||||
import Plug.Conn
|
|
||||||
import HelloWeb.Gettext
|
|
||||||
|
|
||||||
unquote(verified_routes())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def live_view do
|
|
||||||
quote do
|
|
||||||
use Phoenix.LiveView,
|
|
||||||
layout: {HelloWeb.Layouts, :app}
|
|
||||||
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def live_component do
|
|
||||||
quote do
|
|
||||||
use Phoenix.LiveComponent
|
|
||||||
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def html do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Component
|
|
||||||
|
|
||||||
# Import convenience functions from controllers
|
|
||||||
import Phoenix.Controller,
|
|
||||||
only: [get_csrf_token: 0, view_module: 1, view_template: 1]
|
|
||||||
|
|
||||||
# Include general helpers for rendering HTML
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp html_helpers do
|
|
||||||
quote do
|
|
||||||
# HTML escaping functionality
|
|
||||||
import Phoenix.HTML
|
|
||||||
# Core UI components and translation
|
|
||||||
import HelloWeb.CoreComponents
|
|
||||||
import HelloWeb.Gettext
|
|
||||||
|
|
||||||
# Shortcut for generating JS commands
|
|
||||||
alias Phoenix.LiveView.JS
|
|
||||||
|
|
||||||
# Routes generation with the ~p sigil
|
|
||||||
unquote(verified_routes())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def verified_routes do
|
|
||||||
quote do
|
|
||||||
use Phoenix.VerifiedRoutes,
|
|
||||||
endpoint: HelloWeb.Endpoint,
|
|
||||||
router: HelloWeb.Router,
|
|
||||||
statics: HelloWeb.static_paths()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
When used, dispatch to the appropriate controller/live_view/etc.
|
|
||||||
"""
|
|
||||||
defmacro __using__(which) when is_atom(which) do
|
|
||||||
apply(__MODULE__, which, [])
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,676 +0,0 @@
|
|||||||
defmodule HelloWeb.CoreComponents do
|
|
||||||
@moduledoc """
|
|
||||||
Provides core UI components.
|
|
||||||
|
|
||||||
At first glance, this module may seem daunting, but its goal is to provide
|
|
||||||
core building blocks for your application, such as modals, tables, and
|
|
||||||
forms. The components consist mostly of markup and are well-documented
|
|
||||||
with doc strings and declarative assigns. You may customize and style
|
|
||||||
them in any way you want, based on your application growth and needs.
|
|
||||||
|
|
||||||
The default components use Tailwind CSS, a utility-first CSS framework.
|
|
||||||
See the [Tailwind CSS documentation](https://tailwindcss.com) to learn
|
|
||||||
how to customize them or feel free to swap in another framework altogether.
|
|
||||||
|
|
||||||
Icons are provided by [heroicons](https://heroicons.com). See `icon/1` for usage.
|
|
||||||
"""
|
|
||||||
use Phoenix.Component
|
|
||||||
|
|
||||||
alias Phoenix.LiveView.JS
|
|
||||||
import HelloWeb.Gettext
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a modal.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.modal id="confirm-modal">
|
|
||||||
This is a modal.
|
|
||||||
</.modal>
|
|
||||||
|
|
||||||
JS commands may be passed to the `:on_cancel` to configure
|
|
||||||
the closing/cancel event, for example:
|
|
||||||
|
|
||||||
<.modal id="confirm" on_cancel={JS.navigate(~p"/posts")}>
|
|
||||||
This is another modal.
|
|
||||||
</.modal>
|
|
||||||
|
|
||||||
"""
|
|
||||||
attr :id, :string, required: true
|
|
||||||
attr :show, :boolean, default: false
|
|
||||||
attr :on_cancel, JS, default: %JS{}
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def modal(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div
|
|
||||||
id={@id}
|
|
||||||
phx-mounted={@show && show_modal(@id)}
|
|
||||||
phx-remove={hide_modal(@id)}
|
|
||||||
data-cancel={JS.exec(@on_cancel, "phx-remove")}
|
|
||||||
class="relative z-50 hidden"
|
|
||||||
>
|
|
||||||
<div id={"#{@id}-bg"} class="bg-zinc-50/90 fixed inset-0 transition-opacity" aria-hidden="true" />
|
|
||||||
<div
|
|
||||||
class="fixed inset-0 overflow-y-auto"
|
|
||||||
aria-labelledby={"#{@id}-title"}
|
|
||||||
aria-describedby={"#{@id}-description"}
|
|
||||||
role="dialog"
|
|
||||||
aria-modal="true"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<div class="flex min-h-full items-center justify-center">
|
|
||||||
<div class="w-full max-w-3xl p-4 sm:p-6 lg:py-8">
|
|
||||||
<.focus_wrap
|
|
||||||
id={"#{@id}-container"}
|
|
||||||
phx-window-keydown={JS.exec("data-cancel", to: "##{@id}")}
|
|
||||||
phx-key="escape"
|
|
||||||
phx-click-away={JS.exec("data-cancel", to: "##{@id}")}
|
|
||||||
class="shadow-zinc-700/10 ring-zinc-700/10 relative hidden rounded-2xl bg-white p-14 shadow-lg ring-1 transition"
|
|
||||||
>
|
|
||||||
<div class="absolute top-6 right-5">
|
|
||||||
<button
|
|
||||||
phx-click={JS.exec("data-cancel", to: "##{@id}")}
|
|
||||||
type="button"
|
|
||||||
class="-m-3 flex-none p-3 opacity-20 hover:opacity-40"
|
|
||||||
aria-label={gettext("close")}
|
|
||||||
>
|
|
||||||
<.icon name="hero-x-mark-solid" class="h-5 w-5" />
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div id={"#{@id}-content"}>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</div>
|
|
||||||
</.focus_wrap>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders flash notices.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.flash kind={:info} flash={@flash} />
|
|
||||||
<.flash kind={:info} phx-mounted={show("#flash")}>Welcome Back!</.flash>
|
|
||||||
"""
|
|
||||||
attr :id, :string, doc: "the optional id of flash container"
|
|
||||||
attr :flash, :map, default: %{}, doc: "the map of flash messages to display"
|
|
||||||
attr :title, :string, default: nil
|
|
||||||
attr :kind, :atom, values: [:info, :error], doc: "used for styling and flash lookup"
|
|
||||||
attr :rest, :global, doc: "the arbitrary HTML attributes to add to the flash container"
|
|
||||||
|
|
||||||
slot :inner_block, doc: "the optional inner block that renders the flash message"
|
|
||||||
|
|
||||||
def flash(assigns) do
|
|
||||||
assigns = assign_new(assigns, :id, fn -> "flash-#{assigns.kind}" end)
|
|
||||||
|
|
||||||
~H"""
|
|
||||||
<div
|
|
||||||
:if={msg = render_slot(@inner_block) || Phoenix.Flash.get(@flash, @kind)}
|
|
||||||
id={@id}
|
|
||||||
phx-click={JS.push("lv:clear-flash", value: %{key: @kind}) |> hide("##{@id}")}
|
|
||||||
role="alert"
|
|
||||||
class={[
|
|
||||||
"fixed top-2 right-2 mr-2 w-80 sm:w-96 z-50 rounded-lg p-3 ring-1",
|
|
||||||
@kind == :info && "bg-emerald-50 text-emerald-800 ring-emerald-500 fill-cyan-900",
|
|
||||||
@kind == :error && "bg-rose-50 text-rose-900 shadow-md ring-rose-500 fill-rose-900"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<p :if={@title} class="flex items-center gap-1.5 text-sm font-semibold leading-6">
|
|
||||||
<.icon :if={@kind == :info} name="hero-information-circle-mini" class="h-4 w-4" />
|
|
||||||
<.icon :if={@kind == :error} name="hero-exclamation-circle-mini" class="h-4 w-4" />
|
|
||||||
<%= @title %>
|
|
||||||
</p>
|
|
||||||
<p class="mt-2 text-sm leading-5"><%= msg %></p>
|
|
||||||
<button type="button" class="group absolute top-1 right-1 p-2" aria-label={gettext("close")}>
|
|
||||||
<.icon name="hero-x-mark-solid" class="h-5 w-5 opacity-40 group-hover:opacity-70" />
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Shows the flash group with standard titles and content.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.flash_group flash={@flash} />
|
|
||||||
"""
|
|
||||||
attr :flash, :map, required: true, doc: "the map of flash messages"
|
|
||||||
attr :id, :string, default: "flash-group", doc: "the optional id of flash container"
|
|
||||||
|
|
||||||
def flash_group(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div id={@id}>
|
|
||||||
<.flash kind={:info} title={gettext("Success!")} flash={@flash} />
|
|
||||||
<.flash kind={:error} title={gettext("Error!")} flash={@flash} />
|
|
||||||
<.flash
|
|
||||||
id="client-error"
|
|
||||||
kind={:error}
|
|
||||||
title={gettext("We can't find the internet")}
|
|
||||||
phx-disconnected={show(".phx-client-error #client-error")}
|
|
||||||
phx-connected={hide("#client-error")}
|
|
||||||
hidden
|
|
||||||
>
|
|
||||||
<%= gettext("Attempting to reconnect") %>
|
|
||||||
<.icon name="hero-arrow-path" class="ml-1 h-3 w-3 animate-spin" />
|
|
||||||
</.flash>
|
|
||||||
|
|
||||||
<.flash
|
|
||||||
id="server-error"
|
|
||||||
kind={:error}
|
|
||||||
title={gettext("Something went wrong!")}
|
|
||||||
phx-disconnected={show(".phx-server-error #server-error")}
|
|
||||||
phx-connected={hide("#server-error")}
|
|
||||||
hidden
|
|
||||||
>
|
|
||||||
<%= gettext("Hang in there while we get back on track") %>
|
|
||||||
<.icon name="hero-arrow-path" class="ml-1 h-3 w-3 animate-spin" />
|
|
||||||
</.flash>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a simple form.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.simple_form for={@form} phx-change="validate" phx-submit="save">
|
|
||||||
<.input field={@form[:email]} label="Email"/>
|
|
||||||
<.input field={@form[:username]} label="Username" />
|
|
||||||
<:actions>
|
|
||||||
<.button>Save</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
"""
|
|
||||||
attr :for, :any, required: true, doc: "the data structure for the form"
|
|
||||||
attr :as, :any, default: nil, doc: "the server side parameter to collect all input under"
|
|
||||||
|
|
||||||
attr :rest, :global,
|
|
||||||
include: ~w(autocomplete name rel action enctype method novalidate target multipart),
|
|
||||||
doc: "the arbitrary HTML attributes to apply to the form tag"
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
slot :actions, doc: "the slot for form actions, such as a submit button"
|
|
||||||
|
|
||||||
def simple_form(assigns) do
|
|
||||||
~H"""
|
|
||||||
<.form :let={f} for={@for} as={@as} {@rest}>
|
|
||||||
<div class="mt-10 space-y-8 bg-white">
|
|
||||||
<%= render_slot(@inner_block, f) %>
|
|
||||||
<div :for={action <- @actions} class="mt-2 flex items-center justify-between gap-6">
|
|
||||||
<%= render_slot(action, f) %>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</.form>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a button.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.button>Send!</.button>
|
|
||||||
<.button phx-click="go" class="ml-2">Send!</.button>
|
|
||||||
"""
|
|
||||||
attr :type, :string, default: nil
|
|
||||||
attr :class, :string, default: nil
|
|
||||||
attr :rest, :global, include: ~w(disabled form name value)
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def button(assigns) do
|
|
||||||
~H"""
|
|
||||||
<button
|
|
||||||
type={@type}
|
|
||||||
class={[
|
|
||||||
"phx-submit-loading:opacity-75 rounded-lg bg-zinc-900 hover:bg-zinc-700 py-2 px-3",
|
|
||||||
"text-sm font-semibold leading-6 text-white active:text-white/80",
|
|
||||||
@class
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</button>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders an input with label and error messages.
|
|
||||||
|
|
||||||
A `Phoenix.HTML.FormField` may be passed as argument,
|
|
||||||
which is used to retrieve the input name, id, and values.
|
|
||||||
Otherwise all attributes may be passed explicitly.
|
|
||||||
|
|
||||||
## Types
|
|
||||||
|
|
||||||
This function accepts all HTML input types, considering that:
|
|
||||||
|
|
||||||
* You may also set `type="select"` to render a `<select>` tag
|
|
||||||
|
|
||||||
* `type="checkbox"` is used exclusively to render boolean values
|
|
||||||
|
|
||||||
* For live file uploads, see `Phoenix.Component.live_file_input/1`
|
|
||||||
|
|
||||||
See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input
|
|
||||||
for more information. Unsupported types, such as hidden and radio,
|
|
||||||
are best written directly in your templates.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.input field={@form[:email]} type="email" />
|
|
||||||
<.input name="my-input" errors={["oh no!"]} />
|
|
||||||
"""
|
|
||||||
attr :id, :any, default: nil
|
|
||||||
attr :name, :any
|
|
||||||
attr :label, :string, default: nil
|
|
||||||
attr :value, :any
|
|
||||||
|
|
||||||
attr :type, :string,
|
|
||||||
default: "text",
|
|
||||||
values: ~w(checkbox color date datetime-local email file month number password
|
|
||||||
range search select tel text textarea time url week)
|
|
||||||
|
|
||||||
attr :field, Phoenix.HTML.FormField,
|
|
||||||
doc: "a form field struct retrieved from the form, for example: @form[:email]"
|
|
||||||
|
|
||||||
attr :errors, :list, default: []
|
|
||||||
attr :checked, :boolean, doc: "the checked flag for checkbox inputs"
|
|
||||||
attr :prompt, :string, default: nil, doc: "the prompt for select inputs"
|
|
||||||
attr :options, :list, doc: "the options to pass to Phoenix.HTML.Form.options_for_select/2"
|
|
||||||
attr :multiple, :boolean, default: false, doc: "the multiple flag for select inputs"
|
|
||||||
|
|
||||||
attr :rest, :global,
|
|
||||||
include: ~w(accept autocomplete capture cols disabled form list max maxlength min minlength
|
|
||||||
multiple pattern placeholder readonly required rows size step)
|
|
||||||
|
|
||||||
def input(%{field: %Phoenix.HTML.FormField{} = field} = assigns) do
|
|
||||||
errors = if Phoenix.Component.used_input?(field), do: field.errors, else: []
|
|
||||||
|
|
||||||
assigns
|
|
||||||
|> assign(field: nil, id: assigns.id || field.id)
|
|
||||||
|> assign(:errors, Enum.map(errors, &translate_error(&1)))
|
|
||||||
|> assign_new(:name, fn -> if assigns.multiple, do: field.name <> "[]", else: field.name end)
|
|
||||||
|> assign_new(:value, fn -> field.value end)
|
|
||||||
|> input()
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "checkbox"} = assigns) do
|
|
||||||
assigns =
|
|
||||||
assign_new(assigns, :checked, fn ->
|
|
||||||
Phoenix.HTML.Form.normalize_value("checkbox", assigns[:value])
|
|
||||||
end)
|
|
||||||
|
|
||||||
~H"""
|
|
||||||
<div>
|
|
||||||
<label class="flex items-center gap-4 text-sm leading-6 text-zinc-600">
|
|
||||||
<input type="hidden" name={@name} value="false" disabled={@rest[:disabled]} />
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
id={@id}
|
|
||||||
name={@name}
|
|
||||||
value="true"
|
|
||||||
checked={@checked}
|
|
||||||
class="rounded border-zinc-300 text-zinc-900 focus:ring-0"
|
|
||||||
{@rest}
|
|
||||||
/>
|
|
||||||
<%= @label %>
|
|
||||||
</label>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "select"} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<div>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<select
|
|
||||||
id={@id}
|
|
||||||
name={@name}
|
|
||||||
class="mt-2 block w-full rounded-md border border-gray-300 bg-white shadow-sm focus:border-zinc-400 focus:ring-0 sm:text-sm"
|
|
||||||
multiple={@multiple}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<option :if={@prompt} value=""><%= @prompt %></option>
|
|
||||||
<%= Phoenix.HTML.Form.options_for_select(@options, @value) %>
|
|
||||||
</select>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "textarea"} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<div>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<textarea
|
|
||||||
id={@id}
|
|
||||||
name={@name}
|
|
||||||
class={[
|
|
||||||
"mt-2 block w-full rounded-lg text-zinc-900 focus:ring-0 sm:text-sm sm:leading-6 min-h-[6rem]",
|
|
||||||
@errors == [] && "border-zinc-300 focus:border-zinc-400",
|
|
||||||
@errors != [] && "border-rose-400 focus:border-rose-400"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
><%= Phoenix.HTML.Form.normalize_value("textarea", @value) %></textarea>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
# All other inputs text, datetime-local, url, password, etc. are handled here...
|
|
||||||
def input(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<input
|
|
||||||
type={@type}
|
|
||||||
name={@name}
|
|
||||||
id={@id}
|
|
||||||
value={Phoenix.HTML.Form.normalize_value(@type, @value)}
|
|
||||||
class={[
|
|
||||||
"mt-2 block w-full rounded-lg text-zinc-900 focus:ring-0 sm:text-sm sm:leading-6",
|
|
||||||
@errors == [] && "border-zinc-300 focus:border-zinc-400",
|
|
||||||
@errors != [] && "border-rose-400 focus:border-rose-400"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
/>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a label.
|
|
||||||
"""
|
|
||||||
attr :for, :string, default: nil
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def label(assigns) do
|
|
||||||
~H"""
|
|
||||||
<label for={@for} class="block text-sm font-semibold leading-6 text-zinc-800">
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</label>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Generates a generic error message.
|
|
||||||
"""
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def error(assigns) do
|
|
||||||
~H"""
|
|
||||||
<p class="mt-3 flex gap-3 text-sm leading-6 text-rose-600">
|
|
||||||
<.icon name="hero-exclamation-circle-mini" class="mt-0.5 h-5 w-5 flex-none" />
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</p>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a header with title.
|
|
||||||
"""
|
|
||||||
attr :class, :string, default: nil
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
slot :subtitle
|
|
||||||
slot :actions
|
|
||||||
|
|
||||||
def header(assigns) do
|
|
||||||
~H"""
|
|
||||||
<header class={[@actions != [] && "flex items-center justify-between gap-6", @class]}>
|
|
||||||
<div>
|
|
||||||
<h1 class="text-lg font-semibold leading-8 text-zinc-800">
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</h1>
|
|
||||||
<p :if={@subtitle != []} class="mt-2 text-sm leading-6 text-zinc-600">
|
|
||||||
<%= render_slot(@subtitle) %>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="flex-none"><%= render_slot(@actions) %></div>
|
|
||||||
</header>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc ~S"""
|
|
||||||
Renders a table with generic styling.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.table id="users" rows={@users}>
|
|
||||||
<:col :let={user} label="id"><%= user.id %></:col>
|
|
||||||
<:col :let={user} label="username"><%= user.username %></:col>
|
|
||||||
</.table>
|
|
||||||
"""
|
|
||||||
attr :id, :string, required: true
|
|
||||||
attr :rows, :list, required: true
|
|
||||||
attr :row_id, :any, default: nil, doc: "the function for generating the row id"
|
|
||||||
attr :row_click, :any, default: nil, doc: "the function for handling phx-click on each row"
|
|
||||||
|
|
||||||
attr :row_item, :any,
|
|
||||||
default: &Function.identity/1,
|
|
||||||
doc: "the function for mapping each row before calling the :col and :action slots"
|
|
||||||
|
|
||||||
slot :col, required: true do
|
|
||||||
attr :label, :string
|
|
||||||
end
|
|
||||||
|
|
||||||
slot :action, doc: "the slot for showing user actions in the last table column"
|
|
||||||
|
|
||||||
def table(assigns) do
|
|
||||||
assigns =
|
|
||||||
with %{rows: %Phoenix.LiveView.LiveStream{}} <- assigns do
|
|
||||||
assign(assigns, row_id: assigns.row_id || fn {id, _item} -> id end)
|
|
||||||
end
|
|
||||||
|
|
||||||
~H"""
|
|
||||||
<div class="overflow-y-auto px-4 sm:overflow-visible sm:px-0">
|
|
||||||
<table class="w-[40rem] mt-11 sm:w-full">
|
|
||||||
<thead class="text-sm text-left leading-6 text-zinc-500">
|
|
||||||
<tr>
|
|
||||||
<th :for={col <- @col} class="p-0 pb-4 pr-6 font-normal"><%= col[:label] %></th>
|
|
||||||
<th :if={@action != []} class="relative p-0 pb-4">
|
|
||||||
<span class="sr-only"><%= gettext("Actions") %></span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody
|
|
||||||
id={@id}
|
|
||||||
phx-update={match?(%Phoenix.LiveView.LiveStream{}, @rows) && "stream"}
|
|
||||||
class="relative divide-y divide-zinc-100 border-t border-zinc-200 text-sm leading-6 text-zinc-700"
|
|
||||||
>
|
|
||||||
<tr :for={row <- @rows} id={@row_id && @row_id.(row)} class="group hover:bg-zinc-50">
|
|
||||||
<td
|
|
||||||
:for={{col, i} <- Enum.with_index(@col)}
|
|
||||||
phx-click={@row_click && @row_click.(row)}
|
|
||||||
class={["relative p-0", @row_click && "hover:cursor-pointer"]}
|
|
||||||
>
|
|
||||||
<div class="block py-4 pr-6">
|
|
||||||
<span class="absolute -inset-y-px right-0 -left-4 group-hover:bg-zinc-50 sm:rounded-l-xl" />
|
|
||||||
<span class={["relative", i == 0 && "font-semibold text-zinc-900"]}>
|
|
||||||
<%= render_slot(col, @row_item.(row)) %>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td :if={@action != []} class="relative w-14 p-0">
|
|
||||||
<div class="relative whitespace-nowrap py-4 text-right text-sm font-medium">
|
|
||||||
<span class="absolute -inset-y-px -right-4 left-0 group-hover:bg-zinc-50 sm:rounded-r-xl" />
|
|
||||||
<span
|
|
||||||
:for={action <- @action}
|
|
||||||
class="relative ml-4 font-semibold leading-6 text-zinc-900 hover:text-zinc-700"
|
|
||||||
>
|
|
||||||
<%= render_slot(action, @row_item.(row)) %>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a data list.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.list>
|
|
||||||
<:item title="Title"><%= @post.title %></:item>
|
|
||||||
<:item title="Views"><%= @post.views %></:item>
|
|
||||||
</.list>
|
|
||||||
"""
|
|
||||||
slot :item, required: true do
|
|
||||||
attr :title, :string, required: true
|
|
||||||
end
|
|
||||||
|
|
||||||
def list(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mt-14">
|
|
||||||
<dl class="-my-4 divide-y divide-zinc-100">
|
|
||||||
<div :for={item <- @item} class="flex gap-4 py-4 text-sm leading-6 sm:gap-8">
|
|
||||||
<dt class="w-1/4 flex-none text-zinc-500"><%= item.title %></dt>
|
|
||||||
<dd class="text-zinc-700"><%= render_slot(item) %></dd>
|
|
||||||
</div>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a back navigation link.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.back navigate={~p"/posts"}>Back to posts</.back>
|
|
||||||
"""
|
|
||||||
attr :navigate, :any, required: true
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def back(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mt-16">
|
|
||||||
<.link
|
|
||||||
navigate={@navigate}
|
|
||||||
class="text-sm font-semibold leading-6 text-zinc-900 hover:text-zinc-700"
|
|
||||||
>
|
|
||||||
<.icon name="hero-arrow-left-solid" class="h-3 w-3" />
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</.link>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a [Heroicon](https://heroicons.com).
|
|
||||||
|
|
||||||
Heroicons come in three styles – outline, solid, and mini.
|
|
||||||
By default, the outline style is used, but solid and mini may
|
|
||||||
be applied by using the `-solid` and `-mini` suffix.
|
|
||||||
|
|
||||||
You can customize the size and colors of the icons by setting
|
|
||||||
width, height, and background color classes.
|
|
||||||
|
|
||||||
Icons are extracted from the `deps/heroicons` directory and bundled within
|
|
||||||
your compiled app.css by the plugin in your `assets/tailwind.config.js`.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.icon name="hero-x-mark-solid" />
|
|
||||||
<.icon name="hero-arrow-path" class="ml-1 w-3 h-3 animate-spin" />
|
|
||||||
"""
|
|
||||||
attr :name, :string, required: true
|
|
||||||
attr :class, :string, default: nil
|
|
||||||
|
|
||||||
def icon(%{name: "hero-" <> _} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<span class={[@name, @class]} />
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
## JS Commands
|
|
||||||
|
|
||||||
def show(js \\ %JS{}, selector) do
|
|
||||||
JS.show(js,
|
|
||||||
to: selector,
|
|
||||||
time: 300,
|
|
||||||
transition:
|
|
||||||
{"transition-all transform ease-out duration-300",
|
|
||||||
"opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95",
|
|
||||||
"opacity-100 translate-y-0 sm:scale-100"}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def hide(js \\ %JS{}, selector) do
|
|
||||||
JS.hide(js,
|
|
||||||
to: selector,
|
|
||||||
time: 200,
|
|
||||||
transition:
|
|
||||||
{"transition-all transform ease-in duration-200",
|
|
||||||
"opacity-100 translate-y-0 sm:scale-100",
|
|
||||||
"opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def show_modal(js \\ %JS{}, id) when is_binary(id) do
|
|
||||||
js
|
|
||||||
|> JS.show(to: "##{id}")
|
|
||||||
|> JS.show(
|
|
||||||
to: "##{id}-bg",
|
|
||||||
time: 300,
|
|
||||||
transition: {"transition-all transform ease-out duration-300", "opacity-0", "opacity-100"}
|
|
||||||
)
|
|
||||||
|> show("##{id}-container")
|
|
||||||
|> JS.add_class("overflow-hidden", to: "body")
|
|
||||||
|> JS.focus_first(to: "##{id}-content")
|
|
||||||
end
|
|
||||||
|
|
||||||
def hide_modal(js \\ %JS{}, id) do
|
|
||||||
js
|
|
||||||
|> JS.hide(
|
|
||||||
to: "##{id}-bg",
|
|
||||||
transition: {"transition-all transform ease-in duration-200", "opacity-100", "opacity-0"}
|
|
||||||
)
|
|
||||||
|> hide("##{id}-container")
|
|
||||||
|> JS.hide(to: "##{id}", transition: {"block", "block", "hidden"})
|
|
||||||
|> JS.remove_class("overflow-hidden", to: "body")
|
|
||||||
|> JS.pop_focus()
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Translates an error message using gettext.
|
|
||||||
"""
|
|
||||||
def translate_error({msg, opts}) do
|
|
||||||
# When using gettext, we typically pass the strings we want
|
|
||||||
# to translate as a static argument:
|
|
||||||
#
|
|
||||||
# # Translate the number of files with plural rules
|
|
||||||
# dngettext("errors", "1 file", "%{count} files", count)
|
|
||||||
#
|
|
||||||
# However the error messages in our forms and APIs are generated
|
|
||||||
# dynamically, so we need to translate them by calling Gettext
|
|
||||||
# with our gettext backend as first argument. Translations are
|
|
||||||
# available in the errors.po file (as we use the "errors" domain).
|
|
||||||
if count = opts[:count] do
|
|
||||||
Gettext.dngettext(HelloWeb.Gettext, "errors", msg, msg, count, opts)
|
|
||||||
else
|
|
||||||
Gettext.dgettext(HelloWeb.Gettext, "errors", msg, opts)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Translates the errors for a field from a keyword list of errors.
|
|
||||||
"""
|
|
||||||
def translate_errors(errors, field) when is_list(errors) do
|
|
||||||
for {^field, {msg, opts}} <- errors, do: translate_error({msg, opts})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
defmodule HelloWeb.Layouts do
|
|
||||||
@moduledoc """
|
|
||||||
This module holds different layouts used by your application.
|
|
||||||
|
|
||||||
See the `layouts` directory for all templates available.
|
|
||||||
The "root" layout is a skeleton rendered as part of the
|
|
||||||
application router. The "app" layout is set as the default
|
|
||||||
layout on both `use HelloWeb, :controller` and
|
|
||||||
`use HelloWeb, :live_view`.
|
|
||||||
"""
|
|
||||||
use HelloWeb, :html
|
|
||||||
|
|
||||||
embed_templates "layouts/*"
|
|
||||||
end
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
<header class="px-4 sm:px-6 lg:px-8">
|
|
||||||
<div class="flex items-center justify-between border-b border-zinc-100 py-3 text-sm">
|
|
||||||
<div class="flex items-center gap-4">
|
|
||||||
<a href="/">
|
|
||||||
<img src={~p"/images/logo.svg"} width="36" />
|
|
||||||
</a>
|
|
||||||
<p class="bg-brand/5 text-brand rounded-full px-2 font-medium leading-6">
|
|
||||||
v<%= Application.spec(:phoenix, :vsn) %>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-center gap-4 font-semibold leading-6 text-zinc-900">
|
|
||||||
<a href="https://twitter.com/elixirphoenix" class="hover:text-zinc-700">
|
|
||||||
@elixirphoenix
|
|
||||||
</a>
|
|
||||||
<a href="https://github.com/phoenixframework/phoenix" class="hover:text-zinc-700">
|
|
||||||
GitHub
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
href="https://hexdocs.pm/phoenix/overview.html"
|
|
||||||
class="rounded-lg bg-zinc-100 px-2 py-1 hover:bg-zinc-200/80"
|
|
||||||
>
|
|
||||||
Get Started <span aria-hidden="true">→</span>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
<main class="px-4 py-20 sm:px-6 lg:px-8">
|
|
||||||
<div class="mx-auto max-w-2xl">
|
|
||||||
<.flash_group flash={@flash} />
|
|
||||||
<%= @inner_content %>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en" class="[scrollbar-gutter:stable]">
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<meta name="csrf-token" content={get_csrf_token()} />
|
|
||||||
<.live_title suffix=" · Phoenix Framework">
|
|
||||||
<%= assigns[:page_title] || "Hello" %>
|
|
||||||
</.live_title>
|
|
||||||
<link phx-track-static rel="stylesheet" href={~p"/assets/app.css"} />
|
|
||||||
<script defer phx-track-static type="text/javascript" src={~p"/assets/app.js"}>
|
|
||||||
</script>
|
|
||||||
</head>
|
|
||||||
<body class="bg-white">
|
|
||||||
<%= @inner_content %>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
defmodule HelloWeb.ErrorHTML do
|
|
||||||
@moduledoc """
|
|
||||||
This module is invoked by your endpoint in case of errors on HTML requests.
|
|
||||||
|
|
||||||
See config/config.exs.
|
|
||||||
"""
|
|
||||||
use HelloWeb, :html
|
|
||||||
|
|
||||||
# If you want to customize your error pages,
|
|
||||||
# uncomment the embed_templates/1 call below
|
|
||||||
# and add pages to the error directory:
|
|
||||||
#
|
|
||||||
# * lib/hello_web/controllers/error_html/404.html.heex
|
|
||||||
# * lib/hello_web/controllers/error_html/500.html.heex
|
|
||||||
#
|
|
||||||
# embed_templates "error_html/*"
|
|
||||||
|
|
||||||
# The default is to render a plain text page based on
|
|
||||||
# the template name. For example, "404.html" becomes
|
|
||||||
# "Not Found".
|
|
||||||
def render(template, _assigns) do
|
|
||||||
Phoenix.Controller.status_message_from_template(template)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
defmodule HelloWeb.ErrorJSON do
|
|
||||||
@moduledoc """
|
|
||||||
This module is invoked by your endpoint in case of errors on JSON requests.
|
|
||||||
|
|
||||||
See config/config.exs.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# If you want to customize a particular status code,
|
|
||||||
# you may add your own clauses, such as:
|
|
||||||
#
|
|
||||||
# def render("500.json", _assigns) do
|
|
||||||
# %{errors: %{detail: "Internal Server Error"}}
|
|
||||||
# end
|
|
||||||
|
|
||||||
# By default, Phoenix returns the status message from
|
|
||||||
# the template name. For example, "404.json" becomes
|
|
||||||
# "Not Found".
|
|
||||||
def render(template, _assigns) do
|
|
||||||
%{errors: %{detail: Phoenix.Controller.status_message_from_template(template)}}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
defmodule HelloWeb.PageController do
|
|
||||||
use HelloWeb, :controller
|
|
||||||
|
|
||||||
def home(conn, _params) do
|
|
||||||
# The home page is often custom made,
|
|
||||||
# so skip the default app layout.
|
|
||||||
render(conn, :home, layout: false)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
defmodule HelloWeb.PageHTML do
|
|
||||||
@moduledoc """
|
|
||||||
This module contains pages rendered by PageController.
|
|
||||||
|
|
||||||
See the `page_html` directory for all templates available.
|
|
||||||
"""
|
|
||||||
use HelloWeb, :html
|
|
||||||
|
|
||||||
embed_templates "page_html/*"
|
|
||||||
end
|
|
||||||
@@ -1,222 +0,0 @@
|
|||||||
<.flash_group flash={@flash} />
|
|
||||||
<div class="left-[40rem] fixed inset-y-0 right-0 z-0 hidden lg:block xl:left-[50rem]">
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 1480 957"
|
|
||||||
fill="none"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="absolute inset-0 h-full w-full"
|
|
||||||
preserveAspectRatio="xMinYMid slice"
|
|
||||||
>
|
|
||||||
<path fill="#EE7868" d="M0 0h1480v957H0z" />
|
|
||||||
<path
|
|
||||||
d="M137.542 466.27c-582.851-48.41-988.806-82.127-1608.412 658.2l67.39 810 3083.15-256.51L1535.94-49.622l-98.36 8.183C1269.29 281.468 734.115 515.799 146.47 467.012l-8.928-.742Z"
|
|
||||||
fill="#FF9F92"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M371.028 528.664C-169.369 304.988-545.754 149.198-1361.45 665.565l-182.58 792.025 3014.73 694.98 389.42-1689.25-96.18-22.171C1505.28 697.438 924.153 757.586 379.305 532.09l-8.277-3.426Z"
|
|
||||||
fill="#FA8372"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M359.326 571.714C-104.765 215.795-428.003-32.102-1349.55 255.554l-282.3 1224.596 3047.04 722.01 312.24-1354.467C1411.25 1028.3 834.355 935.995 366.435 577.166l-7.109-5.452Z"
|
|
||||||
fill="#E96856"
|
|
||||||
fill-opacity=".6"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1593.87 1236.88c-352.15 92.63-885.498-145.85-1244.602-613.557l-5.455-7.105C-12.347 152.31-260.41-170.8-1225-131.458l-368.63 1599.048 3057.19 704.76 130.31-935.47Z"
|
|
||||||
fill="#C42652"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1411.91 1526.93c-363.79 15.71-834.312-330.6-1085.883-863.909l-3.822-8.102C72.704 125.95-101.074-242.476-1052.01-408.907l-699.85 1484.267 2837.75 1338.01 326.02-886.44Z"
|
|
||||||
fill="#A41C42"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1116.26 1863.69c-355.457-78.98-720.318-535.27-825.287-1115.521l-1.594-8.816C185.286 163.833 112.786-237.016-762.678-643.898L-1822.83 608.665 571.922 2635.55l544.338-771.86Z"
|
|
||||||
fill="#A41C42"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
<div class="px-4 py-10 sm:px-6 sm:py-28 lg:px-8 xl:px-28 xl:py-32">
|
|
||||||
<div class="mx-auto max-w-xl lg:mx-0">
|
|
||||||
<svg viewBox="0 0 71 48" class="h-12" aria-hidden="true">
|
|
||||||
<path
|
|
||||||
d="m26.371 33.477-.552-.1c-3.92-.729-6.397-3.1-7.57-6.829-.733-2.324.597-4.035 3.035-4.148 1.995-.092 3.362 1.055 4.57 2.39 1.557 1.72 2.984 3.558 4.514 5.305 2.202 2.515 4.797 4.134 8.347 3.634 3.183-.448 5.958-1.725 8.371-3.828.363-.316.761-.592 1.144-.886l-.241-.284c-2.027.63-4.093.841-6.205.735-3.195-.16-6.24-.828-8.964-2.582-2.486-1.601-4.319-3.746-5.19-6.611-.704-2.315.736-3.934 3.135-3.6.948.133 1.746.56 2.463 1.165.583.493 1.143 1.015 1.738 1.493 2.8 2.25 6.712 2.375 10.265-.068-5.842-.026-9.817-3.24-13.308-7.313-1.366-1.594-2.7-3.216-4.095-4.785-2.698-3.036-5.692-5.71-9.79-6.623C12.8-.623 7.745.14 2.893 2.361 1.926 2.804.997 3.319 0 4.149c.494 0 .763.006 1.032 0 2.446-.064 4.28 1.023 5.602 3.024.962 1.457 1.415 3.104 1.761 4.798.513 2.515.247 5.078.544 7.605.761 6.494 4.08 11.026 10.26 13.346 2.267.852 4.591 1.135 7.172.555ZM10.751 3.852c-.976.246-1.756-.148-2.56-.962 1.377-.343 2.592-.476 3.897-.528-.107.848-.607 1.306-1.336 1.49Zm32.002 37.924c-.085-.626-.62-.901-1.04-1.228-1.857-1.446-4.03-1.958-6.333-2-1.375-.026-2.735-.128-4.031-.61-.595-.22-1.26-.505-1.244-1.272.015-.78.693-1 1.31-1.184.505-.15 1.026-.247 1.6-.382-1.46-.936-2.886-1.065-4.787-.3-2.993 1.202-5.943 1.06-8.926-.017-1.684-.608-3.179-1.563-4.735-2.408l-.043.03a2.96 2.96 0 0 0 .04-.029c-.038-.117-.107-.12-.197-.054l.122.107c1.29 2.115 3.034 3.817 5.004 5.271 3.793 2.8 7.936 4.471 12.784 3.73A66.714 66.714 0 0 1 37 40.877c1.98-.16 3.866.398 5.753.899Zm-9.14-30.345c-.105-.076-.206-.266-.42-.069 1.745 2.36 3.985 4.098 6.683 5.193 4.354 1.767 8.773 2.07 13.293.51 3.51-1.21 6.033-.028 7.343 3.38.19-3.955-2.137-6.837-5.843-7.401-2.084-.318-4.01.373-5.962.94-5.434 1.575-10.485.798-15.094-2.553Zm27.085 15.425c.708.059 1.416.123 2.124.185-1.6-1.405-3.55-1.517-5.523-1.404-3.003.17-5.167 1.903-7.14 3.972-1.739 1.824-3.31 3.87-5.903 4.604.043.078.054.117.066.117.35.005.699.021 1.047.005 3.768-.17 7.317-.965 10.14-3.7.89-.86 1.685-1.817 2.544-2.71.716-.746 1.584-1.159 2.645-1.07Zm-8.753-4.67c-2.812.246-5.254 1.409-7.548 2.943-1.766 1.18-3.654 1.738-5.776 1.37-.374-.066-.75-.114-1.124-.17l-.013.156c.135.07.265.151.405.207.354.14.702.308 1.07.395 4.083.971 7.992.474 11.516-1.803 2.221-1.435 4.521-1.707 7.013-1.336.252.038.503.083.756.107.234.022.479.255.795.003-2.179-1.574-4.526-2.096-7.094-1.872Zm-10.049-9.544c1.475.051 2.943-.142 4.486-1.059-.452.04-.643.04-.827.076-2.126.424-4.033-.04-5.733-1.383-.623-.493-1.257-.974-1.889-1.457-2.503-1.914-5.374-2.555-8.514-2.5.05.154.054.26.108.315 3.417 3.455 7.371 5.836 12.369 6.008Zm24.727 17.731c-2.114-2.097-4.952-2.367-7.578-.537 1.738.078 3.043.632 4.101 1.728.374.388.763.768 1.182 1.106 1.6 1.29 4.311 1.352 5.896.155-1.861-.726-1.861-.726-3.601-2.452Zm-21.058 16.06c-1.858-3.46-4.981-4.24-8.59-4.008a9.667 9.667 0 0 1 2.977 1.39c.84.586 1.547 1.311 2.243 2.055 1.38 1.473 3.534 2.376 4.962 2.07-.656-.412-1.238-.848-1.592-1.507Zm17.29-19.32c0-.023.001-.045.003-.068l-.006.006.006-.006-.036-.004.021.018.012.053Zm-20 14.744a7.61 7.61 0 0 0-.072-.041.127.127 0 0 0 .015.043c.005.008.038 0 .058-.002Zm-.072-.041-.008-.034-.008.01.008-.01-.022-.006.005.026.024.014Z"
|
|
||||||
fill="#FD4F00"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
<h1 class="text-brand mt-10 flex items-center text-sm font-semibold leading-6">
|
|
||||||
Phoenix Framework
|
|
||||||
<small class="bg-brand/5 text-[0.8125rem] ml-3 rounded-full px-2 font-medium leading-6">
|
|
||||||
v<%= Application.spec(:phoenix, :vsn) %>
|
|
||||||
</small>
|
|
||||||
</h1>
|
|
||||||
<p class="text-[2rem] mt-4 font-semibold leading-10 tracking-tighter text-zinc-900 text-balance">
|
|
||||||
Peace of mind from prototype to production.
|
|
||||||
</p>
|
|
||||||
<p class="mt-4 text-base leading-7 text-zinc-600">
|
|
||||||
Build rich, interactive web applications quickly, with less code and fewer moving parts. Join our growing community of developers using Phoenix to craft APIs, HTML5 apps and more, for fun or at scale.
|
|
||||||
</p>
|
|
||||||
<div class="flex">
|
|
||||||
<div class="w-full sm:w-auto">
|
|
||||||
<div class="mt-10 grid grid-cols-1 gap-x-6 gap-y-4 sm:grid-cols-3">
|
|
||||||
<a
|
|
||||||
href="https://hexdocs.pm/phoenix/overview.html"
|
|
||||||
class="group relative rounded-2xl px-6 py-4 text-sm font-semibold leading-6 text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 rounded-2xl bg-zinc-50 transition group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true" class="h-6 w-6">
|
|
||||||
<path d="m12 4 10-2v18l-10 2V4Z" fill="#18181B" fill-opacity=".15" />
|
|
||||||
<path
|
|
||||||
d="M12 4 2 2v18l10 2m0-18v18m0-18 10-2v18l-10 2"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Guides & Docs
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
href="https://github.com/phoenixframework/phoenix"
|
|
||||||
class="group relative rounded-2xl px-6 py-4 text-sm font-semibold leading-6 text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 rounded-2xl bg-zinc-50 transition group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" aria-hidden="true" class="h-6 w-6">
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M12 0C5.37 0 0 5.506 0 12.303c0 5.445 3.435 10.043 8.205 11.674.6.107.825-.262.825-.585 0-.292-.015-1.261-.015-2.291C6 21.67 5.22 20.346 4.98 19.654c-.135-.354-.72-1.446-1.23-1.738-.42-.23-1.02-.8-.015-.815.945-.015 1.62.892 1.845 1.261 1.08 1.86 2.805 1.338 3.495 1.015.105-.8.42-1.338.765-1.645-2.67-.308-5.46-1.37-5.46-6.075 0-1.338.465-2.446 1.23-3.307-.12-.308-.54-1.569.12-3.26 0 0 1.005-.323 3.3 1.26.96-.276 1.98-.415 3-.415s2.04.139 3 .416c2.295-1.6 3.3-1.261 3.3-1.261.66 1.691.24 2.952.12 3.26.765.861 1.23 1.953 1.23 3.307 0 4.721-2.805 5.767-5.475 6.075.435.384.81 1.122.81 2.276 0 1.645-.015 2.968-.015 3.383 0 .323.225.707.825.585a12.047 12.047 0 0 0 5.919-4.489A12.536 12.536 0 0 0 24 12.304C24 5.505 18.63 0 12 0Z"
|
|
||||||
fill="#18181B"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Source Code
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
href={"https://github.com/phoenixframework/phoenix/blob/v#{Application.spec(:phoenix, :vsn)}/CHANGELOG.md"}
|
|
||||||
class="group relative rounded-2xl px-6 py-4 text-sm font-semibold leading-6 text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 rounded-2xl bg-zinc-50 transition group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true" class="h-6 w-6">
|
|
||||||
<path
|
|
||||||
d="M12 1v6M12 17v6"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
<circle
|
|
||||||
cx="12"
|
|
||||||
cy="12"
|
|
||||||
r="4"
|
|
||||||
fill="#18181B"
|
|
||||||
fill-opacity=".15"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Changelog
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div class="mt-10 grid grid-cols-1 gap-y-4 text-sm leading-6 text-zinc-700 sm:grid-cols-2">
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://twitter.com/elixirphoenix"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="h-4 w-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M5.403 14c5.283 0 8.172-4.617 8.172-8.62 0-.131 0-.262-.008-.391A6.033 6.033 0 0 0 15 3.419a5.503 5.503 0 0 1-1.65.477 3.018 3.018 0 0 0 1.263-1.676 5.579 5.579 0 0 1-1.824.736 2.832 2.832 0 0 0-1.63-.916 2.746 2.746 0 0 0-1.821.319A2.973 2.973 0 0 0 8.076 3.78a3.185 3.185 0 0 0-.182 1.938 7.826 7.826 0 0 1-3.279-.918 8.253 8.253 0 0 1-2.64-2.247 3.176 3.176 0 0 0-.315 2.208 3.037 3.037 0 0 0 1.203 1.836A2.739 2.739 0 0 1 1.56 6.22v.038c0 .7.23 1.377.65 1.919.42.54 1.004.912 1.654 1.05-.423.122-.866.14-1.297.052.184.602.541 1.129 1.022 1.506a2.78 2.78 0 0 0 1.662.598 5.656 5.656 0 0 1-2.007 1.074A5.475 5.475 0 0 1 1 12.64a7.827 7.827 0 0 0 4.403 1.358" />
|
|
||||||
</svg>
|
|
||||||
Follow on Twitter
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://elixirforum.com"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="h-4 w-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M8 13.833c3.866 0 7-2.873 7-6.416C15 3.873 11.866 1 8 1S1 3.873 1 7.417c0 1.081.292 2.1.808 2.995.606 1.05.806 2.399.086 3.375l-.208.283c-.285.386-.01.905.465.85.852-.098 2.048-.318 3.137-.81a3.717 3.717 0 0 1 1.91-.318c.263.027.53.041.802.041Z" />
|
|
||||||
</svg>
|
|
||||||
Discuss on the Elixir Forum
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://web.libera.chat/#elixir"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="h-4 w-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M6.356 2.007a.75.75 0 0 1 .637.849l-1.5 10.5a.75.75 0 1 1-1.485-.212l1.5-10.5a.75.75 0 0 1 .848-.637ZM11.356 2.008a.75.75 0 0 1 .637.848l-1.5 10.5a.75.75 0 0 1-1.485-.212l1.5-10.5a.75.75 0 0 1 .848-.636Z"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M14 5.25a.75.75 0 0 1-.75.75h-9.5a.75.75 0 0 1 0-1.5h9.5a.75.75 0 0 1 .75.75ZM13 10.75a.75.75 0 0 1-.75.75h-9.5a.75.75 0 0 1 0-1.5h9.5a.75.75 0 0 1 .75.75Z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Chat on Libera IRC
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://discord.gg/elixir"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="h-4 w-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M13.545 2.995c-1.02-.46-2.114-.8-3.257-.994a.05.05 0 0 0-.052.024c-.141.246-.297.567-.406.82a12.377 12.377 0 0 0-3.658 0 8.238 8.238 0 0 0-.412-.82.052.052 0 0 0-.052-.024 13.315 13.315 0 0 0-3.257.994.046.046 0 0 0-.021.018C.356 6.063-.213 9.036.066 11.973c.001.015.01.029.02.038a13.353 13.353 0 0 0 3.996 1.987.052.052 0 0 0 .056-.018c.308-.414.582-.85.818-1.309a.05.05 0 0 0-.028-.069 8.808 8.808 0 0 1-1.248-.585.05.05 0 0 1-.005-.084c.084-.062.168-.126.248-.191a.05.05 0 0 1 .051-.007c2.619 1.176 5.454 1.176 8.041 0a.05.05 0 0 1 .053.006c.08.065.164.13.248.192a.05.05 0 0 1-.004.084c-.399.23-.813.423-1.249.585a.05.05 0 0 0-.027.07c.24.457.514.893.817 1.307a.051.051 0 0 0 .056.019 13.31 13.31 0 0 0 4.001-1.987.05.05 0 0 0 .021-.037c.334-3.396-.559-6.345-2.365-8.96a.04.04 0 0 0-.021-.02Zm-8.198 7.19c-.789 0-1.438-.712-1.438-1.587 0-.874.637-1.586 1.438-1.586.807 0 1.45.718 1.438 1.586 0 .875-.637 1.587-1.438 1.587Zm5.316 0c-.788 0-1.438-.712-1.438-1.587 0-.874.637-1.586 1.438-1.586.807 0 1.45.718 1.438 1.586 0 .875-.63 1.587-1.438 1.587Z" />
|
|
||||||
</svg>
|
|
||||||
Join our Discord server
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://fly.io/docs/elixir/getting-started/"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 20 20"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="h-4 w-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M1 12.5A4.5 4.5 0 005.5 17H15a4 4 0 001.866-7.539 3.504 3.504 0 00-4.504-4.272A4.5 4.5 0 004.06 8.235 4.502 4.502 0 001 12.5z" />
|
|
||||||
</svg>
|
|
||||||
Deploy your application
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
defmodule HelloWeb.Endpoint do
|
|
||||||
use Phoenix.Endpoint, otp_app: :hello
|
|
||||||
|
|
||||||
# The session will be stored in the cookie and signed,
|
|
||||||
# this means its contents can be read but not tampered with.
|
|
||||||
# Set :encryption_salt if you would also like to encrypt it.
|
|
||||||
@session_options [
|
|
||||||
store: :cookie,
|
|
||||||
key: "_hello_key",
|
|
||||||
signing_salt: "dEuBdnqI",
|
|
||||||
same_site: "Lax"
|
|
||||||
]
|
|
||||||
|
|
||||||
socket "/live", Phoenix.LiveView.Socket,
|
|
||||||
websocket: [connect_info: [session: @session_options]],
|
|
||||||
longpoll: [connect_info: [session: @session_options]]
|
|
||||||
|
|
||||||
# Serve at "/" the static files from "priv/static" directory.
|
|
||||||
#
|
|
||||||
# You should set gzip to true if you are running phx.digest
|
|
||||||
# when deploying your static files in production.
|
|
||||||
plug Plug.Static,
|
|
||||||
at: "/",
|
|
||||||
from: :hello,
|
|
||||||
gzip: false,
|
|
||||||
only: HelloWeb.static_paths()
|
|
||||||
|
|
||||||
# Code reloading can be explicitly enabled under the
|
|
||||||
# :code_reloader configuration of your endpoint.
|
|
||||||
if code_reloading? do
|
|
||||||
socket "/phoenix/live_reload/socket", Phoenix.LiveReloader.Socket
|
|
||||||
plug Phoenix.LiveReloader
|
|
||||||
plug Phoenix.CodeReloader
|
|
||||||
plug Phoenix.Ecto.CheckRepoStatus, otp_app: :hello
|
|
||||||
end
|
|
||||||
|
|
||||||
plug Phoenix.LiveDashboard.RequestLogger,
|
|
||||||
param_key: "request_logger",
|
|
||||||
cookie_key: "request_logger"
|
|
||||||
|
|
||||||
plug Plug.RequestId
|
|
||||||
plug Plug.Telemetry, event_prefix: [:phoenix, :endpoint]
|
|
||||||
|
|
||||||
plug Plug.Parsers,
|
|
||||||
parsers: [:urlencoded, :multipart, :json],
|
|
||||||
pass: ["*/*"],
|
|
||||||
json_decoder: Phoenix.json_library()
|
|
||||||
|
|
||||||
plug Plug.MethodOverride
|
|
||||||
plug Plug.Head
|
|
||||||
plug Plug.Session, @session_options
|
|
||||||
plug HelloWeb.Router
|
|
||||||
end
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
defmodule HelloWeb.Gettext do
|
|
||||||
@moduledoc """
|
|
||||||
A module providing Internationalization with a gettext-based API.
|
|
||||||
|
|
||||||
By using [Gettext](https://hexdocs.pm/gettext),
|
|
||||||
your module gains a set of macros for translations, for example:
|
|
||||||
|
|
||||||
import HelloWeb.Gettext
|
|
||||||
|
|
||||||
# Simple translation
|
|
||||||
gettext("Here is the string to translate")
|
|
||||||
|
|
||||||
# Plural translation
|
|
||||||
ngettext("Here is the string to translate",
|
|
||||||
"Here are the strings to translate",
|
|
||||||
3)
|
|
||||||
|
|
||||||
# Domain-based translation
|
|
||||||
dgettext("errors", "Here is the error message to translate")
|
|
||||||
|
|
||||||
See the [Gettext Docs](https://hexdocs.pm/gettext) for detailed usage.
|
|
||||||
"""
|
|
||||||
use Gettext, otp_app: :hello
|
|
||||||
end
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
defmodule HelloWeb.Router do
|
|
||||||
use HelloWeb, :router
|
|
||||||
|
|
||||||
pipeline :browser do
|
|
||||||
plug :accepts, ["html"]
|
|
||||||
plug :fetch_session
|
|
||||||
plug :fetch_live_flash
|
|
||||||
plug :put_root_layout, html: {HelloWeb.Layouts, :root}
|
|
||||||
plug :protect_from_forgery
|
|
||||||
plug :put_secure_browser_headers
|
|
||||||
end
|
|
||||||
|
|
||||||
pipeline :api do
|
|
||||||
plug :accepts, ["json"]
|
|
||||||
end
|
|
||||||
|
|
||||||
scope "/", HelloWeb do
|
|
||||||
pipe_through :browser
|
|
||||||
|
|
||||||
get "/", PageController, :home
|
|
||||||
end
|
|
||||||
|
|
||||||
# Other scopes may use custom stacks.
|
|
||||||
# scope "/api", HelloWeb do
|
|
||||||
# pipe_through :api
|
|
||||||
# end
|
|
||||||
|
|
||||||
# Enable LiveDashboard and Swoosh mailbox preview in development
|
|
||||||
if Application.compile_env(:hello, :dev_routes) do
|
|
||||||
# If you want to use the LiveDashboard in production, you should put
|
|
||||||
# it behind authentication and allow only admins to access it.
|
|
||||||
# If your application does not have an admins-only section yet,
|
|
||||||
# you can use Plug.BasicAuth to set up some basic authentication
|
|
||||||
# as long as you are also using SSL (which you should anyway).
|
|
||||||
import Phoenix.LiveDashboard.Router
|
|
||||||
|
|
||||||
scope "/dev" do
|
|
||||||
pipe_through :browser
|
|
||||||
|
|
||||||
live_dashboard "/dashboard", metrics: HelloWeb.Telemetry
|
|
||||||
forward "/mailbox", Plug.Swoosh.MailboxPreview
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
defmodule HelloWeb.Telemetry do
|
|
||||||
use Supervisor
|
|
||||||
import Telemetry.Metrics
|
|
||||||
|
|
||||||
def start_link(arg) do
|
|
||||||
Supervisor.start_link(__MODULE__, arg, name: __MODULE__)
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def init(_arg) do
|
|
||||||
children = [
|
|
||||||
# Telemetry poller will execute the given period measurements
|
|
||||||
# every 10_000ms. Learn more here: https://hexdocs.pm/telemetry_metrics
|
|
||||||
{:telemetry_poller, measurements: periodic_measurements(), period: 10_000}
|
|
||||||
# Add reporters as children of your supervision tree.
|
|
||||||
# {Telemetry.Metrics.ConsoleReporter, metrics: metrics()}
|
|
||||||
]
|
|
||||||
|
|
||||||
Supervisor.init(children, strategy: :one_for_one)
|
|
||||||
end
|
|
||||||
|
|
||||||
def metrics do
|
|
||||||
[
|
|
||||||
# Phoenix Metrics
|
|
||||||
summary("phoenix.endpoint.start.system_time",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.endpoint.stop.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.start.system_time",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.exception.duration",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.stop.duration",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.socket_connected.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.channel_joined.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.channel_handled_in.duration",
|
|
||||||
tags: [:event],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
|
|
||||||
# Database Metrics
|
|
||||||
summary("hello.repo.query.total_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The sum of the other measurements"
|
|
||||||
),
|
|
||||||
summary("hello.repo.query.decode_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent decoding the data received from the database"
|
|
||||||
),
|
|
||||||
summary("hello.repo.query.query_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent executing the query"
|
|
||||||
),
|
|
||||||
summary("hello.repo.query.queue_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent waiting for a database connection"
|
|
||||||
),
|
|
||||||
summary("hello.repo.query.idle_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description:
|
|
||||||
"The time the connection spent waiting before being checked out for the query"
|
|
||||||
),
|
|
||||||
|
|
||||||
# VM Metrics
|
|
||||||
summary("vm.memory.total", unit: {:byte, :kilobyte}),
|
|
||||||
summary("vm.total_run_queue_lengths.total"),
|
|
||||||
summary("vm.total_run_queue_lengths.cpu"),
|
|
||||||
summary("vm.total_run_queue_lengths.io")
|
|
||||||
]
|
|
||||||
end
|
|
||||||
|
|
||||||
defp periodic_measurements do
|
|
||||||
[
|
|
||||||
# A module, function and arguments to be invoked periodically.
|
|
||||||
# This function must call :telemetry.execute/3 and a metric must be added above.
|
|
||||||
# {HelloWeb, :count_users, []}
|
|
||||||
]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
defmodule Hello.MixProject do
|
|
||||||
use Mix.Project
|
|
||||||
|
|
||||||
def project do
|
|
||||||
[
|
|
||||||
app: :hello,
|
|
||||||
version: "0.1.0",
|
|
||||||
elixir: "~> 1.14",
|
|
||||||
elixirc_paths: elixirc_paths(Mix.env()),
|
|
||||||
start_permanent: Mix.env() == :prod,
|
|
||||||
aliases: aliases(),
|
|
||||||
deps: deps()
|
|
||||||
]
|
|
||||||
end
|
|
||||||
|
|
||||||
# Configuration for the OTP application.
|
|
||||||
#
|
|
||||||
# Type `mix help compile.app` for more information.
|
|
||||||
def application do
|
|
||||||
[
|
|
||||||
mod: {Hello.Application, []},
|
|
||||||
extra_applications: [:logger, :runtime_tools]
|
|
||||||
]
|
|
||||||
end
|
|
||||||
|
|
||||||
# Specifies which paths to compile per environment.
|
|
||||||
defp elixirc_paths(:test), do: ["lib", "test/support"]
|
|
||||||
defp elixirc_paths(_), do: ["lib"]
|
|
||||||
|
|
||||||
# Specifies your project dependencies.
|
|
||||||
#
|
|
||||||
# Type `mix help deps` for examples and options.
|
|
||||||
defp deps do
|
|
||||||
[
|
|
||||||
{:phoenix, "~> 1.7.14"},
|
|
||||||
{:phoenix_ecto, "~> 4.5"},
|
|
||||||
{:ecto_sql, "~> 3.10"},
|
|
||||||
{:postgrex, ">= 0.0.0"},
|
|
||||||
{:phoenix_html, "~> 4.1"},
|
|
||||||
{:phoenix_live_reload, "~> 1.2", only: :dev},
|
|
||||||
# TODO bump on release to {:phoenix_live_view, "~> 1.0.0"},
|
|
||||||
{:phoenix_live_view, "~> 1.0.0-rc.1", override: true},
|
|
||||||
{:floki, ">= 0.30.0", only: :test},
|
|
||||||
{:phoenix_live_dashboard, "~> 0.8.3"},
|
|
||||||
{:esbuild, "~> 0.8", runtime: Mix.env() == :dev},
|
|
||||||
{:tailwind, "~> 0.2", runtime: Mix.env() == :dev},
|
|
||||||
{:heroicons,
|
|
||||||
github: "tailwindlabs/heroicons",
|
|
||||||
tag: "v2.1.1",
|
|
||||||
sparse: "optimized",
|
|
||||||
app: false,
|
|
||||||
compile: false,
|
|
||||||
depth: 1},
|
|
||||||
{:swoosh, "~> 1.5"},
|
|
||||||
{:finch, "~> 0.13"},
|
|
||||||
{:telemetry_metrics, "~> 1.0"},
|
|
||||||
{:telemetry_poller, "~> 1.0"},
|
|
||||||
{:gettext, "~> 0.20"},
|
|
||||||
{:jason, "~> 1.2"},
|
|
||||||
{:dns_cluster, "~> 0.1.1"},
|
|
||||||
{:bandit, "~> 1.5"}
|
|
||||||
]
|
|
||||||
end
|
|
||||||
|
|
||||||
# Aliases are shortcuts or tasks specific to the current project.
|
|
||||||
# For example, to install project dependencies and perform other setup tasks, run:
|
|
||||||
#
|
|
||||||
# $ mix setup
|
|
||||||
#
|
|
||||||
# See the documentation for `Mix` for more info on aliases.
|
|
||||||
defp aliases do
|
|
||||||
[
|
|
||||||
setup: ["deps.get", "ecto.setup", "assets.setup", "assets.build"],
|
|
||||||
"ecto.setup": ["ecto.create", "ecto.migrate", "run priv/repo/seeds.exs"],
|
|
||||||
"ecto.reset": ["ecto.drop", "ecto.setup"],
|
|
||||||
test: ["ecto.create --quiet", "ecto.migrate --quiet", "test"],
|
|
||||||
"assets.setup": ["tailwind.install --if-missing", "esbuild.install --if-missing"],
|
|
||||||
"assets.build": ["tailwind hello", "esbuild hello"],
|
|
||||||
"assets.deploy": [
|
|
||||||
"tailwind hello --minify",
|
|
||||||
"esbuild hello --minify",
|
|
||||||
"phx.digest"
|
|
||||||
]
|
|
||||||
]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
%{
|
|
||||||
"bandit": {:hex, :bandit, "1.5.7", "6856b1e1df4f2b0cb3df1377eab7891bec2da6a7fd69dc78594ad3e152363a50", [:mix], [{:hpax, "~> 1.0.0", [hex: :hpax, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:thousand_island, "~> 1.0", [hex: :thousand_island, repo: "hexpm", optional: false]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "f2dd92ae87d2cbea2fa9aa1652db157b6cba6c405cb44d4f6dd87abba41371cd"},
|
|
||||||
"castore": {:hex, :castore, "1.0.8", "dedcf20ea746694647f883590b82d9e96014057aff1d44d03ec90f36a5c0dc6e", [:mix], [], "hexpm", "0b2b66d2ee742cb1d9cb8c8be3b43c3a70ee8651f37b75a8b982e036752983f1"},
|
|
||||||
"db_connection": {:hex, :db_connection, "2.7.0", "b99faa9291bb09892c7da373bb82cba59aefa9b36300f6145c5f201c7adf48ec", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "dcf08f31b2701f857dfc787fbad78223d61a32204f217f15e881dd93e4bdd3ff"},
|
|
||||||
"decimal": {:hex, :decimal, "2.1.1", "5611dca5d4b2c3dd497dec8f68751f1f1a54755e8ed2a966c2633cf885973ad6", [:mix], [], "hexpm", "53cfe5f497ed0e7771ae1a475575603d77425099ba5faef9394932b35020ffcc"},
|
|
||||||
"dns_cluster": {:hex, :dns_cluster, "0.1.3", "0bc20a2c88ed6cc494f2964075c359f8c2d00e1bf25518a6a6c7fd277c9b0c66", [:mix], [], "hexpm", "46cb7c4a1b3e52c7ad4cbe33ca5079fbde4840dedeafca2baf77996c2da1bc33"},
|
|
||||||
"ecto": {:hex, :ecto, "3.12.3", "1a9111560731f6c3606924c81c870a68a34c819f6d4f03822f370ea31a582208", [:mix], [{:decimal, "~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "9efd91506ae722f95e48dc49e70d0cb632ede3b7a23896252a60a14ac6d59165"},
|
|
||||||
"ecto_sql": {:hex, :ecto_sql, "3.12.0", "73cea17edfa54bde76ee8561b30d29ea08f630959685006d9c6e7d1e59113b7d", [:mix], [{:db_connection, "~> 2.4.1 or ~> 2.5", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.12", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.7", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.19 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "dc9e4d206f274f3947e96142a8fdc5f69a2a6a9abb4649ef5c882323b6d512f0"},
|
|
||||||
"esbuild": {:hex, :esbuild, "0.8.1", "0cbf919f0eccb136d2eeef0df49c4acf55336de864e63594adcea3814f3edf41", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.4", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "25fc876a67c13cb0a776e7b5d7974851556baeda2085296c14ab48555ea7560f"},
|
|
||||||
"expo": {:hex, :expo, "1.1.0", "f7b9ed7fb5745ebe1eeedf3d6f29226c5dd52897ac67c0f8af62a07e661e5c75", [:mix], [], "hexpm", "fbadf93f4700fb44c331362177bdca9eeb8097e8b0ef525c9cc501cb9917c960"},
|
|
||||||
"file_system": {:hex, :file_system, "1.0.1", "79e8ceaddb0416f8b8cd02a0127bdbababe7bf4a23d2a395b983c1f8b3f73edd", [:mix], [], "hexpm", "4414d1f38863ddf9120720cd976fce5bdde8e91d8283353f0e31850fa89feb9e"},
|
|
||||||
"finch": {:hex, :finch, "0.19.0", "c644641491ea854fc5c1bbaef36bfc764e3f08e7185e1f084e35e0672241b76d", [:mix], [{:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:mint, "~> 1.6.2 or ~> 1.7", [hex: :mint, repo: "hexpm", optional: false]}, {:nimble_options, "~> 0.4 or ~> 1.0", [hex: :nimble_options, repo: "hexpm", optional: false]}, {:nimble_pool, "~> 1.1", [hex: :nimble_pool, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "fc5324ce209125d1e2fa0fcd2634601c52a787aff1cd33ee833664a5af4ea2b6"},
|
|
||||||
"floki": {:hex, :floki, "0.36.2", "a7da0193538c93f937714a6704369711998a51a6164a222d710ebd54020aa7a3", [:mix], [], "hexpm", "a8766c0bc92f074e5cb36c4f9961982eda84c5d2b8e979ca67f5c268ec8ed580"},
|
|
||||||
"gettext": {:hex, :gettext, "0.26.1", "38e14ea5dcf962d1fc9f361b63ea07c0ce715a8ef1f9e82d3dfb8e67e0416715", [:mix], [{:expo, "~> 0.5.1 or ~> 1.0", [hex: :expo, repo: "hexpm", optional: false]}], "hexpm", "01ce56f188b9dc28780a52783d6529ad2bc7124f9744e571e1ee4ea88bf08734"},
|
|
||||||
"heroicons": {:git, "https://github.com/tailwindlabs/heroicons.git", "88ab3a0d790e6a47404cba02800a6b25d2afae50", [tag: "v2.1.1", sparse: "optimized"]},
|
|
||||||
"hpax": {:hex, :hpax, "1.0.0", "28dcf54509fe2152a3d040e4e3df5b265dcb6cb532029ecbacf4ce52caea3fd2", [:mix], [], "hexpm", "7f1314731d711e2ca5fdc7fd361296593fc2542570b3105595bb0bc6d0fad601"},
|
|
||||||
"jason": {:hex, :jason, "1.4.4", "b9226785a9aa77b6857ca22832cffa5d5011a667207eb2a0ad56adb5db443b8a", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "c5eb0cab91f094599f94d55bc63409236a8ec69a21a67814529e8d5f6cc90b3b"},
|
|
||||||
"mime": {:hex, :mime, "2.0.6", "8f18486773d9b15f95f4f4f1e39b710045fa1de891fada4516559967276e4dc2", [:mix], [], "hexpm", "c9945363a6b26d747389aac3643f8e0e09d30499a138ad64fe8fd1d13d9b153e"},
|
|
||||||
"mint": {:hex, :mint, "1.6.2", "af6d97a4051eee4f05b5500671d47c3a67dac7386045d87a904126fd4bbcea2e", [:mix], [{:castore, "~> 0.1.0 or ~> 1.0", [hex: :castore, repo: "hexpm", optional: true]}, {:hpax, "~> 0.1.1 or ~> 0.2.0 or ~> 1.0", [hex: :hpax, repo: "hexpm", optional: false]}], "hexpm", "5ee441dffc1892f1ae59127f74afe8fd82fda6587794278d924e4d90ea3d63f9"},
|
|
||||||
"nimble_options": {:hex, :nimble_options, "1.1.1", "e3a492d54d85fc3fd7c5baf411d9d2852922f66e69476317787a7b2bb000a61b", [:mix], [], "hexpm", "821b2470ca9442c4b6984882fe9bb0389371b8ddec4d45a9504f00a66f650b44"},
|
|
||||||
"nimble_pool": {:hex, :nimble_pool, "1.1.0", "bf9c29fbdcba3564a8b800d1eeb5a3c58f36e1e11d7b7fb2e084a643f645f06b", [:mix], [], "hexpm", "af2e4e6b34197db81f7aad230c1118eac993acc0dae6bc83bac0126d4ae0813a"},
|
|
||||||
"phoenix": {:hex, :phoenix, "1.7.14", "a7d0b3f1bc95987044ddada111e77bd7f75646a08518942c72a8440278ae7825", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.1", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.7", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.2 or ~> 2.0", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:websock_adapter, "~> 0.5.3", [hex: :websock_adapter, repo: "hexpm", optional: false]}], "hexpm", "c7859bc56cc5dfef19ecfc240775dae358cbaa530231118a9e014df392ace61a"},
|
|
||||||
"phoenix_ecto": {:hex, :phoenix_ecto, "4.6.2", "3b83b24ab5a2eb071a20372f740d7118767c272db386831b2e77638c4dcc606d", [:mix], [{:ecto, "~> 3.5", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.1", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}], "hexpm", "3f94d025f59de86be00f5f8c5dd7b5965a3298458d21ab1c328488be3b5fcd59"},
|
|
||||||
"phoenix_html": {:hex, :phoenix_html, "4.1.1", "4c064fd3873d12ebb1388425a8f2a19348cef56e7289e1998e2d2fa758aa982e", [:mix], [], "hexpm", "f2f2df5a72bc9a2f510b21497fd7d2b86d932ec0598f0210fed4114adc546c6f"},
|
|
||||||
"phoenix_live_dashboard": {:hex, :phoenix_live_dashboard, "0.8.4", "4508e481f791ce62ec6a096e13b061387158cbeefacca68c6c1928e1305e23ed", [:mix], [{:ecto, "~> 3.6.2 or ~> 3.7", [hex: :ecto, repo: "hexpm", optional: true]}, {:ecto_mysql_extras, "~> 0.5", [hex: :ecto_mysql_extras, repo: "hexpm", optional: true]}, {:ecto_psql_extras, "~> 0.7", [hex: :ecto_psql_extras, repo: "hexpm", optional: true]}, {:ecto_sqlite3_extras, "~> 1.1.7 or ~> 1.2.0", [hex: :ecto_sqlite3_extras, repo: "hexpm", optional: true]}, {:mime, "~> 1.6 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.19 or ~> 1.0", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}, {:telemetry_metrics, "~> 0.6 or ~> 1.0", [hex: :telemetry_metrics, repo: "hexpm", optional: false]}], "hexpm", "2984aae96994fbc5c61795a73b8fb58153b41ff934019cfb522343d2d3817d59"},
|
|
||||||
"phoenix_live_reload": {:hex, :phoenix_live_reload, "1.5.3", "f2161c207fda0e4fb55165f650f7f8db23f02b29e3bff00ff7ef161d6ac1f09d", [:mix], [{:file_system, "~> 0.3 or ~> 1.0", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "b4ec9cd73cb01ff1bd1cac92e045d13e7030330b74164297d1aee3907b54803c"},
|
|
||||||
"phoenix_live_view": {:hex, :phoenix_live_view, "1.0.0-rc.6", "47d2669995ea326e5c71f5c1bc9177109cebf211385c638faa7b5862a401e516", [:mix], [{:floki, "~> 0.36", [hex: :floki, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6.15 or ~> 1.7.0", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.3 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.15", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "e56e4f1642a0b20edc2488cab30e5439595e0d8b5b259f76ef98b1c4e2e5b527"},
|
|
||||||
"phoenix_pubsub": {:hex, :phoenix_pubsub, "2.1.3", "3168d78ba41835aecad272d5e8cd51aa87a7ac9eb836eabc42f6e57538e3731d", [:mix], [], "hexpm", "bba06bc1dcfd8cb086759f0edc94a8ba2bc8896d5331a1e2c2902bf8e36ee502"},
|
|
||||||
"phoenix_template": {:hex, :phoenix_template, "1.0.4", "e2092c132f3b5e5b2d49c96695342eb36d0ed514c5b252a77048d5969330d639", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}], "hexpm", "2c0c81f0e5c6753faf5cca2f229c9709919aba34fab866d3bc05060c9c444206"},
|
|
||||||
"plug": {:hex, :plug, "1.16.1", "40c74619c12f82736d2214557dedec2e9762029b2438d6d175c5074c933edc9d", [:mix], [{:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug_crypto, "~> 1.1.1 or ~> 1.2 or ~> 2.0", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.3 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "a13ff6b9006b03d7e33874945b2755253841b238c34071ed85b0e86057f8cddc"},
|
|
||||||
"plug_crypto": {:hex, :plug_crypto, "2.1.0", "f44309c2b06d249c27c8d3f65cfe08158ade08418cf540fd4f72d4d6863abb7b", [:mix], [], "hexpm", "131216a4b030b8f8ce0f26038bc4421ae60e4bb95c5cf5395e1421437824c4fa"},
|
|
||||||
"postgrex": {:hex, :postgrex, "0.19.1", "73b498508b69aded53907fe48a1fee811be34cc720e69ef4ccd568c8715495ea", [:mix], [{:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "8bac7885a18f381e091ec6caf41bda7bb8c77912bb0e9285212829afe5d8a8f8"},
|
|
||||||
"swoosh": {:hex, :swoosh, "1.17.0", "4a082a6ce4d60b1f48ffa725c8da0e2304504569ff550f4ed2d088c923039cb0", [:mix], [{:bandit, ">= 1.0.0", [hex: :bandit, repo: "hexpm", optional: true]}, {:cowboy, "~> 1.1 or ~> 2.4", [hex: :cowboy, repo: "hexpm", optional: true]}, {:ex_aws, "~> 2.1", [hex: :ex_aws, repo: "hexpm", optional: true]}, {:finch, "~> 0.6", [hex: :finch, repo: "hexpm", optional: true]}, {:gen_smtp, "~> 0.13 or ~> 1.0", [hex: :gen_smtp, repo: "hexpm", optional: true]}, {:hackney, "~> 1.9", [hex: :hackney, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:mail, "~> 0.2", [hex: :mail, repo: "hexpm", optional: true]}, {:mime, "~> 1.1 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:mua, "~> 0.2.3", [hex: :mua, repo: "hexpm", optional: true]}, {:multipart, "~> 0.4", [hex: :multipart, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: true]}, {:plug_cowboy, ">= 1.0.0", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:req, "~> 0.5 or ~> 1.0", [hex: :req, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "659b8bc25f7483b872d051a7f0731fb8d5312165be0d0302a3c783b566b0a290"},
|
|
||||||
"tailwind": {:hex, :tailwind, "0.2.3", "277f08145d407de49650d0a4685dc062174bdd1ae7731c5f1da86163a24dfcdb", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "8e45e7a34a676a7747d04f7913a96c770c85e6be810a1d7f91e713d3a3655b5d"},
|
|
||||||
"telemetry": {:hex, :telemetry, "1.3.0", "fedebbae410d715cf8e7062c96a1ef32ec22e764197f70cda73d82778d61e7a2", [:rebar3], [], "hexpm", "7015fc8919dbe63764f4b4b87a95b7c0996bd539e0d499be6ec9d7f3875b79e6"},
|
|
||||||
"telemetry_metrics": {:hex, :telemetry_metrics, "1.0.0", "29f5f84991ca98b8eb02fc208b2e6de7c95f8bb2294ef244a176675adc7775df", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "f23713b3847286a534e005126d4c959ebcca68ae9582118ce436b521d1d47d5d"},
|
|
||||||
"telemetry_poller": {:hex, :telemetry_poller, "1.1.0", "58fa7c216257291caaf8d05678c8d01bd45f4bdbc1286838a28c4bb62ef32999", [:rebar3], [{:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "9eb9d9cbfd81cbd7cdd24682f8711b6e2b691289a0de6826e58452f28c103c8f"},
|
|
||||||
"thousand_island": {:hex, :thousand_island, "1.3.5", "6022b6338f1635b3d32406ff98d68b843ba73b3aa95cfc27154223244f3a6ca5", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "2be6954916fdfe4756af3239fb6b6d75d0b8063b5df03ba76fd8a4c87849e180"},
|
|
||||||
"websock": {:hex, :websock, "0.5.3", "2f69a6ebe810328555b6fe5c831a851f485e303a7c8ce6c5f675abeb20ebdadc", [:mix], [], "hexpm", "6105453d7fac22c712ad66fab1d45abdf049868f253cf719b625151460b8b453"},
|
|
||||||
"websock_adapter": {:hex, :websock_adapter, "0.5.7", "65fa74042530064ef0570b75b43f5c49bb8b235d6515671b3d250022cb8a1f9e", [:mix], [{:bandit, ">= 0.6.0", [hex: :bandit, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.6", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "d0f478ee64deddfec64b800673fd6e0c8888b079d9f3444dd96d2a98383bdbd1"},
|
|
||||||
}
|
|
||||||
@@ -1,112 +0,0 @@
|
|||||||
## `msgid`s in this file come from POT (.pot) files.
|
|
||||||
##
|
|
||||||
## Do not add, change, or remove `msgid`s manually here as
|
|
||||||
## they're tied to the ones in the corresponding POT file
|
|
||||||
## (with the same domain).
|
|
||||||
##
|
|
||||||
## Use `mix gettext.extract --merge` or `mix gettext.merge`
|
|
||||||
## to merge POT files into PO files.
|
|
||||||
msgid ""
|
|
||||||
msgstr ""
|
|
||||||
"Language: en\n"
|
|
||||||
|
|
||||||
## From Ecto.Changeset.cast/4
|
|
||||||
msgid "can't be blank"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.unique_constraint/3
|
|
||||||
msgid "has already been taken"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.put_change/3
|
|
||||||
msgid "is invalid"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_acceptance/3
|
|
||||||
msgid "must be accepted"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_format/3
|
|
||||||
msgid "has invalid format"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_subset/3
|
|
||||||
msgid "has an invalid entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_exclusion/3
|
|
||||||
msgid "is reserved"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_confirmation/3
|
|
||||||
msgid "does not match confirmation"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.no_assoc_constraint/3
|
|
||||||
msgid "is still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "are still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_length/3
|
|
||||||
msgid "should have %{count} item(s)"
|
|
||||||
msgid_plural "should have %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} character(s)"
|
|
||||||
msgid_plural "should be %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} byte(s)"
|
|
||||||
msgid_plural "should be %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at least %{count} item(s)"
|
|
||||||
msgid_plural "should have at least %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} character(s)"
|
|
||||||
msgid_plural "should be at least %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} byte(s)"
|
|
||||||
msgid_plural "should be at least %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at most %{count} item(s)"
|
|
||||||
msgid_plural "should have at most %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} character(s)"
|
|
||||||
msgid_plural "should be at most %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} byte(s)"
|
|
||||||
msgid_plural "should be at most %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_number/3
|
|
||||||
msgid "must be less than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be less than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
@@ -1,109 +0,0 @@
|
|||||||
## This is a PO Template file.
|
|
||||||
##
|
|
||||||
## `msgid`s here are often extracted from source code.
|
|
||||||
## Add new translations manually only if they're dynamic
|
|
||||||
## translations that can't be statically extracted.
|
|
||||||
##
|
|
||||||
## Run `mix gettext.extract` to bring this file up to
|
|
||||||
## date. Leave `msgstr`s empty as changing them here has no
|
|
||||||
## effect: edit them in PO (`.po`) files instead.
|
|
||||||
## From Ecto.Changeset.cast/4
|
|
||||||
msgid "can't be blank"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.unique_constraint/3
|
|
||||||
msgid "has already been taken"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.put_change/3
|
|
||||||
msgid "is invalid"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_acceptance/3
|
|
||||||
msgid "must be accepted"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_format/3
|
|
||||||
msgid "has invalid format"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_subset/3
|
|
||||||
msgid "has an invalid entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_exclusion/3
|
|
||||||
msgid "is reserved"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_confirmation/3
|
|
||||||
msgid "does not match confirmation"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.no_assoc_constraint/3
|
|
||||||
msgid "is still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "are still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_length/3
|
|
||||||
msgid "should have %{count} item(s)"
|
|
||||||
msgid_plural "should have %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} character(s)"
|
|
||||||
msgid_plural "should be %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} byte(s)"
|
|
||||||
msgid_plural "should be %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at least %{count} item(s)"
|
|
||||||
msgid_plural "should have at least %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} character(s)"
|
|
||||||
msgid_plural "should be at least %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} byte(s)"
|
|
||||||
msgid_plural "should be at least %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at most %{count} item(s)"
|
|
||||||
msgid_plural "should have at most %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} character(s)"
|
|
||||||
msgid_plural "should be at most %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} byte(s)"
|
|
||||||
msgid_plural "should be at most %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_number/3
|
|
||||||
msgid "must be less than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be less than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
[
|
|
||||||
import_deps: [:ecto_sql],
|
|
||||||
inputs: ["*.exs"]
|
|
||||||
]
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
# Script for populating the database. You can run it as:
|
|
||||||
#
|
|
||||||
# mix run priv/repo/seeds.exs
|
|
||||||
#
|
|
||||||
# Inside the script, you can read and write to any of your
|
|
||||||
# repositories directly:
|
|
||||||
#
|
|
||||||
# Hello.Repo.insert!(%Hello.SomeSchema{})
|
|
||||||
#
|
|
||||||
# We recommend using the bang functions (`insert!`, `update!`
|
|
||||||
# and so on) as they will fail if something goes wrong.
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 152 B |
@@ -1,6 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 71 48" fill="currentColor" aria-hidden="true">
|
|
||||||
<path
|
|
||||||
d="m26.371 33.477-.552-.1c-3.92-.729-6.397-3.1-7.57-6.829-.733-2.324.597-4.035 3.035-4.148 1.995-.092 3.362 1.055 4.57 2.39 1.557 1.72 2.984 3.558 4.514 5.305 2.202 2.515 4.797 4.134 8.347 3.634 3.183-.448 5.958-1.725 8.371-3.828.363-.316.761-.592 1.144-.886l-.241-.284c-2.027.63-4.093.841-6.205.735-3.195-.16-6.24-.828-8.964-2.582-2.486-1.601-4.319-3.746-5.19-6.611-.704-2.315.736-3.934 3.135-3.6.948.133 1.746.56 2.463 1.165.583.493 1.143 1.015 1.738 1.493 2.8 2.25 6.712 2.375 10.265-.068-5.842-.026-9.817-3.24-13.308-7.313-1.366-1.594-2.7-3.216-4.095-4.785-2.698-3.036-5.692-5.71-9.79-6.623C12.8-.623 7.745.14 2.893 2.361 1.926 2.804.997 3.319 0 4.149c.494 0 .763.006 1.032 0 2.446-.064 4.28 1.023 5.602 3.024.962 1.457 1.415 3.104 1.761 4.798.513 2.515.247 5.078.544 7.605.761 6.494 4.08 11.026 10.26 13.346 2.267.852 4.591 1.135 7.172.555ZM10.751 3.852c-.976.246-1.756-.148-2.56-.962 1.377-.343 2.592-.476 3.897-.528-.107.848-.607 1.306-1.336 1.49Zm32.002 37.924c-.085-.626-.62-.901-1.04-1.228-1.857-1.446-4.03-1.958-6.333-2-1.375-.026-2.735-.128-4.031-.61-.595-.22-1.26-.505-1.244-1.272.015-.78.693-1 1.31-1.184.505-.15 1.026-.247 1.6-.382-1.46-.936-2.886-1.065-4.787-.3-2.993 1.202-5.943 1.06-8.926-.017-1.684-.608-3.179-1.563-4.735-2.408l-.077.057c1.29 2.115 3.034 3.817 5.004 5.271 3.793 2.8 7.936 4.471 12.784 3.73A66.714 66.714 0 0 1 37 40.877c1.98-.16 3.866.398 5.753.899Zm-9.14-30.345c-.105-.076-.206-.266-.42-.069 1.745 2.36 3.985 4.098 6.683 5.193 4.354 1.767 8.773 2.07 13.293.51 3.51-1.21 6.033-.028 7.343 3.38.19-3.955-2.137-6.837-5.843-7.401-2.084-.318-4.01.373-5.962.94-5.434 1.575-10.485.798-15.094-2.553Zm27.085 15.425c.708.059 1.416.123 2.124.185-1.6-1.405-3.55-1.517-5.523-1.404-3.003.17-5.167 1.903-7.14 3.972-1.739 1.824-3.31 3.87-5.903 4.604.043.078.054.117.066.117.35.005.699.021 1.047.005 3.768-.17 7.317-.965 10.14-3.7.89-.86 1.685-1.817 2.544-2.71.716-.746 1.584-1.159 2.645-1.07Zm-8.753-4.67c-2.812.246-5.254 1.409-7.548 2.943-1.766 1.18-3.654 1.738-5.776 1.37-.374-.066-.75-.114-1.124-.17l-.013.156c.135.07.265.151.405.207.354.14.702.308 1.07.395 4.083.971 7.992.474 11.516-1.803 2.221-1.435 4.521-1.707 7.013-1.336.252.038.503.083.756.107.234.022.479.255.795.003-2.179-1.574-4.526-2.096-7.094-1.872Zm-10.049-9.544c1.475.051 2.943-.142 4.486-1.059-.452.04-.643.04-.827.076-2.126.424-4.033-.04-5.733-1.383-.623-.493-1.257-.974-1.889-1.457-2.503-1.914-5.374-2.555-8.514-2.5.05.154.054.26.108.315 3.417 3.455 7.371 5.836 12.369 6.008Zm24.727 17.731c-2.114-2.097-4.952-2.367-7.578-.537 1.738.078 3.043.632 4.101 1.728a13 13 0 0 0 1.182 1.106c1.6 1.29 4.311 1.352 5.896.155-1.861-.726-1.861-.726-3.601-2.452Zm-21.058 16.06c-1.858-3.46-4.981-4.24-8.59-4.008a9.667 9.667 0 0 1 2.977 1.39c.84.586 1.547 1.311 2.243 2.055 1.38 1.473 3.534 2.376 4.962 2.07-.656-.412-1.238-.848-1.592-1.507Zl-.006.006-.036-.004.021.018.012.053Za.127.127 0 0 0 .015.043c.005.008.038 0 .058-.002Zl-.008.01.005.026.024.014Z"
|
|
||||||
fill="#FD4F00"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 3.0 KiB |
@@ -1,5 +0,0 @@
|
|||||||
# See https://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file
|
|
||||||
#
|
|
||||||
# To ban all spiders from the entire site uncomment the next two lines:
|
|
||||||
# User-agent: *
|
|
||||||
# Disallow: /
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
defmodule HelloWeb.ErrorHTMLTest do
|
|
||||||
use HelloWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
# Bring render_to_string/4 for testing custom views
|
|
||||||
import Phoenix.Template
|
|
||||||
|
|
||||||
test "renders 404.html" do
|
|
||||||
assert render_to_string(HelloWeb.ErrorHTML, "404", "html", []) == "Not Found"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders 500.html" do
|
|
||||||
assert render_to_string(HelloWeb.ErrorHTML, "500", "html", []) == "Internal Server Error"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
defmodule HelloWeb.ErrorJSONTest do
|
|
||||||
use HelloWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
test "renders 404" do
|
|
||||||
assert HelloWeb.ErrorJSON.render("404.json", %{}) == %{errors: %{detail: "Not Found"}}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders 500" do
|
|
||||||
assert HelloWeb.ErrorJSON.render("500.json", %{}) ==
|
|
||||||
%{errors: %{detail: "Internal Server Error"}}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
defmodule HelloWeb.PageControllerTest do
|
|
||||||
use HelloWeb.ConnCase
|
|
||||||
|
|
||||||
test "GET /", %{conn: conn} do
|
|
||||||
conn = get(conn, ~p"/")
|
|
||||||
assert html_response(conn, 200) =~ "Peace of mind from prototype to production"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
defmodule HelloWeb.ConnCase do
|
|
||||||
@moduledoc """
|
|
||||||
This module defines the test case to be used by
|
|
||||||
tests that require setting up a connection.
|
|
||||||
|
|
||||||
Such tests rely on `Phoenix.ConnTest` and also
|
|
||||||
import other functionality to make it easier
|
|
||||||
to build common data structures and query the data layer.
|
|
||||||
|
|
||||||
Finally, if the test case interacts with the database,
|
|
||||||
we enable the SQL sandbox, so changes done to the database
|
|
||||||
are reverted at the end of every test. If you are using
|
|
||||||
PostgreSQL, you can even run database tests asynchronously
|
|
||||||
by setting `use HelloWeb.ConnCase, async: true`, although
|
|
||||||
this option is not recommended for other databases.
|
|
||||||
"""
|
|
||||||
|
|
||||||
use ExUnit.CaseTemplate
|
|
||||||
|
|
||||||
using do
|
|
||||||
quote do
|
|
||||||
# The default endpoint for testing
|
|
||||||
@endpoint HelloWeb.Endpoint
|
|
||||||
|
|
||||||
use HelloWeb, :verified_routes
|
|
||||||
|
|
||||||
# Import conveniences for testing with connections
|
|
||||||
import Plug.Conn
|
|
||||||
import Phoenix.ConnTest
|
|
||||||
import HelloWeb.ConnCase
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
setup tags do
|
|
||||||
Hello.DataCase.setup_sandbox(tags)
|
|
||||||
{:ok, conn: Phoenix.ConnTest.build_conn()}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
defmodule Hello.DataCase do
|
|
||||||
@moduledoc """
|
|
||||||
This module defines the setup for tests requiring
|
|
||||||
access to the application's data layer.
|
|
||||||
|
|
||||||
You may define functions here to be used as helpers in
|
|
||||||
your tests.
|
|
||||||
|
|
||||||
Finally, if the test case interacts with the database,
|
|
||||||
we enable the SQL sandbox, so changes done to the database
|
|
||||||
are reverted at the end of every test. If you are using
|
|
||||||
PostgreSQL, you can even run database tests asynchronously
|
|
||||||
by setting `use Hello.DataCase, async: true`, although
|
|
||||||
this option is not recommended for other databases.
|
|
||||||
"""
|
|
||||||
|
|
||||||
use ExUnit.CaseTemplate
|
|
||||||
|
|
||||||
using do
|
|
||||||
quote do
|
|
||||||
alias Hello.Repo
|
|
||||||
|
|
||||||
import Ecto
|
|
||||||
import Ecto.Changeset
|
|
||||||
import Ecto.Query
|
|
||||||
import Hello.DataCase
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
setup tags do
|
|
||||||
Hello.DataCase.setup_sandbox(tags)
|
|
||||||
:ok
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Sets up the sandbox based on the test tags.
|
|
||||||
"""
|
|
||||||
def setup_sandbox(tags) do
|
|
||||||
pid = Ecto.Adapters.SQL.Sandbox.start_owner!(Hello.Repo, shared: not tags[:async])
|
|
||||||
on_exit(fn -> Ecto.Adapters.SQL.Sandbox.stop_owner(pid) end)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
A helper that transforms changeset errors into a map of messages.
|
|
||||||
|
|
||||||
assert {:error, changeset} = Accounts.create_user(%{password: "short"})
|
|
||||||
assert "password is too short" in errors_on(changeset).password
|
|
||||||
assert %{password: ["password is too short"]} = errors_on(changeset)
|
|
||||||
|
|
||||||
"""
|
|
||||||
def errors_on(changeset) do
|
|
||||||
Ecto.Changeset.traverse_errors(changeset, fn {message, opts} ->
|
|
||||||
Regex.replace(~r"%{(\w+)}", message, fn _, key ->
|
|
||||||
opts |> Keyword.get(String.to_existing_atom(key), key) |> to_string()
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
ExUnit.start()
|
|
||||||
Ecto.Adapters.SQL.Sandbox.mode(Hello.Repo, :manual)
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
FROM golang:1.22 as build
|
|
||||||
WORKDIR /app
|
|
||||||
COPY . .
|
|
||||||
RUN go build -o /server .
|
|
||||||
|
|
||||||
FROM scratch
|
|
||||||
COPY --from=build /server /server
|
|
||||||
EXPOSE 3000
|
|
||||||
CMD ["/server"]
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
module server
|
|
||||||
|
|
||||||
go 1.22.0
|
|
||||||
|
|
||||||
require github.com/gin-gonic/gin v1.9.1
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/bytedance/sonic v1.9.1 // indirect
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2 // indirect
|
|
||||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
|
||||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
|
||||||
github.com/go-playground/validator/v10 v10.14.0 // indirect
|
|
||||||
github.com/goccy/go-json v0.10.2 // indirect
|
|
||||||
github.com/json-iterator/go v1.1.12 // indirect
|
|
||||||
github.com/klauspost/cpuid/v2 v2.2.4 // indirect
|
|
||||||
github.com/leodido/go-urn v1.2.4 // indirect
|
|
||||||
github.com/mattn/go-isatty v0.0.19 // indirect
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.8 // indirect
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
|
||||||
github.com/ugorji/go/codec v1.2.11 // indirect
|
|
||||||
golang.org/x/arch v0.3.0 // indirect
|
|
||||||
golang.org/x/crypto v0.9.0 // indirect
|
|
||||||
golang.org/x/net v0.10.0 // indirect
|
|
||||||
golang.org/x/sys v0.8.0 // indirect
|
|
||||||
golang.org/x/text v0.9.0 // indirect
|
|
||||||
google.golang.org/protobuf v1.30.0 // indirect
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
|
||||||
)
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM=
|
|
||||||
github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s=
|
|
||||||
github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU=
|
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA=
|
|
||||||
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
|
||||||
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
|
|
||||||
github.com/gin-gonic/gin v1.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg=
|
|
||||||
github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SUcPTeU=
|
|
||||||
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
|
||||||
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
|
||||||
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
|
||||||
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
|
|
||||||
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
|
|
||||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
|
|
||||||
github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js=
|
|
||||||
github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
|
||||||
github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU=
|
|
||||||
github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
|
|
||||||
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
|
|
||||||
github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU=
|
|
||||||
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
|
||||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
|
||||||
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
|
||||||
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
|
|
||||||
github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
|
|
||||||
github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk=
|
|
||||||
github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY=
|
|
||||||
github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q=
|
|
||||||
github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4=
|
|
||||||
github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA=
|
|
||||||
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
|
||||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
|
||||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ=
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
|
||||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
|
||||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
|
||||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
|
||||||
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
|
||||||
github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY=
|
|
||||||
github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
|
||||||
github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU=
|
|
||||||
github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg=
|
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
|
||||||
golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k=
|
|
||||||
golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
|
||||||
golang.org/x/crypto v0.9.0 h1:LF6fAI+IutBocDJ2OT0Q1g8plpYljMZ4+lty+dsqw3g=
|
|
||||||
golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0=
|
|
||||||
golang.org/x/net v0.10.0 h1:X2//UzNDwYmtCLn7To6G58Wr6f5ahEAQgKNzv9Y951M=
|
|
||||||
golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg=
|
|
||||||
golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/sys v0.8.0 h1:EBmGv8NaZBZTWvrbjNoL6HVt+IVy3QDQpJs7VRIw3tU=
|
|
||||||
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE=
|
|
||||||
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
|
||||||
google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng=
|
|
||||||
google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
)
|
|
||||||
|
|
||||||
var db = make(map[string]string)
|
|
||||||
|
|
||||||
func setupRouter() *gin.Engine {
|
|
||||||
// Disable Console Color
|
|
||||||
// gin.DisableConsoleColor()
|
|
||||||
r := gin.Default()
|
|
||||||
|
|
||||||
// Ping test
|
|
||||||
r.GET("/ping", func(c *gin.Context) {
|
|
||||||
c.String(http.StatusOK, "pong")
|
|
||||||
})
|
|
||||||
|
|
||||||
// Get user value
|
|
||||||
r.GET("/user/:name", func(c *gin.Context) {
|
|
||||||
user := c.Params.ByName("name")
|
|
||||||
value, ok := db[user]
|
|
||||||
if ok {
|
|
||||||
c.JSON(http.StatusOK, gin.H{"user": user, "value": value})
|
|
||||||
} else {
|
|
||||||
c.JSON(http.StatusOK, gin.H{"user": user, "status": "no value"})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Authorized group (uses gin.BasicAuth() middleware)
|
|
||||||
// Same than:
|
|
||||||
// authorized := r.Group("/")
|
|
||||||
// authorized.Use(gin.BasicAuth(gin.Credentials{
|
|
||||||
// "foo": "bar",
|
|
||||||
// "manu": "123",
|
|
||||||
//}))
|
|
||||||
authorized := r.Group("/", gin.BasicAuth(gin.Accounts{
|
|
||||||
"foo": "bar", // user:foo password:bar
|
|
||||||
"manu": "123", // user:manu password:123
|
|
||||||
}))
|
|
||||||
|
|
||||||
/* example curl for /admin with basicauth header
|
|
||||||
Zm9vOmJhcg== is base64("foo:bar")
|
|
||||||
|
|
||||||
curl -X POST \
|
|
||||||
http://localhost:8080/admin \
|
|
||||||
-H 'authorization: Basic Zm9vOmJhcg==' \
|
|
||||||
-H 'content-type: application/json' \
|
|
||||||
-d '{"value":"bar"}'
|
|
||||||
*/
|
|
||||||
authorized.POST("admin", func(c *gin.Context) {
|
|
||||||
user := c.MustGet(gin.AuthUserKey).(string)
|
|
||||||
|
|
||||||
// Parse JSON
|
|
||||||
var json struct {
|
|
||||||
Value string `json:"value" binding:"required"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.Bind(&json) == nil {
|
|
||||||
db[user] = json.Value
|
|
||||||
c.JSON(http.StatusOK, gin.H{"status": "ok"})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
r := setupRouter()
|
|
||||||
// Listen and Server in 0.0.0.0:8080
|
|
||||||
r.Run(":3000")
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
APP_NAME=Laravel
|
|
||||||
APP_ENV=local
|
|
||||||
APP_KEY=
|
|
||||||
APP_DEBUG=true
|
|
||||||
APP_URL=http://localhost
|
|
||||||
|
|
||||||
APP_LOCALE=en
|
|
||||||
APP_FALLBACK_LOCALE=en
|
|
||||||
APP_FAKER_LOCALE=en_US
|
|
||||||
|
|
||||||
APP_MAINTENANCE_DRIVER=file
|
|
||||||
# APP_MAINTENANCE_STORE=database
|
|
||||||
|
|
||||||
PHP_CLI_SERVER_WORKERS=4
|
|
||||||
|
|
||||||
BCRYPT_ROUNDS=12
|
|
||||||
|
|
||||||
LOG_CHANNEL=stack
|
|
||||||
LOG_STACK=single
|
|
||||||
LOG_DEPRECATIONS_CHANNEL=null
|
|
||||||
LOG_LEVEL=debug
|
|
||||||
|
|
||||||
DB_CONNECTION=sqlite
|
|
||||||
# DB_HOST=127.0.0.1
|
|
||||||
# DB_PORT=3306
|
|
||||||
# DB_DATABASE=laravel
|
|
||||||
# DB_USERNAME=root
|
|
||||||
# DB_PASSWORD=
|
|
||||||
|
|
||||||
SESSION_DRIVER=database
|
|
||||||
SESSION_LIFETIME=120
|
|
||||||
SESSION_ENCRYPT=false
|
|
||||||
SESSION_PATH=/
|
|
||||||
SESSION_DOMAIN=null
|
|
||||||
|
|
||||||
BROADCAST_CONNECTION=log
|
|
||||||
FILESYSTEM_DISK=local
|
|
||||||
QUEUE_CONNECTION=database
|
|
||||||
|
|
||||||
CACHE_STORE=database
|
|
||||||
# CACHE_PREFIX=
|
|
||||||
|
|
||||||
MEMCACHED_HOST=127.0.0.1
|
|
||||||
|
|
||||||
REDIS_CLIENT=phpredis
|
|
||||||
REDIS_HOST=127.0.0.1
|
|
||||||
REDIS_PASSWORD=null
|
|
||||||
REDIS_PORT=6379
|
|
||||||
|
|
||||||
MAIL_MAILER=log
|
|
||||||
MAIL_SCHEME=null
|
|
||||||
MAIL_HOST=127.0.0.1
|
|
||||||
MAIL_PORT=2525
|
|
||||||
MAIL_USERNAME=null
|
|
||||||
MAIL_PASSWORD=null
|
|
||||||
MAIL_FROM_ADDRESS="hello@example.com"
|
|
||||||
MAIL_FROM_NAME="${APP_NAME}"
|
|
||||||
|
|
||||||
AWS_ACCESS_KEY_ID=
|
|
||||||
AWS_SECRET_ACCESS_KEY=
|
|
||||||
AWS_DEFAULT_REGION=us-east-1
|
|
||||||
AWS_BUCKET=
|
|
||||||
AWS_USE_PATH_STYLE_ENDPOINT=false
|
|
||||||
|
|
||||||
VITE_APP_NAME="${APP_NAME}"
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user