mirror of
https://github.com/immich-app/immich.git
synced 2025-12-22 09:15:34 +03:00
* feat(server): add `react-mail` as mail template engine and `nodemailer` * feat(server): add `smtp` related configs to `SystemConfig` * feat(web): add page for SMTP settings * feat(server): add `react-email.adapter` This adapter render the React-Email into HTML and plain/text email. The output is set as the body of the email. * feat(server): add `MailRepository` and `MailService` Allow to use the NestJS-modules-mailer module to send SMTP emails. This is the base transport for the `NotificationRepository` * feat(server): register the job dispatcher and Job for async email This allows to queue email sending jobs for the `EmailService`. * feat(server): add `NotificationRepository` and `NotificationService` This act as a middleware to properly route the notification to the right transport. As POC I've only implemented a simple SMTP transport. * feat(server): add `welcome` email template * feat(server): add the first notification on `createUser` in `UserService` This trigger an event for the `NotificationRepository` that once processes by using the global config and per-user config will carry the payload to the right notification transport. * chore: clean up * chore: clean up web * fix: type errors" * fix package lock * fix mail sending, option to ignore certs * chore: open api * chore: clean up * remove unused import * feat: email feature flag * chore: remove unused interface * small styling --------- Co-authored-by: Jason Rasmussen <jrasm91@gmail.com> Co-authored-by: Daniel Dietzler <mail@ddietzler.dev> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
41 lines
1.1 KiB
TypeScript
41 lines
1.1 KiB
TypeScript
import { getServerConfig, getServerFeatures, type ServerConfigDto, type ServerFeaturesDto } from '@immich/sdk';
|
|
import { writable } from 'svelte/store';
|
|
|
|
export type FeatureFlags = ServerFeaturesDto & { loaded: boolean };
|
|
|
|
export const featureFlags = writable<FeatureFlags>({
|
|
loaded: false,
|
|
smartSearch: true,
|
|
facialRecognition: true,
|
|
sidecar: true,
|
|
map: true,
|
|
reverseGeocoding: true,
|
|
search: true,
|
|
oauth: false,
|
|
oauthAutoLaunch: false,
|
|
passwordLogin: true,
|
|
configFile: false,
|
|
trash: true,
|
|
email: false,
|
|
});
|
|
|
|
export type ServerConfig = ServerConfigDto & { loaded: boolean };
|
|
|
|
export const serverConfig = writable<ServerConfig>({
|
|
loaded: false,
|
|
oauthButtonText: '',
|
|
loginPageMessage: '',
|
|
trashDays: 30,
|
|
userDeleteDelay: 7,
|
|
isInitialized: false,
|
|
isOnboarded: false,
|
|
externalDomain: '',
|
|
});
|
|
|
|
export const loadConfig = async () => {
|
|
const [flags, config] = await Promise.all([getServerFeatures(), getServerConfig()]);
|
|
|
|
featureFlags.update(() => ({ ...flags, loaded: true }));
|
|
serverConfig.update(() => ({ ...config, loaded: true }));
|
|
};
|