Module: Studio
- Defined in:
- lib/studio.rb,
lib/studio/s3.rb,
lib/studio/email.rb,
lib/studio/engine.rb,
lib/studio/version.rb,
lib/studio/link_token.rb,
app/models/studio/link.rb,
lib/studio/color_scale.rb,
lib/studio/email_smoke.rb,
lib/studio/image_cache.rb,
lib/studio/ui_primitives.rb,
lib/studio/mail_transport.rb,
lib/studio/theme_resolver.rb,
lib/studio/username_generator.rb,
app/services/studio/email_image.rb,
app/models/studio/email_delivery.rb,
app/jobs/studio/email_delivery_job.rb,
app/controllers/studio/links_controller.rb,
app/controllers/concerns/studio/admin_models.rb,
app/helpers/studio/admin_models_table_helper.rb,
app/controllers/concerns/studio/impersonation.rb,
app/controllers/concerns/studio/error_handling.rb,
app/controllers/studio/email_images_controller.rb,
app/controllers/studio/local_emails_controller.rb,
app/controllers/concerns/studio/link_consumption.rb
Defined Under Namespace
Modules: AdminModels, AdminModelsTableHelper, ColorScale, Email, EmailImage, ErrorHandling, ImageCache, Impersonation, LinkConsumption, LinkToken, S3, UiPrimitives Classes: EmailDelivery, EmailDeliveryJob, EmailImagesController, EmailSmoke, Engine, Link, LinksController, LocalEmailsController, MailTransport, S3ConfigError, ThemeResolver, UserContractError, UsernameGenerator
Constant Summary collapse
- REQUIRED_USER_INSTANCE_METHODS =
Only methods that consumers must explicitly define are checked here. Column accessors (#email, #name, #role) are NOT validated because ActiveRecord defines them lazily — they don’t appear on ‘.instance_methods` until the schema is introspected (typically first record access). Missing columns are caught by the User table schema, not by this validator.
%i[admin? display_name].freeze
- REQUIRED_USER_CLASS_METHODS =
%i[find_by].freeze
- PASSWORD_USER_INSTANCE_METHODS =
#authenticate is only required when email+password sign-in is enabled. Passwordless apps (the default) never call it.
%i[authenticate].freeze
- VERSION =
"0.8.0"
Class Method Summary collapse
-
.auth_method?(method) ⇒ Boolean
True when the given sign-in method is enabled for this app.
- .configure {|_self| ... } ⇒ Object
- .env_truthy?(value) ⇒ Boolean
- .env_value(env, key) ⇒ Object
- .local_email_capture? ⇒ Boolean
-
.logo_for(title) ⇒ Object
Find a logo from theme_logos by title, with fallback chain: 1.
-
.magic_link_via_l_route? ⇒ Boolean
True when the emailed/inbox magic-link URL is the short /l/<token> — i.e.
- .mailer_from_for_transport(env: ENV, ses_from:, resend_from: nil) ⇒ Object
- .marketing_from_for_transport(env: ENV, ses_from:, resend_from: nil) ⇒ Object
- .routes(router) ⇒ Object
- .ses_transport_ready?(env = ENV) ⇒ Boolean
- .theme_config ⇒ Object
- .user_wallet_address(user) ⇒ Object
-
.validate_user_contract!(user_class) ⇒ Object
Verifies that the host app’s User model satisfies the engine’s expected contract.
Class Method Details
.auth_method?(method) ⇒ Boolean
True when the given sign-in method is enabled for this app.
149 150 151 |
# File 'lib/studio.rb', line 149 def self.auth_method?(method) auth_methods.include?(method.to_sym) end |
.configure {|_self| ... } ⇒ Object
114 115 116 |
# File 'lib/studio.rb', line 114 def self.configure yield self end |
.env_truthy?(value) ⇒ Boolean
240 241 242 |
# File 'lib/studio.rb', line 240 def self.env_truthy?(value) %w[1 true yes on].include?(value.to_s.strip.downcase) end |
.env_value(env, key) ⇒ Object
143 144 145 146 |
# File 'lib/studio.rb', line 143 def self.env_value(env, key) value = env[key] value if value && !value.to_s.strip.empty? end |
.local_email_capture? ⇒ Boolean
162 163 164 165 166 167 |
# File 'lib/studio.rb', line 162 def self.local_email_capture? return false if defined?(Rails) && Rails.respond_to?(:env) && Rails.env.production? return !!local_email_capture unless local_email_capture.nil? env_truthy?(ENV["LOCAL_EMAIL_CAPTURE"]) || env_truthy?(ENV["AGENT_WORKTREE"]) end |
.logo_for(title) ⇒ Object
Find a logo from theme_logos by title, with fallback chain:
-
Exact title match
-
“Navbar Logo” fallback
-
First logo in the list
232 233 234 235 236 237 238 |
# File 'lib/studio.rb', line 232 def self.logo_for(title) logos = theme_logos.map { |l| l.is_a?(Hash) ? l : { file: l, title: l } } entry = logos.find { |l| l[:title] == title } entry ||= logos.find { |l| l[:title] == "Navbar Logo" } entry ||= logos.first entry ? "/#{entry[:file]}" : nil end |
.magic_link_via_l_route? ⇒ Boolean
True when the emailed/inbox magic-link URL is the short /l/<token> — i.e. magic links are Studio::Link rows AND this app draws the /l routes. False = the legacy /magic_link/<token> path: the :signed store, OR an app on the :database store that keeps its own /magic_link route (e.g. turf-monster, whose /l is already its landing-page namespace).
158 159 160 |
# File 'lib/studio.rb', line 158 def self.magic_link_via_l_route? magic_link_store == :database && draw_link_routes end |
.mailer_from_for_transport(env: ENV, ses_from:, resend_from: nil) ⇒ Object
118 119 120 121 122 123 124 |
# File 'lib/studio.rb', line 118 def self.mailer_from_for_transport(env: ENV, ses_from:, resend_from: nil) if ses_transport_ready?(env) env_value(env, "MAILER_FROM") || ses_from else env_value(env, "RESEND_MAILER_FROM") || resend_from || resend_mailer_from end end |
.marketing_from_for_transport(env: ENV, ses_from:, resend_from: nil) ⇒ Object
126 127 128 129 130 131 132 133 134 135 |
# File 'lib/studio.rb', line 126 def self.marketing_from_for_transport(env: ENV, ses_from:, resend_from: nil) if ses_transport_ready?(env) env_value(env, "MARKETING_MAILER_FROM") || ses_from else env_value(env, "RESEND_MARKETING_FROM") || env_value(env, "RESEND_MAILER_FROM") || resend_from || resend_mailer_from end end |
.routes(router) ⇒ Object
244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 |
# File 'lib/studio.rb', line 244 def self.routes(router) router.instance_exec do get "login", to: "sessions#new" post "login", to: "sessions#create" post "sso_continue", to: "sessions#sso_continue" get "sso_login", to: "sessions#sso_login" get "logout", to: "sessions#destroy" get "signup", to: "registrations#new" post "signup", to: "registrations#create" get "auth/:provider/callback", to: "omniauth_callbacks#create" get "auth/failure", to: "omniauth_callbacks#failure" unless defined?(Rails) && Rails.env.production? get "_studio/local_emails", to: "studio/local_emails#index", as: :studio_local_emails end # Passwordless email (magic link). Helpers: magic_link_request_path (POST # to request a link), magic_link_path(token) / magic_link_url(token:) # for the emailed GET confirmation page, and magic_link_consume_path(token) # for the scanner-safe POST consume. The token is a URL-safe # MessageVerifier blob but the constraint guards against a stray "." # segment. if Studio.draw_auth_routes && Studio.auth_method?(:magic_link) post "magic_link", to: "magic_links#create", as: :magic_link_request get "magic_link/:token", to: "magic_links#confirm", as: :magic_link, constraints: { token: %r{[^/]+} } post "magic_link/:token", to: "magic_links#consume", as: :magic_link_consume, constraints: { token: %r{[^/]+} } end # Unified short-token links — /l/<token> for magic sign-in links + referral # links (Studio::Link). Studio::LinksController dispatches by kind: a # magic_link renders the scanner-safe confirm interstitial then POSTs to # consume; a referral captures attribution + redirects. Helpers: link_path # / link_url(token:) and link_consume_path. Drawn for every consumer # (including draw_auth_routes=false apps) unless draw_link_routes is off. if Studio.draw_link_routes get "l/:token", to: "studio/links#show", as: :link, constraints: { token: %r{[^/]+} } post "l/:token", to: "studio/links#consume", as: :link_consume, constraints: { token: %r{[^/]+} } end # Solana / Phantom wallet sign-in (nonce challenge + signature verify). # The browser posts to these literal paths from the shared Connect-Wallet # flow; app-specific surfaces (mobile deep-link callback, account-linking, # OAuth popup) stay in the consuming app's routes. if Studio.draw_auth_routes && Studio.auth_method?(:wallet) get "auth/solana/nonce", to: "solana_sessions#nonce", as: :solana_nonce post "auth/solana/verify", to: "solana_sessions#verify", as: :solana_verify end resources :error_logs, only: [:index, :show] # Admin get "admin/theme", to: "theme_settings#edit", as: :admin_theme patch "admin/theme", to: "theme_settings#update", as: :admin_theme_update post "admin/theme/regenerate", to: "theme_settings#regenerate", as: :admin_theme_regenerate get "admin/schema", to: "schema#index", as: :admin_schema # Admin-managed transactional-email banner images (Studio::EmailImage). # index lists each managed email variant + its current banner; update # uploads a replacement. Surfaced from each app's admin hub. get "admin/email_images", to: "studio/email_images#index", as: :admin_email_images patch "admin/email_images/:variant", to: "studio/email_images#update", as: :admin_email_image, constraints: { variant: /[a-z_]+/ } end end |
.ses_transport_ready?(env = ENV) ⇒ Boolean
137 138 139 140 141 |
# File 'lib/studio.rb', line 137 def self.ses_transport_ready?(env = ENV) env["MAIL_TRANSPORT"].to_s.downcase == "ses" && env_value(env, "SES_SMTP_USERNAME") && env_value(env, "SES_SMTP_PASSWORD") end |
.theme_config ⇒ Object
216 217 218 219 220 221 222 223 224 225 226 |
# File 'lib/studio.rb', line 216 def self.theme_config { primary: theme_primary, dark: theme_dark, light: theme_light, success: theme_success, warning: theme_warning, danger: theme_danger, accent: theme_accent }.compact end |
.user_wallet_address(user) ⇒ Object
169 170 171 172 173 174 175 176 177 178 179 180 |
# File 'lib/studio.rb', line 169 def self.user_wallet_address(user) return nil unless user [wallet_address_method, :wallet_address, :solana_address].compact.each do |method| next unless user.respond_to?(method) value = user.public_send(method) return value if value && !(value.respond_to?(:empty?) && value.empty?) end nil end |
.validate_user_contract!(user_class) ⇒ Object
Verifies that the host app’s User model satisfies the engine’s expected contract. Raises Studio::UserContractError with a clear pointer to docs/USER_CONTRACT.md if anything required is missing. Called from Engine#after_initialize. Opt out via Studio.validate_user_contract = false.
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 |
# File 'lib/studio.rb', line 186 def self.validate_user_contract!(user_class) return unless validate_user_contract return unless user_class.is_a?(Class) missing = [] REQUIRED_USER_CLASS_METHODS.each do |m| missing << "User.#{m}" unless user_class.respond_to?(m) end instance_methods = REQUIRED_USER_INSTANCE_METHODS.dup instance_methods.concat(PASSWORD_USER_INSTANCE_METHODS) if auth_method?(:password) instance_methods.each do |m| missing << "User##{m}" unless user_class.instance_methods.include?(m) end return if missing.empty? raise UserContractError, <<~MSG The studio-engine gem's expected User model contract is not satisfied. Missing: #{missing.join(", ")} See the USER_CONTRACT.md doc in the studio-engine repo for the full contract + a minimal compliant example: https://github.com/amcritchie/studio-engine/blob/main/docs/USER_CONTRACT.md To bypass this check temporarily, set Studio.validate_user_contract = false in config/initializers/studio.rb. MSG end |