Outbound email reachable from scripts as email::send(#{...}) (plain
text) and email::send_html(#{...}) (multipart text + HTML). Backed by a
lettre SMTP relay configured from PICLOUD_SMTP_HOST/PORT/USER/PASSWORD/
TLS/TIMEOUT_SECS; if HOST/USER/PASSWORD aren't all set the service runs
in disabled mode (every send throws NotConfigured, warned at startup).
- EmailService trait + OutboundEmail DTO (picloud-shared);
EmailServiceImpl + EmailTransport seam + lettre transport
(manager-core), wired into the Services bundle and Rhai engine.
- Capability::AppEmailSend (→ script:write); seven-scope commitment held.
- Required-field + RFC5322-ish address validation; 25 MB per-message cap
(PICLOUD_EMAIL_MAX_MESSAGE_BYTES). reply_to defaults to from.
- Per-call connection (pooling deferred to v1.2); no per-app from
validation (operator's SMTP/SPF/DKIM concern).
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
164 lines
6.2 KiB
Rust
164 lines
6.2 KiB
Rust
//! Control plane: script storage, scheduling, configuration.
|
|
//!
|
|
//! Single-writer to Postgres. The orchestrator may *read* scripts from
|
|
//! the same DB for now; once we add caching and per-node ingress, the
|
|
//! manager will publish change events.
|
|
|
|
pub mod abandoned_repo;
|
|
pub mod admin_session_repo;
|
|
pub mod admin_user_repo;
|
|
pub mod admin_users_api;
|
|
pub mod api;
|
|
pub mod api_key_repo;
|
|
pub mod api_keys_api;
|
|
pub mod app_bootstrap;
|
|
pub mod app_domain_repo;
|
|
pub mod app_members_api;
|
|
pub mod app_members_repo;
|
|
pub mod app_repo;
|
|
pub mod app_secrets_repo;
|
|
pub mod apps_api;
|
|
pub mod auth;
|
|
pub mod auth_api;
|
|
pub mod auth_bootstrap;
|
|
pub mod auth_middleware;
|
|
pub mod authz;
|
|
pub mod cron_scheduler;
|
|
pub mod dead_letter_repo;
|
|
pub mod dead_letter_service;
|
|
pub mod dead_letters_api;
|
|
pub mod dispatcher;
|
|
pub mod docs_filter;
|
|
pub mod docs_repo;
|
|
pub mod docs_service;
|
|
pub mod email_service;
|
|
pub mod files_api;
|
|
pub mod files_repo;
|
|
pub mod files_service;
|
|
pub mod files_sweep;
|
|
pub mod gc;
|
|
pub mod http_service;
|
|
pub mod kv_repo;
|
|
pub mod kv_service;
|
|
pub mod log_sink;
|
|
pub mod migrations;
|
|
pub mod module_source;
|
|
pub mod outbox_event_emitter;
|
|
pub mod outbox_repo;
|
|
pub mod principal_resolver;
|
|
pub mod pubsub_repo;
|
|
pub mod pubsub_service;
|
|
pub mod realtime_authority;
|
|
pub mod repo;
|
|
pub mod route_admin;
|
|
pub mod route_repo;
|
|
pub mod sandbox;
|
|
pub mod scheduler;
|
|
pub mod secrets_api;
|
|
pub mod secrets_repo;
|
|
pub mod secrets_service;
|
|
pub mod ssrf;
|
|
pub mod topic_repo;
|
|
pub mod topics_api;
|
|
pub mod trigger_config;
|
|
pub mod trigger_repo;
|
|
pub mod triggers_api;
|
|
|
|
pub use abandoned_repo::{
|
|
AbandonedRepo, AbandonedRepoError, NewAbandonedExecution, PostgresAbandonedRepo,
|
|
};
|
|
pub use admin_session_repo::{
|
|
AdminSessionLookup, AdminSessionRepository, AdminSessionRepositoryError,
|
|
PostgresAdminSessionRepository,
|
|
};
|
|
pub use admin_user_repo::{
|
|
AdminUserCredentials, AdminUserRepository, AdminUserRepositoryError, AdminUserRow,
|
|
PostgresAdminUserRepository,
|
|
};
|
|
pub use admin_users_api::{admins_router, AdminsState};
|
|
pub use api::{admin_router, AdminState};
|
|
pub use api_key_repo::{
|
|
ApiKeyRepository, ApiKeyRepositoryError, ApiKeyRow, ApiKeyVerification, NewApiKey,
|
|
PostgresApiKeyRepository,
|
|
};
|
|
pub use api_keys_api::{api_keys_router, ApiKeysState};
|
|
pub use app_bootstrap::{seed_hello_world_if_fresh, HelloWorldOutcome};
|
|
pub use app_domain_repo::{AppDomainRepository, NewAppDomain, PostgresAppDomainRepository};
|
|
pub use app_members_api::{app_members_router, AppMembersApiError, AppMembersState};
|
|
pub use app_members_repo::{
|
|
AppMembersRepository, AppMembersRepositoryError, AppMembershipDetail, AppMembershipRow,
|
|
PostgresAppMembersRepository,
|
|
};
|
|
pub use app_repo::{resolve_app, AppLookup, AppRepository, PostgresAppRepository};
|
|
pub use app_secrets_repo::{
|
|
AppSecretsRepo, AppSecretsRepoError, PostgresAppSecretsRepo, SIGNING_KEY_LEN,
|
|
};
|
|
pub use apps_api::{apps_router, AppsState};
|
|
pub use auth_api::auth_router;
|
|
pub use auth_bootstrap::{
|
|
bootstrap_first_admin, bootstrap_first_admin_with, BootstrapEnv, BootstrapError,
|
|
};
|
|
#[allow(deprecated)]
|
|
pub use auth_middleware::{
|
|
attach_principal_if_present, require_admin, require_authenticated, AuthState, AuthedAdmin,
|
|
API_KEY_PREFIX, API_KEY_PREFIX_LEN, SESSION_COOKIE,
|
|
};
|
|
pub use authz::{can, require, AuthzDenied, AuthzError, AuthzRepo, Capability, Decision};
|
|
pub use cron_scheduler::spawn_cron_scheduler;
|
|
pub use dead_letter_repo::{
|
|
DeadLetterRepo, DeadLetterRepoError, DeadLetterRow, NewDeadLetter, PostgresDeadLetterRepo,
|
|
};
|
|
pub use dead_letter_service::PostgresDeadLetterService;
|
|
pub use dead_letters_api::{dead_letters_router, DeadLettersApiError, DeadLettersState};
|
|
pub use dispatcher::{compute_backoff, Dispatcher, DispatcherError};
|
|
pub use docs_repo::{DocsRepo, DocsRepoError, PostgresDocsRepo};
|
|
pub use docs_service::DocsServiceImpl;
|
|
pub use email_service::{
|
|
EmailConfig, EmailServiceImpl, EmailTransport, LettreEmailTransport, SmtpConfig, SmtpTls,
|
|
DEFAULT_EMAIL_MAX_MESSAGE_BYTES,
|
|
};
|
|
pub use files_api::{files_admin_router, FilesAdminState};
|
|
pub use files_repo::{FilesConfig, FilesRepo, FilesRepoError, FsFilesRepo};
|
|
pub use files_service::FilesServiceImpl;
|
|
pub use files_sweep::{spawn_files_orphan_sweep, sweep_orphan_tmp_files, SweepStats};
|
|
pub use gc::{spawn_abandoned_gc, spawn_dead_letter_gc};
|
|
pub use http_service::{HttpConfig, HttpServiceImpl};
|
|
pub use kv_repo::{KvRepo, KvRepoError, PostgresKvRepo};
|
|
pub use kv_service::KvServiceImpl;
|
|
pub use log_sink::PostgresExecutionLogSink;
|
|
pub use module_source::PostgresModuleSource;
|
|
pub use outbox_event_emitter::OutboxEventEmitter;
|
|
pub use outbox_repo::{
|
|
NewOutboxRow, OutboxRepo, OutboxRepoError, OutboxRow, OutboxSourceKind, PostgresOutboxRepo,
|
|
};
|
|
pub use principal_resolver::{AdminPrincipalResolver, PrincipalResolver, PrincipalResolverError};
|
|
pub use pubsub_repo::{PostgresPubsubRepo, PublishCtx, PubsubRepo, PubsubRepoError};
|
|
pub use pubsub_service::{PubsubServiceImpl, SubscriberTokenConfig};
|
|
pub use realtime_authority::RealtimeAuthorityImpl;
|
|
pub use repo::{
|
|
ExecutionLogRepository, NewScript, PostgresExecutionLogRepository, PostgresScriptRepository,
|
|
RepoResolver, ScriptPatch, ScriptRepository, ScriptRepositoryError,
|
|
};
|
|
pub use route_admin::{compile_routes, route_admin_router, RouteAdminState};
|
|
pub use route_repo::{NewRoute, PostgresRouteRepository, RouteRepository};
|
|
pub use sandbox::{CeilingError, SandboxCeiling};
|
|
pub use secrets_api::{secrets_router, SecretsApiError, SecretsState};
|
|
pub use secrets_repo::{
|
|
PostgresSecretsRepo, SecretMeta, SecretsMetaPage, SecretsNamePage, SecretsRepo,
|
|
SecretsRepoError, StoredSecret,
|
|
};
|
|
pub use secrets_service::{
|
|
open as open_secret, seal as seal_secret, SecretsConfig, SecretsServiceImpl,
|
|
DEFAULT_SECRET_MAX_VALUE_BYTES,
|
|
};
|
|
pub use topic_repo::{PostgresTopicRepo, Topic, TopicAuthMode, TopicRepo, TopicRepoError};
|
|
pub use topics_api::{topics_router, TopicsApiError, TopicsState};
|
|
pub use trigger_config::{BackoffShape, TriggerConfig};
|
|
pub use trigger_repo::{
|
|
collection_matches, CreateDeadLetterTrigger, CreateDocsTrigger, CreateFilesTrigger,
|
|
CreateKvTrigger, CreatePubsubTrigger, DeadLetterTriggerMatch, DocsTriggerMatch,
|
|
FilesTriggerMatch, KvTriggerMatch, PostgresTriggerRepo, Trigger, TriggerDetails,
|
|
TriggerDispatchMode, TriggerKind, TriggerRepo, TriggerRepoError,
|
|
};
|
|
pub use triggers_api::{triggers_router, TriggersApiError, TriggersState};
|