Merge branch 'develop'

This commit is contained in:
Alice Gaudon 2021-11-20 19:09:26 +01:00
commit 2879e014a8
18 changed files with 163 additions and 89 deletions

View File

@ -1,6 +1,6 @@
{ {
"name": "swaf", "name": "swaf",
"version": "0.24.3", "version": "0.24.4",
"description": "Structure Web Application Framework.", "description": "Structure Web Application Framework.",
"repository": "https://eternae.ink/ashpie/swaf", "repository": "https://eternae.ink/ashpie/swaf",
"author": "Alice Gaudon <alice@gaudon.pro>", "author": "Alice Gaudon <alice@gaudon.pro>",
@ -52,7 +52,6 @@
"@typescript-eslint/eslint-plugin": "^5.3.0", "@typescript-eslint/eslint-plugin": "^5.3.0",
"@typescript-eslint/parser": "^5.3.0", "@typescript-eslint/parser": "^5.3.0",
"chokidar": "^3.5.1", "chokidar": "^3.5.1",
"clear-module": "^4.1.1",
"concurrently": "^6.0.0", "concurrently": "^6.0.0",
"eslint": "^8.2.0", "eslint": "^8.2.0",
"eslint-plugin-import": "^2.22.1", "eslint-plugin-import": "^2.22.1",
@ -66,7 +65,6 @@
"maildev": "^1.1.0", "maildev": "^1.1.0",
"node-fetch": "^3.0.0", "node-fetch": "^3.0.0",
"nodemon": "^2.0.6", "nodemon": "^2.0.6",
"normalize.css": "^8.0.1",
"require-from-string": "^2.0.2", "require-from-string": "^2.0.2",
"rollup": "^2.42.3", "rollup": "^2.42.3",
"rollup-plugin-css-only": "^3.1.0", "rollup-plugin-css-only": "^3.1.0",
@ -76,14 +74,13 @@
"rollup-plugin-terser": "^7.0.2", "rollup-plugin-terser": "^7.0.2",
"sass": "^1.32.12", "sass": "^1.32.12",
"supertest": "^6.0.0", "supertest": "^6.0.0",
"svelte": "^3.35.0",
"svelte-check": "^2.2.8", "svelte-check": "^2.2.8",
"svelte-preprocess": "4.6.9",
"ts-jest": "^27.0.7", "ts-jest": "^27.0.7",
"typescript": "^4.0.2" "typescript": "^4.0.2"
}, },
"dependencies": { "dependencies": {
"argon2": "^0.28.2", "argon2": "^0.28.2",
"clear-module": "^4.1.1",
"compression": "^1.7.4", "compression": "^1.7.4",
"config": "^3.3.1", "config": "^3.3.1",
"connect-flash": "^0.1.1", "connect-flash": "^0.1.1",
@ -97,9 +94,12 @@
"mysql": "^2.18.1", "mysql": "^2.18.1",
"nanoid": "^3.1.20", "nanoid": "^3.1.20",
"nodemailer": "^6.4.6", "nodemailer": "^6.4.6",
"normalize.css": "^8.0.1",
"nunjucks": "^3.2.1", "nunjucks": "^3.2.1",
"on-finished": "^2.3.0", "on-finished": "^2.3.0",
"redis": "^3.0.2", "redis": "^3.0.2",
"svelte": "^3.35.0",
"svelte-preprocess": "4.6.9",
"ts-node": "^10.4.0", "ts-node": "^10.4.0",
"tslog": "^3.0.1", "tslog": "^3.0.1",
"uuid": "^8.0.0", "uuid": "^8.0.0",

View File

@ -16,3 +16,7 @@ fs.mkdirSync('dist/types', {recursive: true});
fs.readdirSync('src/types').forEach(file => { fs.readdirSync('src/types').forEach(file => {
copyRecursively(path.join('src/types', file), 'dist/types'); copyRecursively(path.join('src/types', file), 'dist/types');
}); });
fs.readdirSync('src/assets').forEach(file => {
copyRecursively(path.join('src/assets', file), 'dist/assets');
});

View File

@ -18,10 +18,5 @@ if (!fs.existsSync(symlink)) {
fs.symlinkSync(path.resolve('dist/common'), symlink); fs.symlinkSync(path.resolve('dist/common'), symlink);
} }
// Copy assets
fs.readdirSync('src/assets').forEach(file => {
copyRecursively(path.join('src/assets', file), 'dist/assets');
});
// Copy package.json // Copy package.json
fs.copyFileSync('package.json', 'dist/package.json'); fs.copyFileSync('package.json', 'dist/package.json');

View File

@ -3,16 +3,18 @@
import Message from "../../components/Message.svelte"; import Message from "../../components/Message.svelte";
import Form from "../../utils/Form.svelte"; import Form from "../../utils/Form.svelte";
import Field from "../../utils/Field.svelte"; import Field from "../../utils/Field.svelte";
import {hasRoute, route} from "../../../../common/Routing";
let newName = ''; let newName = '';
</script> </script>
{#if hasRoute('change-name')}
<section class="panel"> <section class="panel">
<h2><i data-feather="key"></i> Change name</h2> <h2><i data-feather="key"></i> Change name</h2>
{#if $locals.can_change_name} {#if $locals.can_change_name}
<Form action={$locals.route('change-name')} <Form action={route('change-name')}
submitIcon="save" submitText="Change my name {newName.length > 0 ? 'to ' + newName : ''}" submitIcon="save" submitText="Change my name {newName.length > 0 ? 'to ' + newName : ''}"
confirm="Are you sure you want to change your name to {newName}?"> confirm="Are you sure you want to change your name to {newName}?">
<Field type="text" name="name" icon="user" placeholder="New name" required bind:value={newName}/> <Field type="text" name="name" icon="user" placeholder="New name" required bind:value={newName}/>
@ -29,3 +31,4 @@
<Message type="info" content="You will be able to change your name in {$locals.can_change_name_in}" sticky discreet/> <Message type="info" content="You will be able to change your name in {$locals.can_change_name_in}" sticky discreet/>
{/if} {/if}
</section> </section>
{/if}

View File

@ -2,21 +2,23 @@
import {locals} from "../../../ts/stores"; import {locals} from "../../../ts/stores";
import Form from "../../utils/Form.svelte"; import Form from "../../utils/Form.svelte";
import Field from "../../utils/Field.svelte"; import Field from "../../utils/Field.svelte";
import {hasRoute, route} from "../../../../common/Routing";
let removePasswordMode = false; let removePasswordMode = false;
</script> </script>
{#if hasRoute('remove-password', 'change-password')}
<section class="panel"> <section class="panel">
<h2><i data-feather="key"></i> {$locals.has_password ? 'Change' : 'Set'} password</h2> <h2><i data-feather="key"></i> {$locals.has_password ? 'Change' : 'Set'} password</h2>
{#if removePasswordMode} {#if removePasswordMode}
<Form action={$locals.route('remove-password')} <Form action={route('remove-password')}
submitIcon="trash" submitText="Remove password" submitClass="danger" submitIcon="trash" submitText="Remove password" submitClass="danger"
confirm="Are you sure you want to remove your password?"> confirm="Are you sure you want to remove your password?">
<button type="button" on:click={() => removePasswordMode = false}>Go back</button> <button type="button" on:click={() => removePasswordMode = false}>Go back</button>
</Form> </Form>
{:else} {:else}
<Form action={$locals.route('change-password')} <Form action={route('change-password')}
submitIcon="save" submitText="Set password"> submitIcon="save" submitText="Set password">
{#if $locals.has_password} {#if $locals.has_password}
<Field type="password" name="current_password" icon="key" placeholder="Current password"/> <Field type="password" name="current_password" icon="key" placeholder="Current password"/>
@ -28,3 +30,4 @@
</Form> </Form>
{/if} {/if}
</section> </section>
{/if}

View File

@ -6,6 +6,7 @@
import PasswordPanel from "./PasswordPanel.svelte"; import PasswordPanel from "./PasswordPanel.svelte";
import Form from "../../utils/Form.svelte"; import Form from "../../utils/Form.svelte";
import Field from "../../utils/Field.svelte"; import Field from "../../utils/Field.svelte";
import {route} from "../../../../common/Routing";
const mainEmail = $locals.main_email?.email; const mainEmail = $locals.main_email?.email;
const personalInfoFields = $locals.user_personal_info_fields || []; const personalInfoFields = $locals.user_personal_info_fields || [];
@ -68,13 +69,13 @@
<td>Secondary</td> <td>Secondary</td>
<td>{email.email}</td> <td>{email.email}</td>
<td class="actions"> <td class="actions">
<Form action={$locals.route('set-main-email')} button <Form action={route('set-main-email')} button
submitIcon="refresh-ccw" submitText="Set as main address" submitClass="warning" submitIcon="refresh-ccw" submitText="Set as main address" submitClass="warning"
confirm="Are you sure you want to set {email.email} as your main address?"> confirm="Are you sure you want to set {email.email} as your main address?">
<Field type="hidden" name="id" value={email.id}/> <Field type="hidden" name="id" value={email.id}/>
</Form> </Form>
<Form action={$locals.route('remove-email')} button <Form action={route('remove-email')} button
submitIcon="trash" submitText="Remove" submitClass="danger" submitIcon="trash" submitText="Remove" submitClass="danger"
confirm="Are you sure you want to delete {email.email}?"> confirm="Are you sure you want to delete {email.email}?">
<Field type="hidden" name="id" value={email.id}/> <Field type="hidden" name="id" value={email.id}/>
@ -87,7 +88,7 @@
</table> </table>
</div> </div>
<Form action={$locals.route('add-email')} class="sub-panel" <Form action={route('add-email')} class="sub-panel"
submitIcon="plus" submitText="Add email address"> submitIcon="plus" submitText="Add email address">
<h3>Add an email address:</h3> <h3>Add an email address:</h3>

View File

@ -4,6 +4,7 @@
import Form from "../utils/Form.svelte"; import Form from "../utils/Form.svelte";
import Field from "../utils/Field.svelte"; import Field from "../utils/Field.svelte";
import Icon from "../utils/Icon.svelte"; import Icon from "../utils/Icon.svelte";
import {hasRoute, route} from "../../../common/Routing";
let registerUsingMagicLink = $locals.previousFormData()?.['auth_method'] !== 'password'; let registerUsingMagicLink = $locals.previousFormData()?.['auth_method'] !== 'password';
let loginUsingMagicLink = true; let loginUsingMagicLink = true;
@ -22,10 +23,11 @@
h1="Authentication and registration"> h1="Authentication and registration">
<div class="container"> <div class="container">
{#if hasRoute('login')}
<section class="panel"> <section class="panel">
<h2><i data-feather="log-in"></i> Log in</h2> <h2><i data-feather="log-in"></i> Log in</h2>
<Form action={$locals.route('login') + queryStr} submitText="Authenticate" submitIcon="log-in"> <Form action={route('login') + queryStr} submitText="Authenticate" submitIcon="log-in">
<Field type="text" name="identifier" value={$locals.query?.identifier} icon="at-sign" <Field type="text" name="identifier" value={$locals.query?.identifier} icon="at-sign"
hint={loginUsingMagicLink ? 'You will receive a magic link in your mailbox. Click on the link from any device to authenticate here.' : ''} hint={loginUsingMagicLink ? 'You will receive a magic link in your mailbox. Click on the link from any device to authenticate here.' : ''}
placeholder="Your email address or username" required/> placeholder="Your email address or username" required/>
@ -39,11 +41,13 @@
<Field type="checkbox" name="persist_session" icon="clock" placeholder="Stay logged in on this computer."/> <Field type="checkbox" name="persist_session" icon="clock" placeholder="Stay logged in on this computer."/>
</Form> </Form>
</section> </section>
{/if}
{#if hasRoute('register')}
<section class="panel"> <section class="panel">
<h2><i data-feather="user-plus"></i> Register</h2> <h2><i data-feather="user-plus"></i> Register</h2>
<Form action={$locals.route('register') + queryStr} submitText="Register" submitIcon="check"> <Form action={route('register') + queryStr} submitText="Register" submitIcon="check">
<Field type="hidden" name="auth_method" value={registerUsingMagicLink ? 'magic_link': 'password'}/> <Field type="hidden" name="auth_method" value={registerUsingMagicLink ? 'magic_link': 'password'}/>
{#if $locals.has_username} {#if $locals.has_username}
@ -68,5 +72,6 @@
</Field> </Field>
</Form> </Form>
</section> </section>
{/if}
</div> </div>
</BaseLayout> </BaseLayout>

View File

@ -5,7 +5,7 @@
import Form from "../utils/Form.svelte"; import Form from "../utils/Form.svelte";
import Field from "../utils/Field.svelte"; import Field from "../utils/Field.svelte";
import Breadcrumb from "../components/Breadcrumb.svelte"; import Breadcrumb from "../components/Breadcrumb.svelte";
import {route} from "../../../common/Routing"; import {hasRoute, route} from "../../../common/Routing";
const accounts = $locals.accounts || []; const accounts = $locals.accounts || [];
</script> </script>
@ -49,16 +49,20 @@
<td><time datetime={user.created_at_iso}>{user.created_at_human} ago</time></td> <td><time datetime={user.created_at_iso}>{user.created_at_human} ago</time></td>
<td> <td>
<div class="max-content"> <div class="max-content">
<Form action={$locals.route('approve-account')} {#if hasRoute('approve-account')}
<Form action={route('approve-account')}
submitIcon="check" submitText="Approve" submitClass="success"> submitIcon="check" submitText="Approve" submitClass="success">
<Field type="hidden" name="user_id" value={user.id}/> <Field type="hidden" name="user_id" value={user.id}/>
</Form> </Form>
{/if}
<Form action={$locals.route('reject-account')} {#if hasRoute('reject-account')}
<Form action={route('reject-account')}
submitIcon="trash" submitText="Reject" submitClass="danger" submitIcon="trash" submitText="Reject" submitClass="danger"
confirm="This will irrevocably delete the {user.mainEmailStr || user.name || user.id} account."> confirm="This will irrevocably delete the {user.mainEmailStr || user.name || user.id} account.">
<Field type="hidden" name="user_id" value={user.id}/> <Field type="hidden" name="user_id" value={user.id}/>
</Form> </Form>
{/if}
</div> </div>
</td> </td>
</tr> </tr>

View File

@ -1,6 +1,6 @@
<script lang="ts"> <script lang="ts">
import {locals} from "../ts/stores"; import {locals} from "../ts/stores";
import {route} from "../../common/Routing"; import {route, hasRoute, hasAnyRoute} from "../../common/Routing";
import BaseLayout from "./layouts/BaseLayout.svelte"; import BaseLayout from "./layouts/BaseLayout.svelte";
</script> </script>
@ -9,11 +9,17 @@
<h1>swaf - Svelte Web Application Framework</h1> <h1>swaf - Svelte Web Application Framework</h1>
<p>Welcome to {$locals.app.name}!</p> <p>Welcome to {$locals.app.name}!</p>
{#if hasAnyRoute('tests', 'design')}
<nav> <nav>
<ul> <ul>
{#if hasRoute('tests')}
<li><a href={route('tests')}>Frontend tests</a></li> <li><a href={route('tests')}>Frontend tests</a></li>
{/if}
{#if hasRoute('design')}
<li><a href={route('design')}>Design test</a></li> <li><a href={route('design')}>Design test</a></li>
{/if}
</ul> </ul>
</nav> </nav>
{/if}
</div> </div>
</BaseLayout> </BaseLayout>

View File

@ -1,9 +1,7 @@
<script lang="ts"> <script lang="ts">
import {locals} from "../../ts/stores.js";
import {route} from "../../../common/Routing.js";
import FlashMessages from "../components/FlashMessages.svelte"; import FlashMessages from "../components/FlashMessages.svelte";
import NavMenuItem from "../components/NavMenuItem.svelte"; import BaseFooter from "./base/BaseFooter.svelte";
import NavMenu from "../components/NavMenu.svelte"; import BaseHeader from "./base/BaseHeader.svelte";
export let title: string; export let title: string;
export let h1: string = title; export let h1: string = title;
@ -15,57 +13,11 @@
@import "../../scss/vars"; @import "../../scss/vars";
@import "../../scss/helpers"; @import "../../scss/helpers";
header {
@if $headerContainer {
@include container;
}
display: flex;
flex-direction: row;
justify-content: space-between;
align-items: center;
margin-bottom: 16px;
padding: 0;
height: $headerHeight;
@include medium-le {
z-index: 1;
position: sticky;
top: 0;
flex-direction: row-reverse;
@include surface(3);
}
.logo {
display: flex;
flex-direction: row;
align-items: center;
padding: 0 8px;
font-size: 24px;
img {
flex-shrink: 0;
width: initial;
height: calc(#{$headerHeight} - 16px);
margin-right: 8px;
padding: 8px;
}
}
}
main { main {
@include container; @include container;
flex-grow: 1; flex-grow: 1;
} }
footer {
padding: 8px;
text-align: center;
}
.flash-messages { .flash-messages {
@include container; @include container;
} }
@ -91,21 +43,7 @@
<link rel="stylesheet" href="/css/layout.css"> <link rel="stylesheet" href="/css/layout.css">
</svelte:head> </svelte:head>
<header> <BaseHeader/>
<a href="/" class="logo"><img src="/img/logo.svg" alt="{$locals.app.name} logo"> {$locals.app.name}</a>
<NavMenu>
{#if $locals.user}
{#if $locals.user.is_admin}
<NavMenuItem href={route('backend')} icon="settings" text="Backend"/>
{/if}
<NavMenuItem href={route('account')} icon="user" text={$locals.user.name || 'Account'}/>
<NavMenuItem href={route('logout')} icon="log-out" text="Logout" action/>
{:else}
<NavMenuItem href={route('auth')} icon="log-in" text="Log in / Register"/>
{/if}
</NavMenu>
</header>
<div class="flash-messages"> <div class="flash-messages">
<FlashMessages/> <FlashMessages/>
@ -123,4 +61,4 @@
<slot/> <slot/>
</main> </main>
<footer>{$locals.app.name} v{$locals.app_version} - all rights reserved.</footer> <BaseFooter/>

View File

@ -0,0 +1,12 @@
<script>
import {locals} from "../../../ts/stores.js";
</script>
<style>
footer {
padding: 8px;
text-align: center;
}
</style>
<footer>{$locals.app.name} v{$locals.app_version} - all rights reserved.</footer>

View File

@ -0,0 +1,43 @@
<script>
import BaseHeaderLogo from "./BaseHeaderLogo.svelte";
import NavMenu from "../../components/NavMenu.svelte";
import BaseNavMenuLinks from "./BaseNavMenuLinks.svelte";
import BaseNavMenuAuth from "./BaseNavMenuAuth.svelte";
</script>
<style lang="scss">
@import "../../../scss/vars";
@import "../../../scss/helpers";
header {
@if $headerContainer {
@include container;
}
display: flex;
flex-direction: row;
justify-content: space-between;
align-items: center;
margin-bottom: 16px;
padding: 0;
height: $headerHeight;
@include medium-le {
z-index: 1;
position: sticky;
top: 0;
flex-direction: row-reverse;
@include surface(3);
}
}
</style>
<header>
<BaseHeaderLogo/>
<NavMenu>
<BaseNavMenuLinks/>
<BaseNavMenuAuth/>
</NavMenu>
</header>

View File

@ -0,0 +1,26 @@
<script>
import {locals} from "../../../ts/stores.js";
</script>
<style lang="scss">
@import "../../../scss/vars";
.logo {
display: flex;
flex-direction: row;
align-items: center;
padding: 0 8px;
font-size: 24px;
img {
flex-shrink: 0;
width: initial;
height: calc(#{$headerHeight} - 16px);
margin-right: 8px;
padding: 8px;
}
}
</style>
<a href="/" class="logo"><img src="/img/logo.svg" alt="{$locals.app.name} logo"> {$locals.app.name}</a>

View File

@ -0,0 +1,18 @@
<script>
import {locals} from "../../../ts/stores.js";
import NavMenuItem from "../../components/NavMenuItem.svelte";
import {hasRoute, route} from "../../../../common/Routing";
</script>
{#if hasRoute('auth')}
{#if $locals.user}
{#if $locals.user.is_admin}
<NavMenuItem href={route('backend')} icon="settings" text="Backend"/>
{/if}
<NavMenuItem href={route('account')} icon="user" text={$locals.user.name || 'Account'}/>
<NavMenuItem href={route('logout')} icon="log-out" text="Logout" action/>
{:else}
<NavMenuItem href={route('auth')} icon="log-in" text="Log in / Register"/>
{/if}
{/if}

View File

@ -66,3 +66,19 @@ export function route(
export function getRouteParamRegExp(key: string, flags?: string): RegExp { export function getRouteParamRegExp(key: string, flags?: string): RegExp {
return new RegExp(`:${key}(\\(.+?\\))?\\??`, flags); return new RegExp(`:${key}(\\(.+?\\))?\\??`, flags);
} }
export function hasRoute(...routesToMatch: string[]): boolean {
for (const route of routesToMatch) {
if (!routes[route]) return false;
}
return true;
}
export function hasAnyRoute(...routesToMatch: string[]): boolean {
for (const route of routesToMatch) {
if (routes[route]) return true;
}
return false;
}

View File

@ -295,7 +295,7 @@ export default class SvelteViewEngine extends ViewEngine {
}); });
// Load locals into locals store // Load locals into locals store
const localsModulePath = "../../intermediates/assets/ts/stores.js"; const localsModulePath = path.resolve(this.targetDir, "../ts/stores.js");
const localsModule = await import(localsModulePath); const localsModule = await import(localsModulePath);
const locals = this.getGlobals().get(); const locals = this.getGlobals().get();
const localMap = this.compileBackendCalls(backendCalls, locals, true); const localMap = this.compileBackendCalls(backendCalls, locals, true);