Optionally skip domain in cookie (#43)
If you're using a reverse proxy like Cloudflare Tunnels, you can now set `CONFIG_SKIP_COOKIE_DOMAIN_SECURITY="true"` to avoid login issues. Also makes some UX tweaks to Expenses, and fixes a style issue for Chrome in Windows (#44). Fixes #43 Fixes #44
This commit is contained in:
@@ -19,3 +19,4 @@ CONFIG_FILES_ROOT_PATH="data-files"
|
|||||||
CONFIG_ENABLE_EMAILS="false" # if true, email verification will be required for signups (using Brevo)
|
CONFIG_ENABLE_EMAILS="false" # if true, email verification will be required for signups (using Brevo)
|
||||||
CONFIG_ENABLE_FOREVER_SIGNUP="true" # if true, all signups become active for 100 years
|
CONFIG_ENABLE_FOREVER_SIGNUP="true" # if true, all signups become active for 100 years
|
||||||
# CONFIG_ALLOWED_COOKIE_DOMAINS="example.com,example.net" # can be set to allow more than the BASE_URL's domain for session cookies
|
# CONFIG_ALLOWED_COOKIE_DOMAINS="example.com,example.net" # can be set to allow more than the BASE_URL's domain for session cookies
|
||||||
|
# CONFIG_SKIP_COOKIE_DOMAIN_SECURITY="true" # if true, the cookie domain will not be strictly set and checked against. This skipping slightly reduces security, but is usually necessary for reverse proxies like Cloudflare Tunnel.
|
||||||
|
|||||||
@@ -85,13 +85,15 @@ export default function BudgetModal(
|
|||||||
<label class='text-slate-300 block pb-1' for='budget_value'>Value</label>
|
<label class='text-slate-300 block pb-1' for='budget_value'>Value</label>
|
||||||
<input
|
<input
|
||||||
class='input-field'
|
class='input-field'
|
||||||
type='text'
|
type='number'
|
||||||
name='budget_value'
|
name='budget_value'
|
||||||
id='budget_value'
|
id='budget_value'
|
||||||
value={newBudgetValue.value}
|
value={newBudgetValue.value}
|
||||||
onInput={(event) => {
|
onInput={(event) => {
|
||||||
newBudgetValue.value = Number(event.currentTarget.value);
|
newBudgetValue.value = Number(event.currentTarget.value);
|
||||||
}}
|
}}
|
||||||
|
min='0'
|
||||||
|
inputmode='decimal'
|
||||||
placeholder='100'
|
placeholder='100'
|
||||||
/>
|
/>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
@@ -100,7 +102,7 @@ export default function BudgetModal(
|
|||||||
{budget
|
{budget
|
||||||
? (
|
? (
|
||||||
<button
|
<button
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-red-600 text-white cursor-pointer rounded-md mr-2'
|
class='px-5 py-2 bg-red-600 text-white cursor-pointer rounded-md mr-2 opacity-30 hover:opacity-100'
|
||||||
onClick={() => onClickDelete()}
|
onClick={() => onClickDelete()}
|
||||||
>
|
>
|
||||||
Delete
|
Delete
|
||||||
@@ -109,16 +111,16 @@ export default function BudgetModal(
|
|||||||
: null}
|
: null}
|
||||||
<button
|
<button
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md mr-2'
|
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md mr-2'
|
||||||
onClick={() => onClickSave(newBudgetName.value, newBudgetMonth.value.substring(0, 7), newBudgetValue.value)}
|
|
||||||
>
|
|
||||||
{budget ? 'Update' : 'Create'}
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md ml-2'
|
|
||||||
onClick={() => onClose()}
|
onClick={() => onClose()}
|
||||||
>
|
>
|
||||||
{budget ? 'Cancel' : 'Close'}
|
{budget ? 'Cancel' : 'Close'}
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
class='px-5 py-2 bg-slate-700 hover:bg-slate-500 text-white cursor-pointer rounded-md ml-2'
|
||||||
|
onClick={() => onClickSave(newBudgetName.value, newBudgetMonth.value.substring(0, 7), newBudgetValue.value)}
|
||||||
|
>
|
||||||
|
{budget ? 'Update' : 'Create'}
|
||||||
|
</button>
|
||||||
</footer>
|
</footer>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -119,6 +119,7 @@ export default function ExpenseModal(
|
|||||||
onInput={(event) => {
|
onInput={(event) => {
|
||||||
newExpenseCost.value = Number(event.currentTarget.value);
|
newExpenseCost.value = Number(event.currentTarget.value);
|
||||||
}}
|
}}
|
||||||
|
inputmode='decimal'
|
||||||
placeholder='10.99'
|
placeholder='10.99'
|
||||||
/>
|
/>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
@@ -222,7 +223,7 @@ export default function ExpenseModal(
|
|||||||
{expense
|
{expense
|
||||||
? (
|
? (
|
||||||
<button
|
<button
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-red-600 text-white cursor-pointer rounded-md mr-2'
|
class='px-5 py-2 bg-red-600 text-white cursor-pointer rounded-md mr-2 opacity-30 hover:opacity-100'
|
||||||
onClick={() => onClickDelete()}
|
onClick={() => onClickDelete()}
|
||||||
>
|
>
|
||||||
Delete
|
Delete
|
||||||
@@ -231,6 +232,12 @@ export default function ExpenseModal(
|
|||||||
: null}
|
: null}
|
||||||
<button
|
<button
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md mr-2'
|
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md mr-2'
|
||||||
|
onClick={() => onClose()}
|
||||||
|
>
|
||||||
|
{expense ? 'Cancel' : 'Close'}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
class='px-5 py-2 bg-slate-700 hover:bg-slate-500 text-white cursor-pointer rounded-md ml-2'
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
onClickSave(
|
onClickSave(
|
||||||
newExpenseCost.value as number,
|
newExpenseCost.value as number,
|
||||||
@@ -243,12 +250,6 @@ export default function ExpenseModal(
|
|||||||
>
|
>
|
||||||
{expense ? 'Update' : 'Create'}
|
{expense ? 'Update' : 'Create'}
|
||||||
</button>
|
</button>
|
||||||
<button
|
|
||||||
class='px-5 py-2 bg-slate-600 hover:bg-slate-500 text-white cursor-pointer rounded-md ml-2'
|
|
||||||
onClick={() => onClose()}
|
|
||||||
>
|
|
||||||
{expense ? 'Cancel' : 'Close'}
|
|
||||||
</button>
|
|
||||||
</footer>
|
</footer>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
|
|||||||
14
lib/auth.ts
14
lib/auth.ts
@@ -6,7 +6,7 @@ import 'std/dotenv/load.ts';
|
|||||||
import { baseUrl, generateHash, isRunningLocally } from './utils/misc.ts';
|
import { baseUrl, generateHash, isRunningLocally } from './utils/misc.ts';
|
||||||
import { User, UserSession } from './types.ts';
|
import { User, UserSession } from './types.ts';
|
||||||
import { createUserSession, deleteUserSession, getUserByEmail, validateUserAndSession } from './data/user.ts';
|
import { createUserSession, deleteUserSession, getUserByEmail, validateUserAndSession } from './data/user.ts';
|
||||||
import { isCookieDomainAllowed } from './config.ts';
|
import { isCookieDomainAllowed, isCookieDomainSecurityDisabled } from './config.ts';
|
||||||
|
|
||||||
const JWT_SECRET = Deno.env.get('JWT_SECRET') || '';
|
const JWT_SECRET = Deno.env.get('JWT_SECRET') || '';
|
||||||
export const PASSWORD_SALT = Deno.env.get('PASSWORD_SALT') || '';
|
export const PASSWORD_SALT = Deno.env.get('PASSWORD_SALT') || '';
|
||||||
@@ -173,6 +173,10 @@ export async function logoutUser(request: Request) {
|
|||||||
domain: resolveCookieDomain(request),
|
domain: resolveCookieDomain(request),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (isCookieDomainSecurityDisabled()) {
|
||||||
|
delete cookie.domain;
|
||||||
|
}
|
||||||
|
|
||||||
const response = new Response('Logged Out', {
|
const response = new Response('Logged Out', {
|
||||||
status: 303,
|
status: 303,
|
||||||
headers: { 'Location': '/', 'Content-Type': 'text/html; charset=utf-8' },
|
headers: { 'Location': '/', 'Content-Type': 'text/html; charset=utf-8' },
|
||||||
@@ -222,6 +226,10 @@ export async function createSessionCookie(
|
|||||||
domain: resolveCookieDomain(request),
|
domain: resolveCookieDomain(request),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (isCookieDomainSecurityDisabled()) {
|
||||||
|
delete cookie.domain;
|
||||||
|
}
|
||||||
|
|
||||||
setCookie(response.headers, cookie);
|
setCookie(response.headers, cookie);
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
@@ -246,6 +254,10 @@ export async function updateSessionCookie(
|
|||||||
domain: resolveCookieDomain(request),
|
domain: resolveCookieDomain(request),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (isCookieDomainSecurityDisabled()) {
|
||||||
|
delete cookie.domain;
|
||||||
|
}
|
||||||
|
|
||||||
setCookie(response.headers, cookie);
|
setCookie(response.headers, cookie);
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
|
|||||||
@@ -30,6 +30,12 @@ export function isCookieDomainAllowed(domain: string) {
|
|||||||
return allowedDomains.includes(domain);
|
return allowedDomains.includes(domain);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function isCookieDomainSecurityDisabled() {
|
||||||
|
const isCookieDomainSecurityDisabled = Deno.env.get('CONFIG_SKIP_COOKIE_DOMAIN_SECURITY') === 'true';
|
||||||
|
|
||||||
|
return isCookieDomainSecurityDisabled;
|
||||||
|
}
|
||||||
|
|
||||||
export function isEmailEnabled() {
|
export function isEmailEnabled() {
|
||||||
const areEmailsAllowed = Deno.env.get('CONFIG_ENABLE_EMAILS') === 'true';
|
const areEmailsAllowed = Deno.env.get('CONFIG_ENABLE_EMAILS') === 'true';
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,11 @@
|
|||||||
@tailwind components;
|
@tailwind components;
|
||||||
@tailwind utilities;
|
@tailwind utilities;
|
||||||
|
|
||||||
|
html {
|
||||||
|
/* This is needed to ensure that the system color scheme is dark and matches the dark theme of the app */
|
||||||
|
color-scheme: dark;
|
||||||
|
}
|
||||||
|
|
||||||
h1 {
|
h1 {
|
||||||
@apply text-4xl font-bold;
|
@apply text-4xl font-bold;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user