module Rodauth

  1. lib/rodauth.rb
  2. lib/rodauth/features/account_expiration.rb
  3. lib/rodauth/features/active_sessions.rb
  4. lib/rodauth/features/argon2.rb
  5. lib/rodauth/features/audit_logging.rb
  6. lib/rodauth/features/base.rb
  7. lib/rodauth/features/change_login.rb
  8. lib/rodauth/features/change_password.rb
  9. lib/rodauth/features/change_password_notify.rb
  10. lib/rodauth/features/close_account.rb
  11. lib/rodauth/features/confirm_password.rb
  12. lib/rodauth/features/create_account.rb
  13. lib/rodauth/features/disallow_common_passwords.rb
  14. lib/rodauth/features/disallow_password_reuse.rb
  15. lib/rodauth/features/email_auth.rb
  16. lib/rodauth/features/email_base.rb
  17. lib/rodauth/features/http_basic_auth.rb
  18. lib/rodauth/features/internal_request.rb
  19. lib/rodauth/features/json.rb
  20. lib/rodauth/features/jwt.rb
  21. lib/rodauth/features/jwt_cors.rb
  22. lib/rodauth/features/jwt_refresh.rb
  23. lib/rodauth/features/lockout.rb
  24. lib/rodauth/features/login.rb
  25. lib/rodauth/features/login_password_requirements_base.rb
  26. lib/rodauth/features/logout.rb
  27. lib/rodauth/features/otp.rb
  28. lib/rodauth/features/otp_lockout_email.rb
  29. lib/rodauth/features/otp_modify_email.rb
  30. lib/rodauth/features/otp_unlock.rb
  31. lib/rodauth/features/password_complexity.rb
  32. lib/rodauth/features/password_expiration.rb
  33. lib/rodauth/features/password_grace_period.rb
  34. lib/rodauth/features/password_pepper.rb
  35. lib/rodauth/features/path_class_methods.rb
  36. lib/rodauth/features/recovery_codes.rb
  37. lib/rodauth/features/remember.rb
  38. lib/rodauth/features/reset_password.rb
  39. lib/rodauth/features/reset_password_notify.rb
  40. lib/rodauth/features/reset_password_verifies_account.rb
  41. lib/rodauth/features/session_expiration.rb
  42. lib/rodauth/features/single_session.rb
  43. lib/rodauth/features/sms_codes.rb
  44. lib/rodauth/features/two_factor_base.rb
  45. lib/rodauth/features/update_password_hash.rb
  46. lib/rodauth/features/verify_account.rb
  47. lib/rodauth/features/verify_account_grace_period.rb
  48. lib/rodauth/features/verify_login_change.rb
  49. lib/rodauth/features/webauthn.rb
  50. lib/rodauth/features/webauthn_autofill.rb
  51. lib/rodauth/features/webauthn_login.rb
  52. lib/rodauth/features/webauthn_modify_email.rb
  53. lib/rodauth/features/webauthn_verify_account.rb
  54. lib/rodauth/migrations.rb
  55. lib/rodauth/version.rb
  56. show all

:nocov:

:nocov:

Methods

Public Class

  1. configure
  2. create_database_authentication_functions
  3. create_database_previous_password_check_functions
  4. drop_database_authentication_functions
  5. drop_database_previous_password_check_functions
  6. lib
  7. load_dependencies
  8. new
  9. version

Public Instance

  1. _account_from_email_auth_key
  2. _account_from_id
  3. _account_from_login
  4. _account_from_refresh_token
  5. _account_from_reset_password_key
  6. _account_from_session
  7. _account_from_unlock_key
  8. _account_from_verify_account_key
  9. _account_from_verify_login_change_key
  10. _account_refresh_token_split
  11. _around_rodauth
  12. _email_auth_request
  13. _field_attributes
  14. _field_error_attributes
  15. _filter_links
  16. _formatted_field_error
  17. _get_remember_cookie
  18. _initialize_instance_variables
  19. _json_response_body
  20. _jwt_decode_opts
  21. _jwt_decode_secrets
  22. _login
  23. _login_form_footer
  24. _login_form_footer_links
  25. _login_response
  26. _merge_fixed_locals_opts
  27. _multi_phase_login_forms
  28. _new_account
  29. _otp
  30. _otp_add_key
  31. _otp_for_key
  32. _otp_interval
  33. _otp_key
  34. _otp_tmp_key
  35. _otp_valid_code?
  36. _otp_valid_code_for_old_secret
  37. _override_webauthn_credential_response_verify
  38. _process_raw_hmac
  39. _recovery_codes
  40. _return_json_response
  41. _set_otp_unlock_info
  42. _set_remember_cookie
  43. _setup_account_lockouts_hash
  44. _sms_request_response
  45. _template_opts
  46. _two_factor_auth_links
  47. _two_factor_auth_response
  48. _two_factor_remove_all_from_session
  49. _two_factor_remove_links
  50. _two_factor_setup_links
  51. _update_login
  52. _view
  53. _view_opts
  54. account!
  55. account_activity_ds
  56. account_ds
  57. account_expired?
  58. account_expired_at
  59. account_from_email_auth_key
  60. account_from_id
  61. account_from_key
  62. account_from_login
  63. account_from_refresh_token
  64. account_from_remember_cookie
  65. account_from_reset_password_key
  66. account_from_session
  67. account_from_unlock_key
  68. account_from_verify_account_key
  69. account_from_verify_login_change_key
  70. account_from_webauthn_login
  71. account_id
  72. account_in_unverified_grace_period?
  73. account_initial_status_value
  74. account_lockouts_ds
  75. account_login_failures_ds
  76. account_password_hash_column
  77. account_session_status_filter
  78. account_table_ds
  79. account_webauthn_ids
  80. account_webauthn_usage
  81. account_webauthn_user_id
  82. active_remember_key_ds
  83. active_sessions_ds
  84. active_sessions_insert_hash
  85. active_sessions_update_hash
  86. add_active_session
  87. add_audit_log
  88. add_field_error_class
  89. add_previous_password_hash
  90. add_recovery_code
  91. add_recovery_codes
  92. add_remember_key
  93. add_webauthn_credential
  94. after_change_password
  95. after_close_account
  96. after_create_account
  97. after_login
  98. after_login_entered_during_multi_phase_login
  99. after_login_failure
  100. after_logout
  101. after_otp_authentication_failure
  102. after_otp_disable
  103. after_otp_setup
  104. after_otp_unlock_auth_failure
  105. after_otp_unlock_auth_success
  106. after_otp_unlock_not_yet_available
  107. after_refresh_token
  108. after_reset_password
  109. after_sms_confirm
  110. after_sms_disable
  111. after_webauthn_remove
  112. after_webauthn_setup
  113. allow_email_auth?
  114. allow_resending_verify_account_email?
  115. already_logged_in
  116. argon2_hash_algorithm?
  117. argon2_hash_cost
  118. argon2_password_hash_match?
  119. argon2_password_hash_using_salt_and_secret
  120. argon2_salt_option
  121. audit_log_ds
  122. audit_log_insert_hash
  123. audit_log_message
  124. audit_log_message_default
  125. audit_log_metadata
  126. auth_class_eval
  127. authenticated?
  128. authenticated_by
  129. authenticated_webauthn_id
  130. auto_add_missing_recovery_codes
  131. auto_remove_recovery_codes
  132. autocomplete_for_field?
  133. autologin_session
  134. autologin_type
  135. base32_encode
  136. base_url
  137. before_change_login_route
  138. before_change_password_route
  139. before_create_account_route
  140. before_login_attempt
  141. before_logout
  142. before_otp_setup
  143. before_otp_setup_route
  144. before_reset_password
  145. before_reset_password_request
  146. before_rodauth
  147. before_sms_confirm
  148. before_two_factor_auth_route
  149. before_two_factor_manage_route
  150. before_unlock_account
  151. before_unlock_account_request
  152. before_verify_account
  153. before_verify_account_route
  154. before_view_recovery_codes
  155. before_webauthn_auth_route
  156. before_webauthn_login_route
  157. before_webauthn_remove_route
  158. before_webauthn_setup
  159. before_webauthn_setup_route
  160. button
  161. button_fixed_locals
  162. button_opts
  163. can_add_recovery_codes?
  164. catch_error
  165. change_login
  166. change_login_notice_flash
  167. change_login_requires_password?
  168. change_password_requires_password?
  169. check_account_expiration
  170. check_active_session
  171. check_already_logged_in
  172. check_csrf
  173. check_csrf?
  174. check_password_change_allowed
  175. check_session_expiration
  176. check_single_session
  177. clear_cached_otp
  178. clear_invalid_login_attempts
  179. clear_session
  180. clear_tokens
  181. close_account
  182. close_account_requires_password?
  183. compute_hmac
  184. compute_hmacs
  185. compute_old_hmac
  186. compute_raw_hmac
  187. compute_raw_hmac_with_secret
  188. confirm_password
  189. confirm_password_redirect
  190. convert_email_token_key
  191. convert_response_header_key
  192. convert_session_key
  193. convert_timestamp
  194. convert_token_id
  195. convert_token_id_to_integer
  196. convert_token_key
  197. create_account_autologin?
  198. create_account_notice_flash
  199. create_account_set_password?
  200. create_email
  201. create_email_auth_key
  202. create_email_to
  203. create_reset_password_key
  204. create_verify_account_key
  205. create_verify_login_change_email
  206. create_verify_login_change_key
  207. csrf_tag
  208. currently_active_session?
  209. database_function_password_match?
  210. db
  211. delete_account
  212. delete_account_on_close?
  213. disable_remember_login
  214. domain
  215. email_auth_ds
  216. email_auth_email_link
  217. email_auth_email_recently_sent?
  218. email_auth_key_insert_hash
  219. email_auth_request_form
  220. email_from
  221. email_to
  222. expire_session
  223. extend_remember_deadline
  224. extend_remember_deadline_while_logged_in?
  225. extract_password_hash_cost
  226. features
  227. field_attributes
  228. field_error
  229. field_error_attributes
  230. flash
  231. forget_login
  232. formatted_field_error
  233. function_name
  234. generate_active_sessions_key
  235. generate_email_auth_key_value
  236. generate_refresh_token
  237. generate_remember_key_value
  238. generate_reset_password_key_value
  239. generate_unlock_account_key
  240. generate_verify_account_key_value
  241. generate_verify_login_change_key_value
  242. get_active_refresh_token
  243. get_activity_timestamp
  244. get_email_auth_email_last_sent
  245. get_email_auth_key
  246. get_password_changed_at
  247. get_password_hash
  248. get_password_reset_key
  249. get_remember_key
  250. get_reset_password_email_last_sent
  251. get_unlock_account_email_last_sent
  252. get_unlock_account_key
  253. get_verify_account_email_last_sent
  254. get_verify_account_key
  255. get_verify_login_change_login_and_key
  256. handle_duplicate_active_session_id
  257. handle_webauthn_sign_count_verification_error
  258. has_password?
  259. hmac_secret_rotation?
  260. hook_action
  261. http_basic_auth
  262. inactive_session_cond
  263. include_success_messages?
  264. input_field_string
  265. inputmode_for_field?
  266. internal_request?
  267. internal_request_configuration
  268. invalid_login_attempted
  269. invalid_previous_password_message
  270. json_request?
  271. json_response
  272. json_response_error?
  273. jwt_cors_allow?
  274. jwt_payload
  275. jwt_refresh_token_account_ds
  276. jwt_refresh_token_account_token_ds
  277. jwt_refresh_token_ds
  278. jwt_refresh_token_insert_hash
  279. jwt_refresh_token_match?
  280. jwt_secret
  281. jwt_session_hash
  282. jwt_token
  283. last_account_activity_at
  284. last_account_login_at
  285. load_memory
  286. loaded_templates
  287. locked_out?
  288. logged_in_via_remember_key?
  289. login
  290. login_confirm_label
  291. login_confirmation_matches?
  292. login_does_not_meet_requirements_message
  293. login_failed_reset_password_request_form
  294. login_field_autocomplete_value
  295. login_form_footer_links
  296. login_hidden_field
  297. login_input_type
  298. login_meets_email_requirements?
  299. login_meets_length_requirements?
  300. login_meets_requirements?
  301. login_param_value
  302. login_required
  303. login_return_to_requested_location_path
  304. login_session
  305. login_too_long_message
  306. login_too_many_bytes_message
  307. login_too_short_message
  308. login_uses_email?
  309. login_valid_email?
  310. logout
  311. logout_additional_form_tags
  312. modifications_require_password?
  313. new_account
  314. new_recovery_code
  315. new_webauthn_credential
  316. no_longer_active_session
  317. normalize_login
  318. normalize_session_or_flash_key
  319. null_byte_parameter_value
  320. only_json?
  321. open_account?
  322. otp_add_key
  323. otp_available?
  324. otp_exists?
  325. otp_hmac_old_secret
  326. otp_hmac_secret
  327. otp_issuer
  328. otp_key_ds
  329. otp_keys_use_hmac?
  330. otp_last_use
  331. otp_locked_out?
  332. otp_lockout_redirect
  333. otp_new_secret
  334. otp_provisioning_name
  335. otp_provisioning_uri
  336. otp_qr_code
  337. otp_record_authentication_failure
  338. otp_remove
  339. otp_remove_auth_failures
  340. otp_tmp_key
  341. otp_unlock_auth_failure
  342. otp_unlock_auth_success
  343. otp_unlock_available?
  344. otp_unlock_data
  345. otp_unlock_deadline
  346. otp_unlock_deadline_passed?
  347. otp_unlock_ds
  348. otp_unlock_next_auth_attempt_after
  349. otp_unlock_not_available_set_refresh_header
  350. otp_unlock_num_successes
  351. otp_unlock_refresh_tag
  352. otp_unlock_reset
  353. otp_unlock_success_cooldown_seconds
  354. otp_update_last_use
  355. otp_user_key
  356. otp_valid_code?
  357. otp_valid_key?
  358. over_max_bytesize_param_value
  359. param
  360. param_or_nil
  361. password_confirm_label
  362. password_does_not_contain_null_byte?
  363. password_does_not_meet_requirements_message
  364. password_doesnt_match_previous_password?
  365. password_expiration_ds
  366. password_expired?
  367. password_field_autocomplete_value
  368. password_has_enough_character_groups?
  369. password_has_no_invalid_pattern?
  370. password_hash
  371. password_hash_cost
  372. password_hash_ds
  373. password_hash_match?
  374. password_hash_using_salt
  375. password_match?
  376. password_meets_length_requirements?
  377. password_meets_requirements?
  378. password_not_in_dictionary?
  379. password_not_one_of_the_most_common?
  380. password_not_too_many_repeating_characters?
  381. password_one_of_most_common?
  382. password_recently_entered?
  383. password_reset_ds
  384. password_too_long_message
  385. password_too_many_bytes_message
  386. password_too_short_message
  387. possible_authentication_methods
  388. post_configure
  389. previous_password_ds
  390. raises_uniqueness_violation?
  391. random_key
  392. raw_param
  393. recovery_code_match?
  394. recovery_codes_available?
  395. recovery_codes_ds
  396. recovery_codes_primary?
  397. recovery_codes_remove
  398. redirect
  399. remember_key_ds
  400. remember_login
  401. remembered_session_id
  402. remove_active_session
  403. remove_all_active_sessions
  404. remove_all_active_sessions_except_current
  405. remove_all_active_sessions_except_for
  406. remove_all_webauthn_keys_and_user_ids
  407. remove_current_session
  408. remove_email_auth_key
  409. remove_inactive_sessions
  410. remove_jwt_refresh_token_key
  411. remove_lockout_metadata
  412. remove_remember_key
  413. remove_reset_password_key
  414. remove_session_value
  415. remove_verify_account_key
  416. remove_verify_login_change_key
  417. remove_webauthn_key
  418. render
  419. render_multi_phase_login_forms
  420. request
  421. require_account
  422. require_account_session
  423. require_authentication
  424. require_current_password
  425. require_http_basic_auth
  426. require_login
  427. require_login_confirmation?
  428. require_login_redirect
  429. require_otp_setup
  430. require_password_authentication
  431. require_password_authentication?
  432. require_response
  433. require_sms_available
  434. require_sms_not_setup
  435. require_sms_setup
  436. require_two_factor_authenticated
  437. require_two_factor_not_authenticated
  438. require_two_factor_setup
  439. require_webauthn_setup
  440. rescue_jwt_payload
  441. reset_password_account_status_value
  442. reset_password_email_link
  443. reset_password_email_recently_sent?
  444. reset_password_key_insert_hash
  445. reset_password_request_for_unverified_account
  446. reset_single_session_key
  447. response
  448. retry_on_uniqueness_violation
  449. return_json_response
  450. return_response
  451. route!
  452. route_path
  453. route_url
  454. save_account
  455. send_email
  456. send_verify_login_change_email
  457. serialize_audit_log_metadata
  458. session
  459. session_inactivity_deadline_condition
  460. session_jwt
  461. session_lifetime_deadline_condition
  462. session_value
  463. set_deadline_value
  464. set_deadline_values?
  465. set_email_auth_email_last_sent
  466. set_error_flash
  467. set_error_reason
  468. set_expired
  469. set_field_error
  470. set_http_basic_auth_error_response
  471. set_jwt
  472. set_jwt_refresh_token_hmac_session_key
  473. set_jwt_token
  474. set_last_password_entry
  475. set_login_requirement_error_message
  476. set_new_account_password
  477. set_notice_flash
  478. set_notice_now_flash
  479. set_password
  480. set_password_requirement_error_message
  481. set_redirect_error_flash
  482. set_redirect_error_status
  483. set_remember_cookie
  484. set_reset_password_email_last_sent
  485. set_response_error_reason_status
  486. set_response_error_status
  487. set_response_header
  488. set_session_value
  489. set_single_session_key
  490. set_title
  491. set_unlock_account_email_last_sent
  492. set_verify_account_email_last_sent
  493. setup_account_verification
  494. show_lockout_page
  495. show_otp_auth_link?
  496. single_session_ds
  497. skip_login_field_on_login?
  498. skip_password_field_on_login?
  499. skip_status_checks?
  500. sms
  501. sms_auth_message
  502. sms_available?
  503. sms_code
  504. sms_code_issued_at
  505. sms_code_match?
  506. sms_codes_primary?
  507. sms_confirm
  508. sms_confirm_failure
  509. sms_confirm_message
  510. sms_confirmation_match?
  511. sms_current_auth?
  512. sms_disable
  513. sms_ds
  514. sms_failures
  515. sms_locked_out?
  516. sms_needs_confirmation?
  517. sms_needs_confirmation_notice_flash
  518. sms_new_auth_code
  519. sms_new_confirm_code
  520. sms_normalize_phone
  521. sms_phone
  522. sms_record_failure
  523. sms_remove_expired_confirm_code
  524. sms_remove_failures
  525. sms_send
  526. sms_send_auth_code
  527. sms_send_confirm_code
  528. sms_set_code
  529. sms_setup
  530. sms_setup?
  531. sms_valid_phone?
  532. split_token
  533. template_path
  534. throw_basic_auth_error
  535. throw_error
  536. throw_error_reason
  537. throw_error_status
  538. throw_rodauth_error
  539. timing_safe_eql?
  540. token_link
  541. token_param_value
  542. transaction
  543. translate
  544. two_factor_auth_links
  545. two_factor_authenticate
  546. two_factor_authenticated?
  547. two_factor_authentication_setup?
  548. two_factor_login_type_match?
  549. two_factor_modifications_require_password?
  550. two_factor_partially_authenticated?
  551. two_factor_password_match?
  552. two_factor_remove
  553. two_factor_remove_auth_failures
  554. two_factor_remove_links
  555. two_factor_remove_session
  556. two_factor_setup_links
  557. two_factor_update_session
  558. unique_constraint_violation_class
  559. unlock_account
  560. unlock_account_email_link
  561. unlock_account_email_recently_sent?
  562. unverified_grace_period_expired?
  563. update_account
  564. update_activity
  565. update_current_session?
  566. update_hash_ds
  567. update_last_activity
  568. update_last_login
  569. update_login
  570. update_password_changed_at
  571. update_password_hash?
  572. update_session
  573. update_single_session_key
  574. update_sms
  575. use_database_authentication_functions?
  576. use_date_arithmetic?
  577. use_json?
  578. use_jwt?
  579. use_multi_phase_login?
  580. use_request_specific_csrf_tokens?
  581. use_scope_clear_session?
  582. uses_instance_variables
  583. uses_two_factor_authentication?
  584. valid_jwt?
  585. valid_login_entered?
  586. valid_new_webauthn_credential?
  587. valid_webauthn_credential_auth?
  588. verified_account?
  589. verify_account
  590. verify_account_check_already_logged_in
  591. verify_account_ds
  592. verify_account_email_link
  593. verify_account_email_recently_sent?
  594. verify_account_email_resend
  595. verify_account_key_insert_hash
  596. verify_account_set_password?
  597. verify_account_view
  598. verify_login_change
  599. verify_login_change_ds
  600. verify_login_change_email_body
  601. verify_login_change_email_link
  602. verify_login_change_key_insert_hash
  603. verify_login_change_old_login
  604. view
  605. webauthn_account_id
  606. webauthn_allow
  607. webauthn_auth_additional_form_tags
  608. webauthn_auth_credential_from_form_submission
  609. webauthn_auth_data
  610. webauthn_auth_form_path
  611. webauthn_authenticator_selection
  612. webauthn_create_relying_party_opts
  613. webauthn_credential_options_for_get
  614. webauthn_extensions
  615. webauthn_form_submission_call
  616. webauthn_key_insert_hash
  617. webauthn_keys_ds
  618. webauthn_login_options?
  619. webauthn_login_verification_factor?
  620. webauthn_origin
  621. webauthn_relying_party
  622. webauthn_remove_authenticated_session
  623. webauthn_rp_id
  624. webauthn_rp_name
  625. webauthn_setup?
  626. webauthn_setup_credential_from_form_submission
  627. webauthn_setup_data
  628. webauthn_update_session
  629. webauthn_user_ids_ds
  630. webauthn_user_name
  631. webauthn_user_verification

Constants

FEATURES = {}  
INVALID_DOMAIN = "invalidurl @@.com"  
MAJOR = 2  

The major version of Rodauth, updated only for major changes that are likely to require modification to apps using Rodauth.

MINOR = 44  

The minor version of Rodauth, updated for new feature releases of Rodauth.

TINY = 0  

The patch version of Rodauth, updated only for bug fixes from the last feature release.

VERSION = "#{MAJOR}.#{MINOR}.#{TINY}".freeze  

The full version of Rodauth as a string

VERSION_NUMBER = MAJOR*10000 + MINOR*100 + TINY  

The full version of Rodauth as a number (1.17.0 => 11700)

Public Instance Aliases

account_session_value -> account_id
ignore_uniqueness_violation -> raises_uniqueness_violation?

If you just want to ignore uniqueness violations, this alias makes more sense.

logged_in? -> session_value
raised_uniqueness_violation -> raises_uniqueness_violation?

If you would like to operate/reraise the exception, this alias makes more sense.

webauthn_get_relying_party_opts -> webauthn_create_relying_party_opts

Public Class methods

configure(app, opts={}, &block)
[show source]
   # File lib/rodauth.rb
48 def self.configure(app, opts={}, &block)
49   json_opt = app.opts[:rodauth_json] = opts.fetch(:json, app.opts[:rodauth_json])
50   csrf = app.opts[:rodauth_csrf] = opts.fetch(:csrf, app.opts[:rodauth_csrf])
51   app.opts[:rodauth_route_csrf] = case csrf
52   when false, :rack_csrf
53     false
54   else
55     json_opt != :only
56   end
57   auth_class = (app.opts[:rodauths] ||= {})[opts[:name]] ||= opts[:auth_class] || Class.new(Auth)
58   if !auth_class.roda_class
59     auth_class.roda_class = app
60   elsif auth_class.roda_class != app
61     auth_class = app.opts[:rodauths][opts[:name]] = Class.new(auth_class)
62     auth_class.roda_class = app
63   end
64   auth_class.class_eval{@configuration_name = opts[:name] unless defined?(@configuration_name)}
65   auth_class.configure(&block) if block
66   auth_class.send(:make_shape_friendly)
67   auth_class.allocate.post_configure if auth_class.method_defined?(:post_configure)
68 end
create_database_authentication_functions(db, opts={})
[show source]
    # File lib/rodauth/migrations.rb
  4   def self.create_database_authentication_functions(db, opts={})
  5     table_name = opts[:table_name] || :account_password_hashes
  6     get_salt_name = opts[:get_salt_name] || :rodauth_get_salt
  7     valid_hash_name = opts[:valid_hash_name] || :rodauth_valid_password_hash
  8     argon2 = opts[:argon2]
  9 
 10     case db.database_type
 11     when :postgres
 12       search_path = opts[:search_path] || 'public, pg_temp'
 13       primary_key_type =
 14         case db.schema(table_name).find { |row| row.first == :id }[1][:db_type]
 15         when 'uuid' then :uuid
 16         else :int8
 17         end
 18       table_name = db.literal(table_name) unless table_name.is_a?(String)
 19 
 20       argon_sql = <<END
 21 CASE
 22     WHEN password_hash ~ '^\\$argon2id'
 23       THEN substring(password_hash from '\\$argon2id\\$v=\\d+\\$m=\\d+,t=\\d+,p=\\d+\\$.+\\$')
 24     ELSE substr(password_hash, 0, 30)
 25   END INTO salt
 26 END
 27       db.run <<END
 28 CREATE OR REPLACE FUNCTION #{get_salt_name}(acct_id #{primary_key_type}) RETURNS text AS $$
 29 DECLARE salt text;
 30 BEGIN
 31 SELECT
 32 #{argon2 ? argon_sql : "substr(password_hash, 0, 30) INTO salt"}
 33 FROM #{table_name}
 34 WHERE acct_id = id;
 35 RETURN salt;
 36 END;
 37 $$ LANGUAGE plpgsql
 38 SECURITY DEFINER
 39 SET search_path = #{search_path};
 40 END
 41 
 42       db.run <<END
 43 CREATE OR REPLACE FUNCTION #{valid_hash_name}(acct_id #{primary_key_type}, hash text) RETURNS boolean AS $$
 44 DECLARE valid boolean;
 45 BEGIN
 46 SELECT password_hash = hash INTO valid 
 47 FROM #{table_name}
 48 WHERE acct_id = id;
 49 RETURN valid;
 50 END;
 51 $$ LANGUAGE plpgsql
 52 SECURITY DEFINER
 53 SET search_path = #{search_path};
 54 END
 55     when :mysql
 56       argon_sql = <<END
 57 CASE
 58   WHEN password_hash REGEXP '^.argon2id'
 59     THEN left(password_hash, CHAR_LENGTH(password_hash) - INSTR(REVERSE(password_hash), '$'))
 60   ELSE substr(password_hash, 1, 30)
 61   END
 62 END
 63       db.run <<END
 64 CREATE FUNCTION #{get_salt_name}(acct_id int8) RETURNS varchar(255)
 65 SQL SECURITY DEFINER
 66 READS SQL DATA
 67 BEGIN
 68 RETURN (SELECT
 69 #{argon2 ? argon_sql : "substr(password_hash, 1, 30)"}
 70 FROM #{table_name}
 71 WHERE acct_id = id);
 72 END;
 73 END
 74 
 75       db.run <<END
 76 CREATE FUNCTION #{valid_hash_name}(acct_id int8, hash varchar(255)) RETURNS tinyint(1)
 77 SQL SECURITY DEFINER
 78 READS SQL DATA
 79 BEGIN
 80 DECLARE valid tinyint(1);
 81 DECLARE csr CURSOR FOR 
 82 SELECT password_hash = hash
 83 FROM #{table_name}
 84 WHERE acct_id = id;
 85 OPEN csr;
 86 FETCH csr INTO valid;
 87 CLOSE csr;
 88 RETURN valid;
 89 END;
 90 END
 91     when :mssql
 92       argon_sql = <<END
 93 CASE
 94   WHEN password_hash LIKE '[$]argon2id%'
 95     THEN left(password_hash, len(password_hash) - charindex('$', reverse(password_hash)))
 96   ELSE substring(password_hash, 0, 30)
 97   END
 98 END
 99       db.run <<END
100 CREATE FUNCTION #{get_salt_name}(@account_id bigint) RETURNS nvarchar(255)
101 WITH EXECUTE AS OWNER
102 AS
103 BEGIN
104 DECLARE @salt nvarchar(255);
105 SELECT @salt =
106 #{argon2 ? argon_sql : "substring(password_hash, 0, 30)"}
107 FROM #{table_name}
108 WHERE id = @account_id;
109 RETURN @salt;
110 END;
111 END
112 
113       db.run <<END
114 CREATE FUNCTION #{valid_hash_name}(@account_id bigint, @hash nvarchar(255)) RETURNS bit
115 WITH EXECUTE AS OWNER
116 AS
117 BEGIN
118 DECLARE @valid bit;
119 DECLARE @ph nvarchar(255);
120 SELECT @ph = password_hash
121 FROM #{table_name}
122 WHERE id = @account_id;
123 IF(@hash = @ph)
124   SET @valid = 1;
125 ELSE
126   SET @valid = 0
127 RETURN @valid;
128 END;
129 END
130     end
131   end
create_database_previous_password_check_functions(db, opts={})
[show source]
    # File lib/rodauth/migrations.rb
153 def self.create_database_previous_password_check_functions(db, opts={})
154   create_database_authentication_functions(db, {:table_name=>:account_previous_password_hashes, :get_salt_name=>:rodauth_get_previous_salt, :valid_hash_name=>:rodauth_previous_password_hash_match}.merge(opts))
155 end
drop_database_authentication_functions(db, opts={})
[show source]
    # File lib/rodauth/migrations.rb
133 def self.drop_database_authentication_functions(db, opts={})
134   table_name = opts[:table_name] || :account_password_hashes
135   get_salt_name = opts[:get_salt_name] || :rodauth_get_salt
136   valid_hash_name = opts[:valid_hash_name] || :rodauth_valid_password_hash
137 
138   case db.database_type
139   when :postgres
140     primary_key_type =
141       case db.schema(table_name).find { |row| row.first == :id }[1][:db_type]
142       when 'uuid' then :uuid
143       else :int8
144       end
145     db.run "DROP FUNCTION #{get_salt_name}(#{primary_key_type})"
146     db.run "DROP FUNCTION #{valid_hash_name}(#{primary_key_type}, text)"
147   when :mysql, :mssql
148     db.run "DROP FUNCTION #{get_salt_name}"
149     db.run "DROP FUNCTION #{valid_hash_name}"
150   end
151 end
drop_database_previous_password_check_functions(db, opts={})
[show source]
    # File lib/rodauth/migrations.rb
157 def self.drop_database_previous_password_check_functions(db, opts={})
158   drop_database_authentication_functions(db, {:table_name=>:account_previous_password_hashes, :get_salt_name=>:rodauth_get_previous_salt, :valid_hash_name=>:rodauth_previous_password_hash_match}.merge(opts))
159 end
lib(opts={}, &block)
[show source]
   # File lib/rodauth.rb
 8 def self.lib(opts={}, &block) 
 9   require 'roda'
10   c = Class.new(Roda)
11   c.plugin(:rodauth, opts) do
12     enable :internal_request
13     instance_exec(&block)
14   end
15   c.freeze
16   c.rodauth
17 end
load_dependencies(app, opts={}, &_)
[show source]
   # File lib/rodauth.rb
19 def self.load_dependencies(app, opts={}, &_)
20   json_opt = opts.fetch(:json, app.opts[:rodauth_json])
21   if json_opt
22     app.plugin :json
23     app.plugin :json_parser
24   end
25 
26   unless json_opt == :only
27     unless opts[:render] == false
28       require 'tilt/string'
29       app.plugin :render
30     end
31 
32     case opts.fetch(:csrf, app.opts[:rodauth_csrf])
33     when false
34       # nothing
35     when :rack_csrf
36       # :nocov:
37       app.plugin :csrf
38       # :nocov:
39     else
40       app.plugin :route_csrf
41     end
42 
43     app.plugin :flash unless opts[:flash] == false
44     app.plugin :h
45   end
46 end
new(scope)
[show source]
    # File lib/rodauth/features/base.rb
173 def initialize(scope)
174   @scope = scope
175   _initialize_instance_variables
176 end
version()
[show source]
   # File lib/rodauth/version.rb
21 def self.version
22   VERSION
23 end

Public Instance methods

_account_from_email_auth_key(token)
[show source]
    # File lib/rodauth/features/email_auth.rb
238 def _account_from_email_auth_key(token)
239   account_from_key(token, account_open_status_value){|id| get_email_auth_key(id)}
240 end
_account_from_id(id, status_id=nil)
[show source]
    # File lib/rodauth/features/base.rb
845 def _account_from_id(id, status_id=nil)
846   ds = account_ds(id)
847   ds = ds.where(account_status_column=>status_id) if status_id && !skip_status_checks?
848   ds.first
849 end
_account_from_login(login)
[show source]
    # File lib/rodauth/features/base.rb
832 def _account_from_login(login)
833   ds = account_table_ds.where(login_column=>login)
834   ds = ds.select(*account_select) if account_select
835   ds = ds.where(account_status_column=>[account_unverified_status_value, account_open_status_value]) unless skip_status_checks?
836   ds.first
837 end
_account_from_refresh_token(token)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
118 def _account_from_refresh_token(token)
119   id, token_id, key = _account_refresh_token_split(token)
120 
121   unless key &&
122          (id.to_s == session_value.to_s) &&
123          (actual = get_active_refresh_token(id, token_id)) &&
124          (timing_safe_eql?(key, convert_token_key(actual)) || (hmac_secret_rotation? && timing_safe_eql?(key, compute_old_hmac(actual)))) &&
125          jwt_refresh_token_match?(key)
126     return
127   end
128 
129   ds = account_ds(id)
130   ds = ds.where(account_session_status_filter) unless skip_status_checks?
131   ds.first
132 end
_account_from_reset_password_key(token)
[show source]
    # File lib/rodauth/features/reset_password.rb
258 def _account_from_reset_password_key(token)
259   account_from_key(token, reset_password_account_status_value){|id| get_password_reset_key(id)}
260 end
_account_from_session()
[show source]
    # File lib/rodauth/features/base.rb
839 def _account_from_session
840   ds = account_ds(session_value)
841   ds = ds.where(account_session_status_filter) unless skip_status_checks?
842   ds.first
843 end
_account_from_unlock_key(token)
[show source]
    # File lib/rodauth/features/lockout.rb
305 def _account_from_unlock_key(token)
306   account_from_key(token){|id| account_lockouts_ds(id).get(account_lockouts_key_column)}
307 end
_account_from_verify_account_key(token)
[show source]
    # File lib/rodauth/features/verify_account.rb
309 def _account_from_verify_account_key(token)
310   account_from_key(token, account_unverified_status_value){|id| get_verify_account_key(id)}
311 end
_account_from_verify_login_change_key(token)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
212 def _account_from_verify_login_change_key(token)
213   account_from_key(token) do |id|
214     @verify_login_change_new_login, key = get_verify_login_change_login_and_key(id)
215     key
216   end
217 end
_account_refresh_token_split(token)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
134 def _account_refresh_token_split(token)
135   id, token = split_token(token)
136   id = convert_token_id(id)
137   return unless id && token
138 
139   token_id, key = split_token(token)
140   token_id = convert_token_id(token_id)
141   return unless token_id && key
142 
143   [id, token_id, key]
144 end
_around_rodauth()
[show source]
    # File lib/rodauth/features/base.rb
590 def _around_rodauth
591   yield
592 end
_email_auth_request()
[show source]
    # File lib/rodauth/features/email_auth.rb
185 def _email_auth_request
186   if email_auth_email_recently_sent?
187     set_redirect_error_flash email_auth_email_recently_sent_error_flash
188     redirect email_auth_email_recently_sent_redirect
189   end
190 
191   generate_email_auth_key_value
192   transaction do
193     before_email_auth_request
194     create_email_auth_key
195     send_email_auth_email
196     after_email_auth_request
197   end
198 
199   email_auth_email_sent_response
200 end
_field_attributes(field)
[show source]
    # File lib/rodauth/features/base.rb
864 def _field_attributes(field)
865   nil
866 end
_field_error_attributes(field)
[show source]
    # File lib/rodauth/features/base.rb
868 def _field_error_attributes(field)
869   " aria-invalid=\"true\" aria-describedby=\"#{field}_error_message\" "
870 end
_formatted_field_error(field, error)
[show source]
    # File lib/rodauth/features/base.rb
872 def _formatted_field_error(field, error)
873   "<span class=\"#{input_field_error_message_class}\" id=\"#{field}_error_message\">#{error}</span>"
874 end
_initialize_instance_variables()
[show source]
    # File lib/rodauth/features/base.rb
587 def _initialize_instance_variables
588 end
_json_response_body(hash)
[show source]
    # File lib/rodauth/features/json.rb
222 def _json_response_body(hash)
223   request.send(:convert_to_json, hash)
224 end
_jwt_decode_opts()
[show source]
    # File lib/rodauth/features/jwt.rb
106 def _jwt_decode_opts
107   jwt_decode_opts
108 end
_jwt_decode_secrets()
[show source]
    # File lib/rodauth/features/jwt.rb
111 def _jwt_decode_secrets
112   secrets = [jwt_secret, jwt_old_secret]
113   secrets.compact!
114   secrets
115 end
_login(auth_type)
[show source]
    # File lib/rodauth/features/login.rb
178 def _login(auth_type)
179   warn("Deprecated #_login method called, use #login instead.")
180   login(auth_type)
181 end
_login_response()
[show source]
    # File lib/rodauth/features/login.rb
158 def _login_response
159   set_notice_flash login_notice_flash
160   redirect(saved_login_redirect || login_redirect)
161 end
_merge_fixed_locals_opts(opts, fixed_locals)
[show source]
     # File lib/rodauth/features/base.rb
1010 def _merge_fixed_locals_opts(opts, fixed_locals)
1011   if use_template_fixed_locals? && !opts[:locals]
1012     fixed_locals_opts = {default_fixed_locals: fixed_locals}
1013     fixed_locals_opts.merge!(opts[:template_opts]) if opts[:template_opts]
1014     opts[:template_opts] = fixed_locals_opts
1015   end
1016 end
_multi_phase_login_forms()
[show source]
    # File lib/rodauth/features/email_auth.rb
179 def _multi_phase_login_forms
180   forms = super
181   forms << [30, email_auth_request_form, :_email_auth_request] if valid_login_entered? && allow_email_auth?
182   forms
183 end
_new_account(login)
[show source]
    # File lib/rodauth/features/create_account.rb
121 def _new_account(login)
122   acc = {login_column=>login}
123   unless skip_status_checks?
124     acc[account_status_column] = account_initial_status_value
125   end
126   acc
127 end
_otp()
[show source]
    # File lib/rodauth/features/otp.rb
448 def _otp
449   _otp_for_key(otp_user_key)
450 end
_otp_add_key(secret)
[show source]
    # File lib/rodauth/features/otp.rb
433 def _otp_add_key(secret)
434   # Uniqueness errors can't be handled here, as we can't be sure the secret provided
435   # is the same as the current secret.
436   otp_key_ds.insert(otp_keys_id_column=>session_value, otp_keys_column=>secret)
437 end
_otp_for_key(key)
[show source]
    # File lib/rodauth/features/otp.rb
444 def _otp_for_key(key)
445   otp_class.new(key, :issuer=>otp_issuer, :digits=>otp_digits, :interval=>otp_interval)
446 end
_otp_interval()
[show source]
    # File lib/rodauth/features/otp.rb
425 def _otp_interval
426   otp_interval || 30
427 end
_otp_key()
[show source]
    # File lib/rodauth/features/otp.rb
439 def _otp_key
440   @otp_user_key = nil
441   otp_key_ds.get(otp_keys_column) || false
442 end
_otp_tmp_key(secret)
[show source]
    # File lib/rodauth/features/otp.rb
419 def _otp_tmp_key(secret)
420   @otp_tmp_key = true
421   @otp_user_key = nil
422   @otp_key = secret
423 end
_otp_valid_code?(ot_pass, otp)
[show source]
    # File lib/rodauth/features/otp.rb
264 def _otp_valid_code?(ot_pass, otp)
265   return false unless otp_exists?
266   ot_pass = ot_pass.gsub(/\s+/, '')
267   if drift = otp_drift
268     if otp.respond_to?(:verify_with_drift)
269       # :nocov:
270       otp.verify_with_drift(ot_pass, drift)
271       # :nocov:
272     else
273       otp.verify(ot_pass, :drift_behind=>drift, :drift_ahead=>drift)
274     end
275   else
276     otp.verify(ot_pass)
277   end
278 end
_otp_valid_code_for_old_secret()

Called for valid OTP codes for old secrets

[show source]
    # File lib/rodauth/features/otp.rb
430 def _otp_valid_code_for_old_secret
431 end
_override_webauthn_credential_response_verify(webauthn_credential)
[show source]
    # File lib/rodauth/features/webauthn.rb
443 def _override_webauthn_credential_response_verify(webauthn_credential)
444   # no need to override
445 end
_process_raw_hmac(hmac)
[show source]
    # File lib/rodauth/features/base.rb
594 def _process_raw_hmac(hmac)
595   s = [hmac].pack('m')
596   s.chomp!("=\n")
597   s.tr!('+/', '-_')
598   s
599 end
_recovery_codes()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
262 def _recovery_codes
263   recovery_codes_ds.select_map(recovery_codes_column)
264 end
_return_json_response()
[show source]
    # File lib/rodauth/features/json.rb
212 def _return_json_response
213   response.status ||= json_response_error_status if json_response_error?
214   response.headers[convert_response_header_key('content-type')] ||= json_response_content_type
215   return_response _json_response_body(json_response)
216 end
_set_otp_unlock_info()
[show source]
   # File lib/rodauth/features/json.rb
82 def _set_otp_unlock_info
83   if use_json?
84     json_response[:num_successes] = otp_unlock_num_successes
85     json_response[:required_successes] = otp_unlock_auths_required
86     json_response[:next_attempt_after] = otp_unlock_next_auth_attempt_after.to_i
87   end
88 end
_setup_account_lockouts_hash(account_id, key)
[show source]
    # File lib/rodauth/features/lockout.rb
172 def _setup_account_lockouts_hash(account_id, key)
173   hash = {account_lockouts_id_column=>account_id, account_lockouts_key_column=>key}
174   set_deadline_value(hash, account_lockouts_deadline_column, account_lockouts_deadline_interval)
175   hash
176 end
_sms_request_response()
[show source]
    # File lib/rodauth/features/sms_codes.rb
473 def _sms_request_response
474   set_notice_flash sms_request_notice_flash
475   redirect sms_auth_redirect
476 end
_template_opts(opts, page)

Set the template path only if there isn’t an overridden template in the application. Result should replace existing template opts.

[show source]
     # File lib/rodauth/features/base.rb
1020 def _template_opts(opts, page)
1021   opts = scope.send(:find_template, scope.send(:parse_template_opts, page, opts))
1022   unless File.file?(scope.send(:template_path, opts))
1023     opts[:path] = template_path(page)
1024   end
1025   opts
1026 end
_two_factor_auth_response()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
250 def _two_factor_auth_response
251   saved_two_factor_auth_redirect = remove_session_value(two_factor_auth_redirect_session_key)
252   set_notice_flash two_factor_auth_notice_flash
253   redirect(saved_two_factor_auth_redirect || two_factor_auth_redirect)
254 end
_two_factor_remove_all_from_session()
[show source]
    # File lib/rodauth/features/otp.rb
367 def _two_factor_remove_all_from_session
368   two_factor_remove_session('totp')
369   super
370 end
_update_login(login)
[show source]
   # File lib/rodauth/features/change_login.rb
83 def _update_login(login)
84   updated = nil
85   raised = raises_uniqueness_violation?{updated = update_account({login_column=>login}, account_ds.exclude(login_column=>login)) == 1}
86   if raised
87     set_login_requirement_error_message(:already_an_account_with_this_login, already_an_account_with_this_login_message)
88   end
89   change_made = updated && !raised
90   clear_tokens(:change_login) if change_made
91   change_made
92 end
_view(meth, page)
[show source]
     # File lib/rodauth/features/base.rb
1028 def _view(meth, page)
1029   unless scope.respond_to?(meth)
1030     raise ConfigurationError, "attempted to render a built-in view/email template (#{page.inspect}), but rendering is disabled"
1031   end
1032 
1033   scope.send(meth, _view_opts(page))
1034 end
_view_opts(page)
[show source]
     # File lib/rodauth/features/base.rb
1000 def _view_opts(page)
1001   opts = template_opts.dup
1002   _merge_fixed_locals_opts(opts, '(rodauth: self.rodauth)')
1003   opts[:locals] = opts[:locals] ? opts[:locals].dup : {}
1004   opts[:locals][:rodauth] = self
1005   opts[:cache] = cache_templates
1006   opts[:cache_key] = :"rodauth_#{page}"
1007   _template_opts(opts, page)
1008 end
account!()
[show source]
    # File lib/rodauth/features/base.rb
415 def account!
416   account || (session_value && account_from_session)
417 end
account_activity_ds(account_id)
[show source]
    # File lib/rodauth/features/account_expiration.rb
104 def account_activity_ds(account_id)
105   db[account_activity_table].
106     where(account_activity_id_column=>account_id)
107 end
account_ds(id=account_id)
[show source]
    # File lib/rodauth/features/base.rb
884 def account_ds(id=account_id)
885   raise ArgumentError, "invalid account id passed to account_ds" unless id
886   ds = account_table_ds.where(account_id_column=>id)
887   ds = ds.select(*account_select) if account_select
888   ds
889 end
account_expired?()
[show source]
   # File lib/rodauth/features/account_expiration.rb
54 def account_expired?
55   columns = [account_activity_last_activity_column, account_activity_last_login_column, account_activity_expired_column]
56   last_activity, last_login, expired = account_activity_ds(account_id).get(columns)
57   return true if expired
58   timestamp = convert_timestamp(expire_account_on_last_activity? ? last_activity : last_login)
59   return false unless timestamp
60   timestamp < Time.now - expire_account_after
61 end
account_expired_at()
[show source]
   # File lib/rodauth/features/account_expiration.rb
35 def account_expired_at
36   get_activity_timestamp(account_id, account_activity_expired_column)
37 end
account_from_email_auth_key(key)
[show source]
    # File lib/rodauth/features/email_auth.rb
133 def account_from_email_auth_key(key)
134   @account = _account_from_email_auth_key(key)
135 end
account_from_id(id, status_id=nil)
[show source]
    # File lib/rodauth/features/base.rb
423 def account_from_id(id, status_id=nil)
424   @account = _account_from_id(id, status_id)
425 end
account_from_key(token, status_id=nil)
[show source]
   # File lib/rodauth/features/email_base.rb
65 def account_from_key(token, status_id=nil)
66   id, key = split_token(token)
67   id = convert_token_id(id)
68   return unless id && key
69 
70   return unless actual = yield(id)
71 
72   unless (hmac_secret && timing_safe_eql?(key, convert_email_token_key(actual))) ||
73      (hmac_secret_rotation? && timing_safe_eql?(key, compute_old_hmac(actual))) ||
74      ((!hmac_secret || allow_raw_email_token?) && timing_safe_eql?(key, actual))
75     return
76   end
77   _account_from_id(id, status_id)
78 end
account_from_login(login)
[show source]
    # File lib/rodauth/features/base.rb
310 def account_from_login(login)
311   @account = _account_from_login(login)
312 end
account_from_refresh_token(token)
[show source]
   # File lib/rodauth/features/jwt_refresh.rb
90 def account_from_refresh_token(token)
91   @account = _account_from_refresh_token(token)
92 end
account_from_reset_password_key(key)
[show source]
   # File lib/rodauth/features/password_expiration.rb
45 def account_from_reset_password_key(key)
46   if a = super
47     check_password_change_allowed
48   end
49   a
50 end
account_from_session()
[show source]
    # File lib/rodauth/features/base.rb
419 def account_from_session
420   @account = _account_from_session
421 end
account_from_unlock_key(key)
[show source]
    # File lib/rodauth/features/lockout.rb
223 def account_from_unlock_key(key)
224   @account = _account_from_unlock_key(key)
225 end
account_from_verify_account_key(key)
[show source]
    # File lib/rodauth/features/verify_account.rb
198 def account_from_verify_account_key(key)
199   @account = _account_from_verify_account_key(key)
200 end
account_from_verify_login_change_key(key)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
122 def account_from_verify_login_change_key(key)
123   @account = _account_from_verify_login_change_key(key)
124 end
account_from_webauthn_login()
[show source]
   # File lib/rodauth/features/webauthn_autofill.rb
45 def account_from_webauthn_login
46   return super if param_or_nil(login_param)
47 
48   credential_id = webauthn_auth_data["id"]
49   account_id = db[webauthn_keys_table]
50     .where(webauthn_keys_webauthn_id_column => credential_id)
51     .get(webauthn_keys_account_id_column)
52 
53   unless account_id
54     throw_error_reason(:invalid_webauthn_id, invalid_field_error_status, webauthn_auth_param, webauthn_invalid_webauthn_id_message)
55   end
56 
57   account_from_id(account_id)
58 end
account_id()
[show source]
    # File lib/rodauth/features/base.rb
300 def account_id
301   account[account_id_column]
302 end
account_in_unverified_grace_period?()
[show source]
   # File lib/rodauth/features/verify_account_grace_period.rb
85 def account_in_unverified_grace_period?
86   return false unless account!
87   account[account_status_column] == account_unverified_status_value &&
88     verify_account_grace_period &&
89     !verify_account_ds.where(Sequel.date_add(verification_requested_at_column, :seconds=>verify_account_grace_period) > Sequel::CURRENT_TIMESTAMP).empty?
90 end
account_initial_status_value()
[show source]
    # File lib/rodauth/features/base.rb
411 def account_initial_status_value
412   account_open_status_value
413 end
account_lockouts_ds(id=account_id)
[show source]
    # File lib/rodauth/features/lockout.rb
301 def account_lockouts_ds(id=account_id)
302   db[account_lockouts_table].where(account_lockouts_id_column=>id)
303 end
account_login_failures_ds()
[show source]
    # File lib/rodauth/features/lockout.rb
297 def account_login_failures_ds
298   db[account_login_failures_table].where(account_login_failures_id_column=>account_id)
299 end
account_password_hash_column()

If the account_password_hash_column is set, the password hash is verified in ruby, it will not use a database function to do so, it will check the password hash using bcrypt.

[show source]
    # File lib/rodauth/features/base.rb
334 def account_password_hash_column
335   nil
336 end
account_session_status_filter()
[show source]
    # File lib/rodauth/features/base.rb
876 def account_session_status_filter
877   {account_status_column=>account_open_status_value}
878 end
account_table_ds()
[show source]
    # File lib/rodauth/features/base.rb
891 def account_table_ds
892   db[accounts_table]
893 end
account_webauthn_ids()
[show source]
    # File lib/rodauth/features/webauthn.rb
271 def account_webauthn_ids
272   webauthn_keys_ds.select_map(webauthn_keys_webauthn_id_column)
273 end
account_webauthn_usage()
[show source]
    # File lib/rodauth/features/webauthn.rb
275 def account_webauthn_usage
276   webauthn_keys_ds.select_hash(webauthn_keys_webauthn_id_column, webauthn_keys_last_use_column)
277 end
account_webauthn_user_id()
[show source]
    # File lib/rodauth/features/webauthn.rb
279 def account_webauthn_user_id
280   unless webauthn_id = webauthn_user_ids_ds.get(webauthn_user_ids_webauthn_id_column)
281     webauthn_id = WebAuthn.generate_user_id
282     if e = raised_uniqueness_violation do
283           webauthn_user_ids_ds.insert(
284             webauthn_user_ids_account_id_column => webauthn_account_id,
285             webauthn_user_ids_webauthn_id_column => webauthn_id
286           )
287         end
288       # If two requests to create a webauthn user id are sent at the same time and an insert
289       # is attempted for both, one will fail with a unique constraint violation.  In that case
290       # it is safe for the second one to use the webauthn user id inserted by the other request.
291       # If there is still no webauthn user id at this point, then we'll just reraise the
292       # exception.
293       # :nocov:
294       raise e unless webauthn_id = webauthn_user_ids_ds.get(webauthn_user_ids_webauthn_id_column)
295       # :nocov:
296     end
297   end
298 
299   webauthn_id
300 end
active_remember_key_ds(id=account_id)
[show source]
    # File lib/rodauth/features/remember.rb
267 def active_remember_key_ds(id=account_id)
268   remember_key_ds(id).where(Sequel.expr(remember_deadline_column) > Sequel::CURRENT_TIMESTAMP)
269 end
active_sessions_ds()
[show source]
    # File lib/rodauth/features/active_sessions.rb
236 def active_sessions_ds
237   db[active_sessions_table].
238     where(active_sessions_account_id_column=>session_value || account_id)
239 end
active_sessions_insert_hash()
[show source]
    # File lib/rodauth/features/active_sessions.rb
199 def active_sessions_insert_hash
200   {active_sessions_account_id_column => session_value, active_sessions_session_id_column => compute_hmac(active_sessions_key)}
201 end
active_sessions_update_hash()
[show source]
    # File lib/rodauth/features/active_sessions.rb
203 def active_sessions_update_hash
204   h = {active_sessions_last_use_column => Sequel::CURRENT_TIMESTAMP}
205 
206   if hmac_secret_rotation?
207     h[active_sessions_session_id_column] = compute_hmac(session[session_id_session_key])
208   end
209 
210   h
211 end
add_active_session()
[show source]
   # File lib/rodauth/features/active_sessions.rb
70 def add_active_session
71   key = generate_active_sessions_key
72   set_session_value(session_id_session_key, key)
73   if e = raises_uniqueness_violation?{active_sessions_ds.insert(active_sessions_insert_hash)}
74     handle_duplicate_active_session_id(e)
75   end
76   nil
77 end
add_audit_log(account_id, action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
39 def add_audit_log(account_id, action)
40   if hash = audit_log_insert_hash(account_id, action)
41     audit_log_ds.insert(hash)
42   end
43 end
add_field_error_class(field)
[show source]
    # File lib/rodauth/features/base.rb
215 def add_field_error_class(field)
216   if field_error(field)
217     " #{input_field_error_class}"
218   end
219 end
add_previous_password_hash(hash)
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
27 def add_previous_password_hash(hash) 
28   ds = previous_password_ds
29 
30   unless @dont_check_previous_password
31     keep_before = ds.reverse(previous_password_id_column).
32       limit(nil, previous_passwords_to_check).
33       get(previous_password_id_column)
34 
35     if keep_before
36       ds.where(Sequel.expr(previous_password_id_column) <= keep_before).
37         delete
38     end
39   end
40 
41   # This should never raise uniqueness violations, as it uses a serial primary key
42   ds.insert(previous_password_account_id_column=>account_id, previous_password_hash_column=>hash)
43 end
add_recovery_code()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
189 def add_recovery_code
190   # This should never raise uniqueness violations unless the recovery code is the same, and the odds of that
191   # are 1/256**32 assuming a good random number generator.  Still, attempt to handle that case by retrying
192   # on such a uniqueness violation.
193   retry_on_uniqueness_violation do
194     recovery_codes_ds.insert(recovery_codes_id_column=>session_value, recovery_codes_column=>new_recovery_code)
195   end
196 end
add_recovery_codes(number)
[show source]
    # File lib/rodauth/features/recovery_codes.rb
179 def add_recovery_codes(number)
180   return if number <= 0
181   transaction do
182     number.times do
183       add_recovery_code
184     end
185   end
186   @recovery_codes = nil
187 end
add_remember_key()
[show source]
    # File lib/rodauth/features/remember.rb
161 def add_remember_key
162   hash = {remember_id_column=>account_id, remember_key_column=>remember_key_value}
163   set_deadline_value(hash, remember_deadline_column, remember_deadline_interval)
164 
165   if e = raised_uniqueness_violation{remember_key_ds.insert(hash)}
166     # If inserting into the remember key table causes a violation, we can pull the
167     # existing row from the table.  If there is no invalid row, we can then reraise.
168     raise e unless @remember_key_value = active_remember_key_ds.get(remember_key_column)
169   end
170 end
add_webauthn_credential(_)
[show source]
    # File lib/rodauth/features/recovery_codes.rb
153 def add_webauthn_credential(_)
154   super if defined?(super)
155   auto_add_missing_recovery_codes
156 end
after_change_password()
[show source]
   # File lib/rodauth/features/change_password_notify.rb
11 def after_change_password
12   super
13   send_password_changed_email
14 end
after_close_account()
[show source]
    # File lib/rodauth/features/account_expiration.rb
 99 def after_close_account
100   super if defined?(super)
101   account_activity_ds(account_id).delete
102 end
after_create_account()
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
89 def after_create_account
90   if account_password_hash_column && !(respond_to?(:verify_account_set_password?) && verify_account_set_password?)
91     add_previous_password_hash(password_hash(param(password_param)))
92   end
93   super if defined?(super)
94 end
after_login()
[show source]
    # File lib/rodauth/features/email_auth.rb
208 def after_login
209   # Remove the email auth key after any login, even if
210   # it is a password login.  This is done to invalidate
211   # the email login when a user has a password and requests
212   # email authentication, but then remembers their password
213   # and doesn't need the link.  At that point, the link
214   # that allows login access to the account becomes a
215   # security liability, and it is best to remove it.
216   remove_email_auth_key
217   super
218 end
after_login_entered_during_multi_phase_login()
[show source]
    # File lib/rodauth/features/email_auth.rb
151 def after_login_entered_during_multi_phase_login
152   # If forcing email auth, just send the email link.
153   _email_auth_request if force_email_auth?
154 
155   super
156 end
after_login_failure()
[show source]
    # File lib/rodauth/features/lockout.rb
268 def after_login_failure
269   invalid_login_attempted
270   super
271 end
after_logout()
[show source]
    # File lib/rodauth/features/remember.rb
243 def after_logout
244   forget_login
245   super if defined?(super)
246 end
after_otp_authentication_failure()
[show source]
   # File lib/rodauth/features/otp_lockout_email.rb
18 def after_otp_authentication_failure
19   super
20 
21   if otp_locked_out? && send_otp_locked_out_email?
22     send_otp_locked_out_email
23   end
24 end
after_otp_disable()
[show source]
   # File lib/rodauth/features/otp_modify_email.rb
18 def after_otp_disable
19   super
20   send_otp_disabled_email
21 end
after_otp_setup()
[show source]
    # File lib/rodauth/features/active_sessions.rb
153 def after_otp_setup
154   super if defined?(super)
155   remove_all_active_sessions_except_current if @clear_active_sessions_after_two_factor_setup
156 end
after_otp_unlock_auth_failure()
[show source]
    # File lib/rodauth/features/json.rb
 98 def after_otp_unlock_auth_failure
 99   super if defined?(super)
100   _set_otp_unlock_info
101 end
after_otp_unlock_auth_success()
[show source]
   # File lib/rodauth/features/json.rb
90 def after_otp_unlock_auth_success
91   super if defined?(super)
92   if otp_locked_out?
93     _set_otp_unlock_info
94     json_response[:deadline] = otp_unlock_deadline.to_i
95   end
96 end
after_otp_unlock_not_yet_available()
[show source]
    # File lib/rodauth/features/json.rb
103 def after_otp_unlock_not_yet_available
104   super if defined?(super)
105   _set_otp_unlock_info
106 end
after_refresh_token()
[show source]
    # File lib/rodauth/features/active_sessions.rb
137 def after_refresh_token
138   super if defined?(super)
139   if prev_key = session[session_id_session_key]
140     key = generate_active_sessions_key
141     set_session_value(session_id_session_key, key)
142     active_sessions_ds.
143       where(active_sessions_session_id_column => compute_hmacs(prev_key)).
144       update(active_sessions_session_id_column => compute_hmac(key))
145   end
146 end
after_reset_password()
[show source]
   # File lib/rodauth/features/password_grace_period.rb
41 def after_reset_password
42   super if defined?(super)
43   @last_password_entry = Time.now.to_i
44 end
after_sms_confirm()
[show source]
    # File lib/rodauth/features/active_sessions.rb
158 def after_sms_confirm
159   super if defined?(super)
160   remove_all_active_sessions_except_current if @clear_active_sessions_after_two_factor_setup
161 end
after_sms_disable()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
232 def after_sms_disable
233   super if defined?(super)
234   auto_remove_recovery_codes
235 end
after_webauthn_remove()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
237 def after_webauthn_remove
238   super if defined?(super)
239   auto_remove_recovery_codes
240 end
after_webauthn_setup()
[show source]
    # File lib/rodauth/features/active_sessions.rb
163 def after_webauthn_setup
164   super if defined?(super)
165   remove_all_active_sessions_except_current if @clear_active_sessions_after_two_factor_setup
166 end
allow_email_auth?()
[show source]
    # File lib/rodauth/features/email_auth.rb
204 def allow_email_auth?
205   defined?(super) ? super : true
206 end
allow_resending_verify_account_email?()
[show source]
    # File lib/rodauth/features/verify_account.rb
165 def allow_resending_verify_account_email?
166   account[account_status_column] == account_unverified_status_value
167 end
already_logged_in()
[show source]
    # File lib/rodauth/features/base.rb
342 def already_logged_in
343   nil
344 end
argon2_hash_algorithm?(hash)

:nocov:

[show source]
    # File lib/rodauth/features/argon2.rb
107 def argon2_hash_algorithm?(hash)
108   hash.start_with?('$argon2id$')
109 end
argon2_hash_cost()
[show source]
   # File lib/rodauth/features/argon2.rb
78 def argon2_hash_cost
79   { t_cost: 1, m_cost: 5, p_cost: 1 }
80 end
argon2_password_hash_match?(hash, password)
[show source]
    # File lib/rodauth/features/argon2.rb
111 def argon2_password_hash_match?(hash, password)
112   ret = ::Argon2::Password.verify_password(password, hash, argon2_secret)
113 
114   if ret == false && argon2_old_secret != argon2_secret && (ret = ::Argon2::Password.verify_password(password, hash, argon2_old_secret))
115     @update_password_hash = true
116   end
117 
118   ret
119 end
argon2_password_hash_using_salt_and_secret(password, salt, secret)
[show source]
   # File lib/rodauth/features/argon2.rb
62 def argon2_password_hash_using_salt_and_secret(password, salt, secret)
63   argon2_params = Hash[extract_password_hash_cost(salt)]
64   argon2_params[argon2_salt_option] = salt.split('$').last.unpack("m")[0]
65   argon2_params[:secret] = secret
66   ::Argon2::Password.new(argon2_params).create(password)
67 end
argon2_salt_option()
[show source]
   # File lib/rodauth/features/argon2.rb
36 def argon2_salt_option
37   :salt_do_not_supply
38 end
audit_log_ds()
[show source]
   # File lib/rodauth/features/audit_logging.rb
83 def audit_log_ds
84   ds = db[audit_logging_table]
85   # :nocov:
86   if db.database_type == :postgres
87   # :nocov:
88     # For PostgreSQL, use RETURNING NULL. This allows the feature
89     # to be used with INSERT but not SELECT permissions on the
90     # table, useful for audit logging where the database user
91     # the application is running as should not need to read the
92     # logs.
93     ds = ds.returning(nil)
94   end
95   ds
96 end
audit_log_insert_hash(account_id, action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
45 def audit_log_insert_hash(account_id, action)
46   if message = audit_log_message(action)
47     {
48       audit_logging_account_id_column => account_id,
49       audit_logging_message_column => message,
50       audit_logging_metadata_column => serialize_audit_log_metadata(audit_log_metadata(action))
51     }
52   end
53 end
audit_log_message(action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
63 def audit_log_message(action)
64   meth = :"audit_log_message_for_#{action}"
65   if respond_to?(meth, true)
66     send(meth)
67   else
68     audit_log_message_default(action)
69   end
70 end
audit_log_message_default(action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
59 def audit_log_message_default(action)
60   action.to_s
61 end
audit_log_metadata(action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
72 def audit_log_metadata(action)
73   meth = :"audit_log_metadata_for_#{action}"
74   if respond_to?(meth, true)
75     send(meth)
76   else
77     audit_log_metadata_default
78   end
79 end
auth_class_eval(&block)
[show source]
    # File lib/rodauth/features/base.rb
150 def auth_class_eval(&block)
151   auth.class_eval(&block)
152 end
authenticated?()
[show source]
    # File lib/rodauth/features/base.rb
398 def authenticated?
399   logged_in?
400 end
authenticated_by()
[show source]
    # File lib/rodauth/features/base.rb
502 def authenticated_by
503   session[authenticated_by_session_key]
504 end
authenticated_webauthn_id()
[show source]
    # File lib/rodauth/features/webauthn.rb
251 def authenticated_webauthn_id
252   session[authenticated_webauthn_id_session_key]
253 end
auto_add_missing_recovery_codes()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
250 def auto_add_missing_recovery_codes
251   if auto_add_recovery_codes?
252     add_recovery_codes(recovery_codes_limit - recovery_codes.length)
253   end
254 end
auto_remove_recovery_codes()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
256 def auto_remove_recovery_codes
257   if auto_remove_recovery_codes? && (%w'totp webauthn sms_code' & possible_authentication_methods).empty?
258     recovery_codes_remove
259   end
260 end
autocomplete_for_field?(_param)
[show source]
    # File lib/rodauth/features/base.rb
245 def autocomplete_for_field?(_param)
246   mark_input_fields_with_autocomplete?
247 end
autologin_session(autologin_type)
[show source]
    # File lib/rodauth/features/base.rb
515 def autologin_session(autologin_type)
516   login_session('autologin')
517   set_session_value(autologin_type_session_key, autologin_type)
518 end
autologin_type()
[show source]
    # File lib/rodauth/features/base.rb
511 def autologin_type
512   session[autologin_type_session_key]
513 end
base32_encode(data, length)
[show source]
    # File lib/rodauth/features/otp.rb
414 def base32_encode(data, length)
415   chars = 'abcdefghijklmnopqrstuvwxyz234567'
416   length.times.map{|i|chars[data[i].ord % 32]}.join
417 end
base_url()
[show source]
    # File lib/rodauth/features/base.rb
561 def base_url
562   url = String.new("#{request.scheme}://#{domain}")
563   url << ":#{request.port}" if request.port != Rack::Request::DEFAULT_PORTS[request.scheme]
564   url
565 end
before_change_login_route()
[show source]
   # File lib/rodauth/features/verify_account_grace_period.rb
58 def before_change_login_route
59   unless verified_account?
60     set_redirect_error_flash unverified_change_login_error_flash
61     redirect unverified_change_login_redirect
62   end
63   super if defined?(super)
64 end
before_change_password_route()
[show source]
   # File lib/rodauth/features/password_expiration.rb
90 def before_change_password_route
91   check_password_change_allowed
92   super
93 end
before_create_account_route()
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
79 def before_create_account_route
80   super if defined?(super)
81   @dont_check_previous_password = true
82 end
before_login_attempt()
[show source]
    # File lib/rodauth/features/lockout.rb
256 def before_login_attempt
257   if locked_out?
258     show_lockout_page
259   end
260   super
261 end
before_logout()
[show source]
    # File lib/rodauth/features/active_sessions.rb
168 def before_logout
169   if param_or_nil(global_logout_param)
170     remove_remember_key(session_value) if respond_to?(:remove_remember_key)
171     remove_all_active_sessions
172   else
173     remove_current_session
174   end
175   super
176 end
before_otp_setup()
[show source]
    # File lib/rodauth/features/active_sessions.rb
178 def before_otp_setup
179   @clear_active_sessions_after_two_factor_setup = !two_factor_authentication_setup?
180   super if defined?(super)
181 end
before_otp_setup_route()
[show source]
    # File lib/rodauth/features/json.rb
172 def before_otp_setup_route
173   super if defined?(super)
174   if use_json? && otp_keys_use_hmac? && !param_or_nil(otp_setup_raw_param)
175     _otp_tmp_key(otp_new_secret)
176     json_response[otp_setup_param] = otp_user_key
177     json_response[otp_setup_raw_param] = otp_key
178   end
179 end
before_reset_password()
[show source]
   # File lib/rodauth/features/account_expiration.rb
79 def before_reset_password
80   check_account_expiration
81   super if defined?(super)
82 end
before_reset_password_request()
[show source]
   # File lib/rodauth/features/account_expiration.rb
84 def before_reset_password_request
85   check_account_expiration
86   super if defined?(super)
87 end
before_rodauth()
[show source]
    # File lib/rodauth/features/json.rb
181 def before_rodauth
182   if json_request?
183     if json_check_accept? && (accept = request.env['HTTP_ACCEPT']) && accept !~ json_accept_regexp
184       response.status = 406
185       json_response[json_response_error_key] = json_not_accepted_error_message
186       _return_json_response
187     end
188 
189     unless request.post?
190       response.status = 405
191       set_response_header('allow', 'POST')
192       json_response[json_response_error_key] = json_non_post_error_message
193       return_json_response
194     end
195   elsif only_json?
196     response.status = json_response_error_status
197     return_response non_json_request_error_message
198   end
199 
200   super
201 end
before_sms_confirm()
[show source]
    # File lib/rodauth/features/active_sessions.rb
183 def before_sms_confirm
184   @clear_active_sessions_after_two_factor_setup = !two_factor_authentication_setup?
185   super if defined?(super)
186 end
before_two_factor_auth_route()
[show source]
    # File lib/rodauth/features/json.rb
118 def before_two_factor_auth_route
119   super if defined?(super)
120   if use_json?
121     json_response[:auth_links] = two_factor_auth_links.sort.map{|_,link| link}
122     json_response[json_response_success_key] ||= "" if include_success_messages?
123     return_json_response
124   end
125 end
before_two_factor_manage_route()
[show source]
    # File lib/rodauth/features/json.rb
108 def before_two_factor_manage_route
109   super if defined?(super)
110   if use_json?
111     json_response[:setup_links] = two_factor_setup_links.sort.map{|_,link| link}
112     json_response[:remove_links] = two_factor_remove_links.sort.map{|_,link| link}
113     json_response[json_response_success_key] ||= "" if include_success_messages?
114     return_json_response
115   end
116 end
before_unlock_account()
[show source]
   # File lib/rodauth/features/account_expiration.rb
89 def before_unlock_account
90   check_account_expiration
91   super if defined?(super)
92 end
before_unlock_account_request()
[show source]
   # File lib/rodauth/features/account_expiration.rb
94 def before_unlock_account_request
95   check_account_expiration
96   super if defined?(super)
97 end
before_verify_account()
[show source]
   # File lib/rodauth/features/webauthn_verify_account.rb
32 def before_verify_account
33   super
34   if features.include?(:json) && use_json? && !param_or_nil(webauthn_setup_param)
35     cred = new_webauthn_credential
36     json_response[webauthn_setup_param] = cred.as_json
37     json_response[webauthn_setup_challenge_param] = cred.challenge
38     json_response[webauthn_setup_challenge_hmac_param] = compute_hmac(cred.challenge)
39   end
40   @webauthn_credential = webauthn_setup_credential_from_form_submission
41   add_webauthn_credential(@webauthn_credential)
42 end
before_verify_account_route()
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
84 def before_verify_account_route
85   super if defined?(super)
86   @dont_check_previous_password = true
87 end
before_view_recovery_codes()
[show source]
    # File lib/rodauth/features/json.rb
127 def before_view_recovery_codes
128   super if defined?(super)
129   if use_json?
130     json_response[:codes] = recovery_codes
131     json_response[json_response_success_key] ||= "" if include_success_messages?
132   end
133 end
before_webauthn_auth_route()
[show source]
    # File lib/rodauth/features/json.rb
145 def before_webauthn_auth_route
146   super if defined?(super)
147   if use_json? && !param_or_nil(webauthn_auth_param)
148     cred = webauthn_credential_options_for_get
149     json_response[webauthn_auth_param] = cred.as_json
150     json_response[webauthn_auth_challenge_param] = cred.challenge
151     json_response[webauthn_auth_challenge_hmac_param] = compute_hmac(cred.challenge)
152   end
153 end
before_webauthn_login_route()
[show source]
    # File lib/rodauth/features/json.rb
155 def before_webauthn_login_route
156   super if defined?(super)
157   if use_json? && !param_or_nil(webauthn_auth_param) && webauthn_login_options?
158     cred = webauthn_credential_options_for_get
159     json_response[webauthn_auth_param] = cred.as_json
160     json_response[webauthn_auth_challenge_param] = cred.challenge
161     json_response[webauthn_auth_challenge_hmac_param] = compute_hmac(cred.challenge)
162   end
163 end
before_webauthn_remove_route()
[show source]
    # File lib/rodauth/features/json.rb
165 def before_webauthn_remove_route
166   super if defined?(super)
167   if use_json? && !param_or_nil(webauthn_remove_param)
168     json_response[webauthn_remove_param] = account_webauthn_usage
169   end
170 end
before_webauthn_setup()
[show source]
    # File lib/rodauth/features/active_sessions.rb
188 def before_webauthn_setup
189   @clear_active_sessions_after_two_factor_setup = !two_factor_authentication_setup?
190   super if defined?(super)
191 end
before_webauthn_setup_route()
[show source]
    # File lib/rodauth/features/json.rb
135 def before_webauthn_setup_route
136   super if defined?(super)
137   if use_json? && !param_or_nil(webauthn_setup_param)
138     cred = new_webauthn_credential
139     json_response[webauthn_setup_param] = cred.as_json
140     json_response[webauthn_setup_challenge_param] = cred.challenge
141     json_response[webauthn_setup_challenge_hmac_param] = compute_hmac(cred.challenge)
142   end
143 end
button(value, opts={})
[show source]
    # File lib/rodauth/features/base.rb
452 def button(value, opts={})
453   scope.render(button_opts(value, opts))
454 end
button_fixed_locals()
[show source]
    # File lib/rodauth/features/base.rb
638 def button_fixed_locals
639   '(value:, opts:)'
640 end
button_opts(value, opts)
[show source]
    # File lib/rodauth/features/base.rb
443 def button_opts(value, opts)
444   opts = Hash[template_opts].merge!(opts)
445   _merge_fixed_locals_opts(opts, button_fixed_locals)
446   opts[:locals] = {:value=>value, :opts=>opts}
447   opts[:cache] = cache_templates
448   opts[:cache_key] = :rodauth_button
449   _template_opts(opts, 'button')
450 end
can_add_recovery_codes?()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
175 def can_add_recovery_codes?
176   recovery_codes.length < recovery_codes_limit
177 end
catch_error(&block)
[show source]
    # File lib/rodauth/features/base.rb
738 def catch_error(&block)
739   catch(:rodauth_error, &block)
740 end
change_login(login)
[show source]
   # File lib/rodauth/features/change_login.rb
68 def change_login(login)
69   if account_ds.get(login_column).downcase == login.downcase
70     set_login_requirement_error_message(:same_as_current_login, same_as_current_login_message)
71     return false
72   end
73 
74   update_login(login)
75 end
change_login_notice_flash()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
138 def change_login_notice_flash
139   change_login_needs_verification_notice_flash
140 end
change_login_requires_password?()
[show source]
   # File lib/rodauth/features/change_login.rb
64 def change_login_requires_password?
65   modifications_require_password?
66 end
change_password_requires_password?()
[show source]
   # File lib/rodauth/features/change_password.rb
68 def change_password_requires_password?
69   modifications_require_password?
70 end
check_account_expiration()
[show source]
   # File lib/rodauth/features/account_expiration.rb
63 def check_account_expiration
64   if account_expired?
65     set_expired unless account_expired_at
66     set_redirect_error_flash account_expiration_error_flash
67     redirect account_expiration_redirect
68   end
69   update_last_login
70 end
check_active_session()
[show source]
   # File lib/rodauth/features/active_sessions.rb
56 def check_active_session
57   if logged_in? && !currently_active_session?
58     no_longer_active_session
59   end
60 end
check_already_logged_in()
[show source]
    # File lib/rodauth/features/base.rb
338 def check_already_logged_in
339   already_logged_in if logged_in?
340 end
check_csrf()
[show source]
    # File lib/rodauth/features/base.rb
427 def check_csrf
428   scope.check_csrf!(check_csrf_opts, &check_csrf_block)
429 end
check_csrf?()
[show source]
    # File lib/rodauth/features/base.rb
800 def check_csrf?
801   scope.opts[:rodauth_route_csrf]
802 end
check_password_change_allowed()
[show source]
   # File lib/rodauth/features/password_expiration.rb
30 def check_password_change_allowed
31   if password_changed_at = get_password_changed_at
32     if password_changed_at > Time.now - allow_password_change_after
33       set_redirect_error_flash password_not_changeable_yet_error_flash
34       redirect password_not_changeable_yet_redirect
35     end
36   end
37 end
check_session_expiration()
[show source]
   # File lib/rodauth/features/session_expiration.rb
15 def check_session_expiration
16   return unless logged_in?
17 
18   unless session.has_key?(session_last_activity_session_key) && session.has_key?(session_created_session_key)
19     if session_expiration_default
20       expire_session
21     end
22 
23     return
24   end
25 
26   time = Time.now.to_i
27 
28   if session[session_last_activity_session_key] + session_inactivity_timeout < time
29     expire_session
30   end
31   set_session_value(session_last_activity_session_key, time)
32 
33   if session[session_created_session_key] + max_session_lifetime < time
34     expire_session
35   end
36 end
check_single_session()
[show source]
   # File lib/rodauth/features/single_session.rb
52 def check_single_session
53   if logged_in? && !currently_active_session?
54     no_longer_active_session
55   end
56 end
clear_cached_otp()
[show source]
    # File lib/rodauth/features/otp.rb
372 def clear_cached_otp
373   @otp = nil
374 end
clear_invalid_login_attempts()
[show source]
    # File lib/rodauth/features/lockout.rb
168 def clear_invalid_login_attempts
169   unlock_account
170 end
clear_session()
[show source]
    # File lib/rodauth/features/base.rb
354 def clear_session
355   if use_scope_clear_session?
356     scope.clear_session
357   else
358     session.clear
359   end
360 end
clear_tokens(reason)
[show source]
    # File lib/rodauth/features/active_sessions.rb
130 def clear_tokens(reason)
131   super
132   remove_all_active_sessions_except_current
133 end
close_account()
[show source]
   # File lib/rodauth/features/close_account.rb
67 def close_account
68   unless skip_status_checks?
69     update_account(account_status_column=>account_closed_status_value)
70   end
71 
72   unless account_password_hash_column
73     password_hash_ds.delete
74   end
75 end
close_account_requires_password?()
[show source]
   # File lib/rodauth/features/close_account.rb
63 def close_account_requires_password?
64   modifications_require_password?
65 end
compute_hmac(data)

Return urlsafe base64 HMAC for data, assumes hmac_secret is set.

[show source]
    # File lib/rodauth/features/base.rb
279 def compute_hmac(data)
280   _process_raw_hmac(compute_raw_hmac(data))
281 end
compute_hmacs(data)

Return array of hmacs. Array has two strings if hmac_old_secret is set, or one string otherwise.

[show source]
    # File lib/rodauth/features/base.rb
290 def compute_hmacs(data)
291   hmacs = [compute_hmac(data)]
292 
293   if hmac_old_secret
294     hmacs << compute_old_hmac(data)
295   end
296 
297   hmacs
298 end
compute_old_hmac(data)

Return urlsafe base64 HMAC for data using hmac_old_secret, assumes hmac_old_secret is set.

[show source]
    # File lib/rodauth/features/base.rb
284 def compute_old_hmac(data)
285   _process_raw_hmac(compute_raw_hmac_with_secret(data, hmac_old_secret))
286 end
compute_raw_hmac(data)
[show source]
    # File lib/rodauth/features/base.rb
855 def compute_raw_hmac(data)
856   raise ConfigurationError, "hmac_secret not set" unless hmac_secret
857   compute_raw_hmac_with_secret(data, hmac_secret)
858 end
compute_raw_hmac_with_secret(data, secret)
[show source]
    # File lib/rodauth/features/base.rb
860 def compute_raw_hmac_with_secret(data, secret)
861   OpenSSL::HMAC.digest(OpenSSL::Digest::SHA256.new, secret, data)
862 end
confirm_password()
[show source]
   # File lib/rodauth/features/confirm_password.rb
63 def confirm_password
64   authenticated_by.delete('autologin')
65   authenticated_by.delete('remember')
66   authenticated_by.delete('email_auth')
67   authenticated_by.delete('password')
68   authenticated_by.unshift("password")
69   remove_session_value(autologin_type_session_key)
70   nil
71 end
confirm_password_redirect()
[show source]
   # File lib/rodauth/features/confirm_password.rb
73 def confirm_password_redirect
74   remove_session_value(confirm_password_redirect_session_key) || default_redirect
75 end
convert_email_token_key(key)
[show source]
   # File lib/rodauth/features/email_base.rb
61 def convert_email_token_key(key)
62   convert_token_key(key)
63 end
convert_response_header_key(key)
[show source]
    # File lib/rodauth/features/base.rb
606 def convert_response_header_key(key)
607   key
608 end
convert_session_key(key)
[show source]
    # File lib/rodauth/features/base.rb
717 def convert_session_key(key)
718   key = :"#{session_key_prefix}#{key}" if session_key_prefix
719   normalize_session_or_flash_key(key)
720 end
convert_timestamp(timestamp)

This is needed for jdbc/sqlite, which returns timestamp columns as strings

[show source]
    # File lib/rodauth/features/base.rb
900 def convert_timestamp(timestamp)
901   timestamp = db.to_application_timestamp(timestamp) if timestamp.is_a?(String)
902   timestamp
903 end
convert_token_id(id)
[show source]
    # File lib/rodauth/features/base.rb
671 def convert_token_id(id)
672   if convert_token_id_to_integer?
673     convert_token_id_to_integer(id)
674   else
675     id
676   end
677 end
convert_token_id_to_integer(id)
[show source]
    # File lib/rodauth/features/base.rb
679 def convert_token_id_to_integer(id)
680   if id = (Integer(id, 10) rescue nil)
681     if id > 9223372036854775807 || id < -9223372036854775808
682       # Only allow 64-bit signed integer range to avoid problems on PostgreSQL
683       id = nil
684     end
685   end
686 
687   id
688 end
convert_token_key(key)
[show source]
    # File lib/rodauth/features/base.rb
659 def convert_token_key(key)
660   if key && hmac_secret
661     compute_hmac(key)
662   else
663     key
664   end
665 end
create_account_autologin?()
[show source]
    # File lib/rodauth/features/verify_account.rb
218 def create_account_autologin?
219   false
220 end
create_account_notice_flash()
[show source]
    # File lib/rodauth/features/verify_account.rb
185 def create_account_notice_flash
186   verify_account_email_sent_notice_flash
187 end
create_account_set_password?()
[show source]
    # File lib/rodauth/features/verify_account.rb
222 def create_account_set_password?
223   return false if verify_account_set_password?
224   super
225 end
create_email(subject, body)
[show source]
   # File lib/rodauth/features/email_base.rb
40 def create_email(subject, body)
41   create_email_to(email_to, subject, body)
42 end
create_email_auth_key()
[show source]
    # File lib/rodauth/features/email_auth.rb
104 def create_email_auth_key
105   transaction do
106     if email_auth_key_value = get_email_auth_key(account_id)
107       set_email_auth_email_last_sent
108       @email_auth_key_value = email_auth_key_value
109     elsif e = raised_uniqueness_violation{email_auth_ds.insert(email_auth_key_insert_hash)}
110       # If inserting into the email auth table causes a violation, we can pull the
111       # existing email auth key from the table, or reraise.
112       raise e unless @email_auth_key_value = get_email_auth_key(account_id)
113     end
114   end
115 end
create_email_to(to, subject, body)
[show source]
   # File lib/rodauth/features/email_base.rb
44 def create_email_to(to, subject, body)
45   m = Mail.new
46   m.from = email_from
47   m.to = to
48   m.subject = "#{email_subject_prefix}#{subject}"
49   m.body = body
50   m
51 end
create_reset_password_key()
[show source]
    # File lib/rodauth/features/reset_password.rb
165 def create_reset_password_key
166   transaction do
167     if reset_password_key_value = get_password_reset_key(account_id)
168       set_reset_password_email_last_sent
169       @reset_password_key_value = reset_password_key_value
170     elsif e = raised_uniqueness_violation{password_reset_ds.insert(reset_password_key_insert_hash)}
171       # If inserting into the reset password table causes a violation, we can pull the
172       # existing reset password key from the table, or reraise.
173       raise e unless @reset_password_key_value = get_password_reset_key(account_id)
174     end
175   end
176 end
create_verify_account_key()
[show source]
    # File lib/rodauth/features/verify_account.rb
288 def create_verify_account_key
289   ds = verify_account_ds
290   transaction do
291     if ds.empty?
292       if e = raised_uniqueness_violation{ds.insert(verify_account_key_insert_hash)}
293         # If inserting into the verify account table causes a violation, we can pull the
294         # key from the verify account table, or reraise.
295         raise e unless @verify_account_key_value = get_verify_account_key(account_id)
296       end
297     end
298   end
299 end
create_verify_login_change_email(login)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
200 def create_verify_login_change_email(login)
201   create_email_to(login, verify_login_change_email_subject, verify_login_change_email_body)
202 end
create_verify_login_change_key(login)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
178 def create_verify_login_change_key(login)
179   ds = verify_login_change_ds
180   transaction do
181     ds.where((Sequel::CURRENT_TIMESTAMP > verify_login_change_deadline_column) | ~Sequel.expr(verify_login_change_login_column=>login)).delete
182     if e = raised_uniqueness_violation{ds.insert(verify_login_change_key_insert_hash(login))}
183       old_login, key = get_verify_login_change_login_and_key(account_id)
184       # If inserting into the verify login change table causes a violation, we can pull the
185       # key from the verify login change table if the logins match, or reraise.
186       @verify_login_change_key_value = if old_login.downcase == login.downcase
187         key
188       end
189       raise e unless @verify_login_change_key_value
190     end
191   end
192 end
csrf_tag(path=request.path)
[show source]
    # File lib/rodauth/features/base.rb
431 def csrf_tag(path=request.path)
432   return unless scope.respond_to?(:csrf_tag)
433 
434   if use_request_specific_csrf_tokens?
435     scope.csrf_tag(path)
436   else
437     # :nocov:
438     scope.csrf_tag
439     # :nocov:
440   end
441 end
currently_active_session?()
[show source]
   # File lib/rodauth/features/active_sessions.rb
42 def currently_active_session?
43   return false unless session_id = session[session_id_session_key]
44 
45   remove_inactive_sessions
46   ds = active_sessions_ds.
47     where(active_sessions_session_id_column => compute_hmacs(session_id))
48 
49   if update_current_session?
50     ds.update(active_sessions_update_hash) == 1
51   else
52     ds.count == 1
53   end
54 end
database_function_password_match?(name, hash_id, password, salt)
[show source]
    # File lib/rodauth/features/argon2.rb
121 def database_function_password_match?(name, hash_id, password, salt)
122   return true if super
123 
124   if use_argon2? && argon2_hash_algorithm?(salt) && argon2_old_secret != argon2_secret && (ret = db.get(Sequel.function(function_name(name), hash_id, argon2_password_hash_using_salt_and_secret(password, salt, argon2_old_secret))))
125     @update_password_hash = true
126   end
127 
128   !!ret
129 end
db()
[show source]
    # File lib/rodauth/features/base.rb
318 def db
319   Sequel::DATABASES.first or raise "Sequel database connection is missing"
320 end
delete_account()
[show source]
   # File lib/rodauth/features/close_account.rb
77 def delete_account
78   account_ds.delete
79 end
delete_account_on_close?()
[show source]
   # File lib/rodauth/features/close_account.rb
81 def delete_account_on_close?
82   skip_status_checks?
83 end
disable_remember_login()
[show source]
    # File lib/rodauth/features/remember.rb
157 def disable_remember_login
158   remove_remember_key
159 end
domain()
[show source]
    # File lib/rodauth/features/base.rb
567 def domain
568   request.host
569 end
email_auth_ds(id=account_id)
[show source]
    # File lib/rodauth/features/email_auth.rb
234 def email_auth_ds(id=account_id)
235   db[email_auth_table].where(email_auth_id_column=>id)
236 end
email_auth_email_recently_sent?()
[show source]
    # File lib/rodauth/features/email_auth.rb
168 def email_auth_email_recently_sent?
169   (email_last_sent = get_email_auth_email_last_sent) && (Time.now - email_last_sent < email_auth_skip_resend_email_within)
170 end
email_auth_key_insert_hash()
[show source]
    # File lib/rodauth/features/email_auth.rb
228 def email_auth_key_insert_hash
229   hash = {email_auth_id_column=>account_id, email_auth_key_column=>email_auth_key_value}
230   set_deadline_value(hash, email_auth_deadline_column, email_auth_deadline_interval)
231   hash
232 end
email_auth_request_form()
[show source]
    # File lib/rodauth/features/email_auth.rb
147 def email_auth_request_form
148   render('email-auth-request-form')
149 end
email_from()
[show source]
   # File lib/rodauth/features/email_base.rb
26 def email_from
27   "webmaster@#{domain}"
28 end
email_to()
[show source]
   # File lib/rodauth/features/email_base.rb
30 def email_to
31   account[login_column]
32 end
expire_session()
[show source]
   # File lib/rodauth/features/session_expiration.rb
38 def expire_session
39   clear_session
40   set_redirect_error_status session_expiration_error_status
41   set_error_reason :session_expired
42   set_redirect_error_flash session_expiration_error_flash
43   redirect session_expiration_redirect
44 end
extend_remember_deadline()
[show source]
    # File lib/rodauth/features/remember.rb
214 def extend_remember_deadline
215   active_remember_key_ds.update(remember_deadline_column=>Sequel.date_add(Sequel::CURRENT_TIMESTAMP, remember_period))
216   remember_login
217 end
extend_remember_deadline_while_logged_in?()
[show source]
    # File lib/rodauth/features/remember.rb
202 def extend_remember_deadline_while_logged_in?
203   return false unless extend_remember_deadline?
204 
205   if extended_at = session[remember_deadline_extended_session_key]
206     extended_at + extend_remember_deadline_period < Time.now.to_i
207   elsif logged_in_via_remember_key?
208     # Handle existing sessions before the change to extend remember deadline
209     # while logged in.
210     true
211   end
212 end
extract_password_hash_cost(hash)
[show source]
   # File lib/rodauth/features/argon2.rb
70 def extract_password_hash_cost(hash)
71   return super unless argon2_hash_algorithm?(hash)
72 
73   /\A\$argon2id\$v=\d+\$m=(\d+),t=(\d+),p=(\d+)/ =~ hash
74   { t_cost: $2.to_i, m_cost: Math.log2($1.to_i).to_i, p_cost: $3.to_i }
75 end
features()
[show source]
    # File lib/rodauth/features/base.rb
178 def features
179   self.class.features
180 end
field_attributes(field)
[show source]
    # File lib/rodauth/features/base.rb
253 def field_attributes(field)
254   _field_attributes(field) || default_field_attributes
255 end
field_error(field)
[show source]
    # File lib/rodauth/features/base.rb
210 def field_error(field)
211   return nil unless @field_errors
212   @field_errors[field]
213 end
field_error_attributes(field)
[show source]
    # File lib/rodauth/features/base.rb
257 def field_error_attributes(field)
258   if field_error(field)
259     _field_error_attributes(field)
260   end
261 end
flash()
[show source]
    # File lib/rodauth/features/base.rb
194 def flash
195   scope.flash
196 end
forget_login()
[show source]
    # File lib/rodauth/features/remember.rb
140 def forget_login
141   opts = Hash[remember_cookie_options]
142   opts[:path] = "/" unless opts.key?(:path)
143   ::Rack::Utils.delete_cookie_header!(response.headers, remember_cookie_key, opts)
144 end
formatted_field_error(field)
[show source]
    # File lib/rodauth/features/base.rb
263 def formatted_field_error(field)
264   if error = field_error(field)
265     _formatted_field_error(field, error)
266   end
267 end
function_name(name)
[show source]
    # File lib/rodauth/features/base.rb
804 def function_name(name)
805   if db.database_type == :mssql
806     # :nocov:
807     "dbo.#{name}"
808     # :nocov:
809   else
810     name
811   end
812 end
generate_active_sessions_key()
[show source]
    # File lib/rodauth/features/active_sessions.rb
195 def generate_active_sessions_key
196   @active_sessions_key = random_key
197 end
generate_email_auth_key_value()
[show source]
    # File lib/rodauth/features/email_auth.rb
220 def generate_email_auth_key_value
221   @email_auth_key_value = random_key
222 end
generate_refresh_token()
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
192 def generate_refresh_token
193   hash = jwt_refresh_token_insert_hash
194   [account_id, jwt_refresh_token_ds.insert(hash), convert_token_key(hash[jwt_refresh_token_key_column])].join(token_separator)
195 end
generate_remember_key_value()
[show source]
    # File lib/rodauth/features/remember.rb
255 def generate_remember_key_value
256   @remember_key_value = random_key
257 end
generate_reset_password_key_value()
[show source]
    # File lib/rodauth/features/reset_password.rb
236 def generate_reset_password_key_value
237   @reset_password_key_value = random_key
238 end
generate_unlock_account_key()
[show source]
    # File lib/rodauth/features/lockout.rb
278 def generate_unlock_account_key
279   random_key
280 end
generate_verify_account_key_value()
[show source]
    # File lib/rodauth/features/verify_account.rb
284 def generate_verify_account_key_value
285   @verify_account_key_value = random_key
286 end
generate_verify_login_change_key_value()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
174 def generate_verify_login_change_key_value
175   @verify_login_change_key_value = random_key
176 end
get_active_refresh_token(account_id, token_id)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
165 def get_active_refresh_token(account_id, token_id)
166   jwt_refresh_token_account_ds(account_id).
167     where(Sequel::CURRENT_TIMESTAMP > jwt_refresh_token_deadline_column).
168     delete
169 
170   jwt_refresh_token_account_token_ds(account_id, token_id).
171     get(jwt_refresh_token_key_column)
172 end
get_activity_timestamp(account_id, column)
[show source]
    # File lib/rodauth/features/account_expiration.rb
109 def get_activity_timestamp(account_id, column)
110   convert_timestamp(account_activity_ds(account_id).get(column))
111 end
get_email_auth_email_last_sent()
[show source]
    # File lib/rodauth/features/email_auth.rb
121 def get_email_auth_email_last_sent
122   if column = email_auth_email_last_sent_column
123     if ts = email_auth_ds.get(column)
124       convert_timestamp(ts)
125     end
126   end
127 end
get_email_auth_key(id)
[show source]
    # File lib/rodauth/features/email_auth.rb
141 def get_email_auth_key(id)
142   ds = email_auth_ds(id)
143   ds.where(Sequel::CURRENT_TIMESTAMP > email_auth_deadline_column).delete
144   ds.get(email_auth_key_column)
145 end
get_password_changed_at()
[show source]
   # File lib/rodauth/features/password_expiration.rb
26 def get_password_changed_at
27   convert_timestamp(password_expiration_ds.get(password_expiration_changed_at_column))
28 end
get_password_hash()

Get the password hash for the user. When using database authentication functions, note that only the salt is returned.

[show source]
    # File lib/rodauth/features/base.rb
820 def get_password_hash
821   if account_password_hash_column
822     account[account_password_hash_column] if account!
823   elsif use_database_authentication_functions?
824     db.get(Sequel.function(function_name(:rodauth_get_salt), account ? account_id : session_value))
825   else
826     # :nocov:
827     password_hash_ds.get(password_hash_column)
828     # :nocov:
829   end
830 end
get_password_reset_key(id)
[show source]
    # File lib/rodauth/features/reset_password.rb
194 def get_password_reset_key(id)
195   ds = password_reset_ds(id)
196   ds.where(Sequel::CURRENT_TIMESTAMP > reset_password_deadline_column).delete
197   ds.get(reset_password_key_column)
198 end
get_remember_key()
[show source]
    # File lib/rodauth/features/remember.rb
146 def get_remember_key
147   unless @remember_key_value = active_remember_key_ds.get(remember_key_column)
148    generate_remember_key_value
149    transaction do
150      remove_remember_key
151      add_remember_key
152    end
153   end
154   nil
155 end
get_reset_password_email_last_sent()
[show source]
    # File lib/rodauth/features/reset_password.rb
204 def get_reset_password_email_last_sent
205   if column = reset_password_email_last_sent_column
206     if ts = password_reset_ds.get(column)
207       convert_timestamp(ts)
208     end
209   end
210 end
get_unlock_account_email_last_sent()
[show source]
    # File lib/rodauth/features/lockout.rb
231 def get_unlock_account_email_last_sent
232   if column = account_lockouts_email_last_sent_column
233     if ts = account_lockouts_ds.get(column)
234       convert_timestamp(ts)
235     end
236   end
237 end
get_unlock_account_key()
[show source]
    # File lib/rodauth/features/lockout.rb
219 def get_unlock_account_key
220   account_lockouts_ds.get(account_lockouts_key_column)
221 end
get_verify_account_email_last_sent()
[show source]
    # File lib/rodauth/features/verify_account.rb
231 def get_verify_account_email_last_sent
232   if column = verify_account_email_last_sent_column
233     if ts = verify_account_ds.get(column)
234       convert_timestamp(ts)
235     end
236   end
237 end
get_verify_account_key(id)
[show source]
    # File lib/rodauth/features/verify_account.rb
210 def get_verify_account_key(id)
211   verify_account_ds(id).get(verify_account_key_column)
212 end
get_verify_login_change_login_and_key(id)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
134 def get_verify_login_change_login_and_key(id)
135   verify_login_change_ds(id).get([verify_login_change_login_column, verify_login_change_key_column])
136 end
handle_duplicate_active_session_id(_e)
[show source]
   # File lib/rodauth/features/active_sessions.rb
79 def handle_duplicate_active_session_id(_e)
80   # Do nothing by default as session is already tracked.  This will result in
81   # the current session and the existing session with the same id
82   # being tracked together, so that a logout of one will logout
83   # the other, and updating the last use on one will update the other,
84   # but this should be acceptable.  However, this can be overridden if different
85   # behavior is desired.
86 end
handle_webauthn_sign_count_verification_error()
[show source]
    # File lib/rodauth/features/webauthn.rb
352 def handle_webauthn_sign_count_verification_error
353   throw_error_reason(:invalid_webauthn_sign_count, invalid_field_error_status, webauthn_auth_param, webauthn_invalid_sign_count_message) 
354 end
has_password?()
[show source]
    # File lib/rodauth/features/base.rb
579 def has_password?
580   return @has_password unless @has_password.nil?
581   return false unless account || session_value
582   @has_password = !!get_password_hash
583 end
hmac_secret_rotation?()
[show source]
    # File lib/rodauth/features/base.rb
851 def hmac_secret_rotation?
852   hmac_secret && hmac_old_secret && hmac_secret != hmac_old_secret
853 end
hook_action(hook_type, action)
[show source]
   # File lib/rodauth/features/audit_logging.rb
31 def hook_action(hook_type, action)
32   super
33   # In after_logout, session is already cleared, so use before_logout in that case
34   if (hook_type == :after || action == :logout) && (id = account ? account_id : session_value)
35     add_audit_log(id, action)
36   end
37 end
http_basic_auth()
[show source]
   # File lib/rodauth/features/http_basic_auth.rb
36 def http_basic_auth
37   unless @checked_http_basic_auth.nil?
38     return (@checked_http_basic_auth ? true : nil)
39   end
40 
41   @checked_http_basic_auth = false
42   return unless token = ((v = request.env['HTTP_AUTHORIZATION']) && v[/\A *Basic (.*)\Z/, 1])
43 
44   username, password = token.unpack("m*").first.split(/:/, 2)
45   return unless username && password
46 
47   catch_error do
48     unless account_from_login(username)
49       after_no_matching_login
50       throw_basic_auth_error(login_param, no_matching_login_message)
51     end
52 
53     before_login_attempt
54 
55     unless open_account?
56       throw_basic_auth_error(login_param, no_matching_login_message)
57     end
58 
59     unless password_match?(password)
60       after_login_failure
61       throw_basic_auth_error(password_param, invalid_password_message)
62     end
63 
64     transaction do
65       before_login
66       login_session('password')
67       after_login
68     end
69 
70     @checked_http_basic_auth = true
71     return true
72   end
73 
74   nil
75 end
inactive_session_cond()
[show source]
    # File lib/rodauth/features/active_sessions.rb
225 def inactive_session_cond
226   cond = session_inactivity_deadline_condition
227   cond2 = session_lifetime_deadline_condition
228   return false unless cond || cond2
229   Sequel.|(*[cond, cond2].compact)
230 end
include_success_messages?()
[show source]
    # File lib/rodauth/features/json.rb
218 def include_success_messages?
219   !json_response_success_key.nil?
220 end
input_field_string(param, id, opts={})
[show source]
    # File lib/rodauth/features/base.rb
221 def input_field_string(param, id, opts={})
222   type = opts.fetch(:type, "text")
223 
224   unless type == "password"
225     value = opts.fetch(:value){scope.h param(param)}
226   end
227 
228   field_class = opts.fetch(:class, "form-control")
229 
230   if autocomplete_for_field?(param) && opts[:autocomplete]
231     autocomplete = "autocomplete=\"#{opts[:autocomplete]}\""
232   end
233 
234   if inputmode_for_field?(param) && opts[:inputmode]
235     inputmode = "inputmode=\"#{opts[:inputmode]}\""
236   end
237 
238   if mark_input_fields_as_required? && opts[:required] != false
239     required = "required=\"required\""
240   end
241 
242   "<input #{opts[:attr]} #{autocomplete} #{inputmode} #{required} #{field_attributes(param)} #{field_error_attributes(param)} type=\"#{type}\" class=\"#{field_class}#{add_field_error_class(param)}\" name=\"#{param}\" id=\"#{id}\" value=\"#{value}\"/> #{formatted_field_error(param) unless opts[:skip_error_message]}"
243 end
inputmode_for_field?(_param)
[show source]
    # File lib/rodauth/features/base.rb
249 def inputmode_for_field?(_param)
250   mark_input_fields_with_inputmode?
251 end
internal_request?()
[show source]
    # File lib/rodauth/features/base.rb
965 def internal_request?
966   false
967 end
internal_request_configuration(&block)
[show source]
    # File lib/rodauth/features/internal_request.rb
387 def internal_request_configuration(&block)
388   @auth.instance_exec do
389     (@internal_request_configuration_blocks ||= []) << block
390   end
391 end
invalid_login_attempted()
[show source]
    # File lib/rodauth/features/lockout.rb
178 def invalid_login_attempted
179   ds = account_login_failures_ds.
180       where(account_login_failures_id_column=>account_id)
181 
182   number = if db.database_type == :postgres
183     ds.returning(account_login_failures_number_column).
184       with_sql(:update_sql, account_login_failures_number_column=>Sequel.expr(account_login_failures_number_column)+1).
185       single_value
186   else
187     # :nocov:
188     if ds.update(account_login_failures_number_column=>Sequel.expr(account_login_failures_number_column)+1) > 0
189       ds.get(account_login_failures_number_column)
190     end
191     # :nocov:
192   end
193 
194   unless number
195     # Ignoring the violation is safe here.  It may allow slightly more than max_invalid_logins invalid logins before
196     # lockout, but allowing a few extra is OK if the race is lost.
197     ignore_uniqueness_violation{account_login_failures_ds.insert(account_login_failures_id_column=>account_id)}
198     number = 1
199   end
200 
201   if number >= max_invalid_logins
202     @unlock_account_key_value = generate_unlock_account_key
203     hash = _setup_account_lockouts_hash(account_id, unlock_account_key_value)
204 
205     if e = raised_uniqueness_violation{account_lockouts_ds.insert(hash)}
206       # If inserting into the lockout table raises a violation, we should just be able to pull the already inserted
207       # key out of it.  If that doesn't return a valid key, we should reraise the error.
208       raise e unless @unlock_account_key_value = account_lockouts_ds.get(account_lockouts_key_column)
209 
210       after_account_lockout
211       show_lockout_page
212     else
213       after_account_lockout
214       e
215     end
216   end
217 end
invalid_previous_password_message()
[show source]
   # File lib/rodauth/features/change_password.rb
72 def invalid_previous_password_message
73   invalid_password_message
74 end
json_request?()
[show source]
   # File lib/rodauth/features/json.rb
57 def json_request?
58   return @json_request unless @json_request.nil?
59   @json_request = !(request.content_type !~ json_request_content_type_regexp)
60 end
json_response()
[show source]
    # File lib/rodauth/features/json.rb
226 def json_response
227   @json_response ||= {}
228 end
json_response_error?()
[show source]
   # File lib/rodauth/features/json.rb
71 def json_response_error?
72   !!json_response[json_response_error_key]
73 end
jwt_cors_allow?()
[show source]
   # File lib/rodauth/features/jwt_cors.rb
15 def jwt_cors_allow?
16   return false unless origin = request.env['HTTP_ORIGIN']
17 
18   case allowed = jwt_cors_allow_origin
19   when String
20     timing_safe_eql?(origin, allowed)
21   when Array
22     allowed.any?{|s| timing_safe_eql?(origin, s)}
23   when Regexp
24     allowed =~ origin
25   when true
26     true
27   else
28     false
29   end
30 end
jwt_payload()
[show source]
    # File lib/rodauth/features/jwt.rb
124 def jwt_payload
125   return @jwt_payload unless @jwt_payload.nil?
126   @jwt_payload = JWT.decode(jwt_token, _jwt_decode_secrets, true, _jwt_decode_opts.merge(:algorithm=>jwt_algorithm))[0]
127 rescue JWT::DecodeError => e
128   rescue_jwt_payload(e)
129 end
jwt_refresh_token_account_ds(account_id)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
174 def jwt_refresh_token_account_ds(account_id)
175   jwt_refresh_token_ds.where(jwt_refresh_token_account_id_column => account_id)
176 end
jwt_refresh_token_account_token_ds(account_id, token_id)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
178 def jwt_refresh_token_account_token_ds(account_id, token_id)
179   jwt_refresh_token_account_ds(account_id).
180     where(jwt_refresh_token_id_column=>token_id)
181 end
jwt_refresh_token_ds()
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
183 def jwt_refresh_token_ds
184   db[jwt_refresh_token_table]
185 end
jwt_refresh_token_insert_hash()
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
197 def jwt_refresh_token_insert_hash
198   hash = {jwt_refresh_token_account_id_column => account_id, jwt_refresh_token_key_column => random_key}
199   set_deadline_value(hash, jwt_refresh_token_deadline_column, jwt_refresh_token_deadline_interval)
200   hash
201 end
jwt_refresh_token_match?(key)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
154 def jwt_refresh_token_match?(key)
155   # We don't need to match tokens if we are requiring a valid current access token
156   return true unless allow_refresh_with_expired_jwt_access_token?
157 
158   # If allowing with expired jwt access token, check the expired session contains
159   # hmac matching submitted and active refresh token.
160   s = session[jwt_refresh_token_hmac_session_key].to_s
161   h = session[jwt_refresh_token_data_session_key].to_s + key
162   timing_safe_eql?(compute_hmac(h), s) || (hmac_secret_rotation? && timing_safe_eql?(compute_old_hmac(h), s))
163 end
jwt_secret()
[show source]
   # File lib/rodauth/features/jwt.rb
68 def jwt_secret
69   raise ConfigurationError, "jwt_secret not set"
70 end
jwt_session_hash()
[show source]
   # File lib/rodauth/features/jwt.rb
72 def jwt_session_hash
73   jwt_session_key ? {jwt_session_key=>session} : session
74 end
jwt_token()
[show source]
   # File lib/rodauth/features/jwt.rb
80 def jwt_token
81   return @jwt_token if @jwt_token
82 
83   if (v = request.env['HTTP_AUTHORIZATION']) && v !~ jwt_authorization_ignore
84     @jwt_token = v.sub(jwt_authorization_remove, '')
85   end
86 end
last_account_activity_at()
[show source]
   # File lib/rodauth/features/account_expiration.rb
27 def last_account_activity_at
28   get_activity_timestamp(session_value, account_activity_last_activity_column)
29 end
last_account_login_at()
[show source]
   # File lib/rodauth/features/account_expiration.rb
31 def last_account_login_at
32   get_activity_timestamp(session_value, account_activity_last_login_column)
33 end
load_memory()
[show source]
    # File lib/rodauth/features/remember.rb
116 def load_memory
117   if logged_in?
118     if extend_remember_deadline_while_logged_in?
119       if account_from_session
120         extend_remember_deadline
121       else
122         forget_login
123         clear_session
124       end
125     end
126   elsif account_from_remember_cookie
127     before_load_memory
128     login_session('remember')
129     extend_remember_deadline if extend_remember_deadline?
130     after_load_memory
131   end
132 end
loaded_templates()
[show source]
    # File lib/rodauth/features/base.rb
905 def loaded_templates
906   []
907 end
locked_out?()
[show source]
    # File lib/rodauth/features/lockout.rb
149 def locked_out?
150   if t = convert_timestamp(account_lockouts_ds.get(account_lockouts_deadline_column))
151     if Time.now < t
152       true
153     else
154       unlock_account
155       false
156     end
157   else
158     false
159   end
160 end
logged_in_via_remember_key?()
[show source]
    # File lib/rodauth/features/remember.rb
176 def logged_in_via_remember_key?
177   authenticated_by.include?('remember')
178 end
login(auth_type)
[show source]
    # File lib/rodauth/features/login.rb
 97 def login(auth_type)
 98   @saved_login_redirect = remove_session_value(login_redirect_session_key)
 99   transaction do
100     before_login
101     login_session(auth_type)
102     yield if block_given?
103     after_login
104   end
105   require_response(:_login_response)
106 end
login_confirm_label()
[show source]
   # File lib/rodauth/features/login_password_requirements_base.rb
49 def login_confirm_label
50   "Confirm #{login_label}"
51 end
login_confirmation_matches?(login, login_confirmation)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
133 def login_confirmation_matches?(login, login_confirmation)
134   login.casecmp?(login_confirmation)
135 end
login_does_not_meet_requirements_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
111 def login_does_not_meet_requirements_message
112   "invalid login#{", #{login_requirement_message}" if login_requirement_message}"
113 end
login_failed_reset_password_request_form()
[show source]
    # File lib/rodauth/features/reset_password.rb
240 def login_failed_reset_password_request_form
241   render("reset-password-request")
242 end
login_field_autocomplete_value()
[show source]
    # File lib/rodauth/features/base.rb
322 def login_field_autocomplete_value
323   login_uses_email? ? "email" : "on"
324 end
login_hidden_field()
[show source]
    # File lib/rodauth/features/login.rb
140 def login_hidden_field
141   "<input type='hidden' name=\"#{login_param}\" value=\"#{scope.h param(login_param)}\" />"
142 end
login_input_type()
[show source]
    # File lib/rodauth/features/base.rb
346 def login_input_type
347   login_uses_email? ? 'email' : 'text'
348 end
login_meets_email_requirements?(login)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
159 def login_meets_email_requirements?(login)
160   return true unless require_email_address_logins?
161   return true if login_valid_email?(login)
162   set_login_requirement_error_message(:login_not_valid_email, login_not_valid_email_message)
163   return false
164 end
login_meets_length_requirements?(login)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
144 def login_meets_length_requirements?(login)
145   if login_minimum_length > login.length
146     set_login_requirement_error_message(:login_too_short, login_too_short_message)
147     false
148   elsif login_maximum_length < login.length
149     set_login_requirement_error_message(:login_too_long, login_too_long_message)
150     false
151   elsif login_maximum_bytes < login.bytesize
152     set_login_requirement_error_message(:login_too_many_bytes, login_too_many_bytes_message)
153     false
154   else
155     true
156   end
157 end
login_meets_requirements?(login)
[show source]
   # File lib/rodauth/features/login_password_requirements_base.rb
57 def login_meets_requirements?(login)
58   login_meets_length_requirements?(login) && \
59     login_meets_email_requirements?(login)
60 end
login_param_value()

The normalized value of the login parameter

[show source]
    # File lib/rodauth/features/base.rb
544 def login_param_value
545   normalize_login(param(login_param))
546 end
login_required()
[show source]
    # File lib/rodauth/features/base.rb
365 def login_required
366   set_redirect_error_status(login_required_error_status)
367   set_error_reason :login_required
368   set_redirect_error_flash require_login_error_flash
369   redirect require_login_redirect
370 end
login_return_to_requested_location_path()
[show source]
    # File lib/rodauth/features/login.rb
115 def login_return_to_requested_location_path
116   request.fullpath if request.get?
117 end
login_session(auth_type)
[show source]
    # File lib/rodauth/features/base.rb
506 def login_session(auth_type)
507   update_session
508   set_session_value(authenticated_by_session_key, [auth_type])
509 end
login_too_long_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
115 def login_too_long_message
116   "maximum #{login_maximum_length} characters"
117 end
login_too_many_bytes_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
119 def login_too_many_bytes_message
120   "maximum #{login_maximum_bytes} bytes"
121 end
login_too_short_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
123 def login_too_short_message
124   "minimum #{login_minimum_length} characters"
125 end
login_uses_email?()
[show source]
    # File lib/rodauth/features/base.rb
350 def login_uses_email?
351   login_column == :email
352 end
login_valid_email?(login)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
166 def login_valid_email?(login)
167   login =~ login_email_regexp
168 end
logout()
[show source]
   # File lib/rodauth/features/logout.rb
34 def logout
35   clear_session
36 end
logout_additional_form_tags()
[show source]
    # File lib/rodauth/features/active_sessions.rb
120 def logout_additional_form_tags
121   super.to_s + render('global-logout-field')
122 end
modifications_require_password?()
[show source]
    # File lib/rodauth/features/base.rb
571 def modifications_require_password?
572   has_password?
573 end
new_account(login)
[show source]
   # File lib/rodauth/features/create_account.rb
96 def new_account(login)
97   @account = _new_account(login)
98 end
new_recovery_code()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
242 def new_recovery_code
243   random_key
244 end
new_webauthn_credential()
[show source]
    # File lib/rodauth/features/webauthn.rb
302 def new_webauthn_credential
303   WebAuthn::Credential.options_for_create(
304     :timeout => webauthn_setup_timeout,
305     :user => {:id=>account_webauthn_user_id, :name=>webauthn_user_name},
306     :authenticator_selection => webauthn_authenticator_selection,
307     :attestation => webauthn_attestation,
308     :extensions => webauthn_extensions,
309     :exclude => account_webauthn_ids,
310     **webauthn_create_relying_party_opts
311   )
312 end
no_longer_active_session()
[show source]
   # File lib/rodauth/features/active_sessions.rb
62 def no_longer_active_session
63   clear_session
64   set_redirect_error_status inactive_session_error_status
65   set_error_reason :inactive_session
66   set_redirect_error_flash active_sessions_error_flash
67   redirect active_sessions_redirect
68 end
normalize_login(login)
[show source]
    # File lib/rodauth/features/base.rb
548 def normalize_login(login)
549   login
550 end
normalize_session_or_flash_key(key)
[show source]
    # File lib/rodauth/features/base.rb
722 def normalize_session_or_flash_key(key)
723   scope.opts[:sessions_convert_symbols] ? key.to_s : key
724 end
null_byte_parameter_value(key, value)

Return nil by default for values with null bytes

[show source]
    # File lib/rodauth/features/base.rb
553 def null_byte_parameter_value(key, value)
554   nil
555 end
only_json?()
[show source]
    # File lib/rodauth/features/base.rb
465 def only_json?
466   scope.class.opts[:rodauth_json] == :only
467 end
open_account?()
[show source]
    # File lib/rodauth/features/base.rb
314 def open_account?
315   skip_status_checks? || account[account_status_column] == account_open_status_value 
316 end
otp_add_key()
[show source]
    # File lib/rodauth/features/otp.rb
285 def otp_add_key
286   _otp_add_key(otp_key)
287   super if defined?(super)
288 end
otp_available?()
[show source]
    # File lib/rodauth/features/otp.rb
245 def otp_available?
246   otp_exists? && !otp_locked_out?
247 end
otp_exists?()
[show source]
    # File lib/rodauth/features/otp.rb
249 def otp_exists?
250   !!otp_key
251 end
otp_hmac_old_secret(key)
[show source]
    # File lib/rodauth/features/otp.rb
385 def otp_hmac_old_secret(key)
386   base32_encode(compute_raw_hmac_with_secret(ROTP::Base32.decode(key), hmac_old_secret), key.bytesize)
387 end
otp_hmac_secret(key)
[show source]
    # File lib/rodauth/features/otp.rb
381 def otp_hmac_secret(key)
382   base32_encode(compute_raw_hmac(ROTP::Base32.decode(key)), key.bytesize)
383 end
otp_issuer()
[show source]
    # File lib/rodauth/features/otp.rb
316 def otp_issuer
317   domain
318 end
otp_key_ds()
[show source]
    # File lib/rodauth/features/otp.rb
452 def otp_key_ds
453   db[otp_keys_table].where(otp_keys_id_column=>session_value)
454 end
otp_keys_use_hmac?()
[show source]
    # File lib/rodauth/features/otp.rb
337 def otp_keys_use_hmac?
338   !!hmac_secret
339 end
otp_last_use()
[show source]
    # File lib/rodauth/features/otp.rb
296 def otp_last_use
297   convert_timestamp(otp_key_ds.get(otp_keys_last_use_column))
298 end
otp_locked_out?()
[show source]
    # File lib/rodauth/features/otp.rb
308 def otp_locked_out?
309   otp_key_ds.get(otp_keys_failures_column) >= otp_auth_failures_limit
310 end
otp_lockout_redirect()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
212 def otp_lockout_redirect
213   otp_unlock_path
214 end
otp_new_secret()
[show source]
    # File lib/rodauth/features/otp.rb
403 def otp_new_secret
404   ROTP::Base32.random_base32.downcase
405 end
otp_provisioning_name()
[show source]
    # File lib/rodauth/features/otp.rb
320 def otp_provisioning_name
321   account[login_column]
322 end
otp_provisioning_uri()
[show source]
    # File lib/rodauth/features/otp.rb
312 def otp_provisioning_uri
313   otp.provisioning_uri(otp_provisioning_name)
314 end
otp_qr_code()
[show source]
    # File lib/rodauth/features/otp.rb
324 def otp_qr_code
325   svg = RQRCode::QRCode.new(otp_provisioning_uri).as_svg(:module_size=>8, :viewbox=>true, :use_path=>true, :fill=>"fff")
326   svg.sub(/\A<\?xml version="1\.0" standalone="yes"\?>/, '')
327 end
otp_record_authentication_failure()
[show source]
    # File lib/rodauth/features/otp.rb
300 def otp_record_authentication_failure
301   otp_key_ds.update(otp_keys_failures_column=>Sequel.identifier(otp_keys_failures_column) + 1)
302 end
otp_remove()
[show source]
    # File lib/rodauth/features/otp.rb
280 def otp_remove
281   otp_key_ds.delete
282   @otp_key = false
283 end
otp_remove_auth_failures()
[show source]
    # File lib/rodauth/features/otp.rb
304 def otp_remove_auth_failures
305   otp_key_ds.update(otp_keys_failures_column=>0)
306 end
otp_tmp_key(secret)
[show source]
    # File lib/rodauth/features/otp.rb
376 def otp_tmp_key(secret)
377   _otp_tmp_key(secret)
378   clear_cached_otp
379 end
otp_unlock_auth_failure()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
147 def otp_unlock_auth_failure
148   h = {
149     otp_unlock_num_successes_column=>0,
150     otp_unlock_next_auth_attempt_after_column=>Sequel.date_add(Sequel::CURRENT_TIMESTAMP, :seconds=>otp_unlock_auth_failure_cooldown_seconds)
151   }
152 
153   if otp_unlock_ds.update(h) == 0
154     h[otp_unlock_id_column] = session_value
155 
156     # If row already exists when inserting, no need to do anything
157     raises_uniqueness_violation?{otp_unlock_ds.insert(h)}
158   end
159 end
otp_unlock_auth_success()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
161 def otp_unlock_auth_success
162   deadline = Sequel.date_add(Sequel::CURRENT_TIMESTAMP, :seconds=>otp_unlock_success_cooldown_seconds)
163 
164   # Add WHERE to avoid possible race condition when multiple unlock auth requests
165   # are sent at the same time (only the first should increment num successes).
166   if otp_unlock_ds.
167       where(Sequel[otp_unlock_next_auth_attempt_after_column] < Sequel::CURRENT_TIMESTAMP).
168       update(
169         otp_unlock_num_successes_column=>Sequel[otp_unlock_num_successes_column]+1,
170         otp_unlock_next_auth_attempt_after_column=>deadline
171       ) == 0
172 
173     # Ignore uniqueness errors when inserting after a failed update,
174     # which could be caused due to the race condition mentioned above.
175     raises_uniqueness_violation? do
176       otp_unlock_ds.insert(
177         otp_unlock_id_column=>session_value,
178         otp_unlock_next_auth_attempt_after_column=>deadline
179       )
180     end
181   end
182 
183   @otp_unlock_data = nil
184   # :nocov:
185   if otp_unlock_data
186   # :nocov:
187     if otp_unlock_num_successes >= otp_unlock_auths_required
188       # At least the requisite number of consecutive successful unlock
189       # authentications. Unlock OTP authentication.
190       otp_key_ds.update(otp_keys_failures_column => 0)
191 
192       # Remove OTP unlock metadata when unlocking OTP authentication
193       otp_unlock_reset
194     # else
195     #  # Still need additional consecutive successful unlock attempts.
196     end
197   # else
198   #  # if row isn't available, probably the process was reset during this,
199   #  # and it's safe to do nothing in that case.
200   end
201 end
otp_unlock_available?()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
117 def otp_unlock_available?
118   if otp_unlock_data
119     next_auth_attempt_after = otp_unlock_next_auth_attempt_after
120     current_timestamp = Time.now
121 
122     if (next_auth_attempt_after < current_timestamp - otp_unlock_deadline_seconds)
123       # Unlock process not fully completed within deadline, reset process
124       otp_unlock_reset
125       true
126     else
127       if next_auth_attempt_after > current_timestamp
128         # If next auth attempt after timestamp is in the future, that means the next
129         # unlock attempt cannot happen until then.
130         false 
131       else
132         if otp_unlock_num_successes == 0
133           # 0 value indicates previous attempt was a failure. Since failure cooldown
134           # period has passed, reset process so user gets full deadline period
135           otp_unlock_reset
136         end
137         true
138       end
139     end
140   else
141     # No row means no unlock attempts yet (or previous attempt was more than the
142     # deadline account, so unlocking is available
143     true
144   end
145 end
otp_unlock_data()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
242 def otp_unlock_data
243   @otp_unlock_data ||= otp_unlock_ds.first
244 end
otp_unlock_deadline()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
224 def otp_unlock_deadline
225   otp_unlock_next_auth_attempt_after + otp_unlock_deadline_seconds
226 end
otp_unlock_deadline_passed?()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
203 def otp_unlock_deadline_passed?
204   otp_unlock_data ? (otp_unlock_next_auth_attempt_after < Time.now - otp_unlock_deadline_seconds) : false
205 end
otp_unlock_ds()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
255 def otp_unlock_ds
256   db[otp_unlock_table].where(otp_unlock_id_column=>session_value)
257 end
otp_unlock_next_auth_attempt_after()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
216 def otp_unlock_next_auth_attempt_after
217   if otp_unlock_data
218     convert_timestamp(otp_unlock_data[otp_unlock_next_auth_attempt_after_column])
219   else
220     Time.now
221   end
222 end
otp_unlock_not_available_set_refresh_header()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
232 def otp_unlock_not_available_set_refresh_header
233   response.headers["refresh"] = ((otp_unlock_next_auth_attempt_after - Time.now).to_i + 1).to_s
234 end
otp_unlock_num_successes()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
228 def otp_unlock_num_successes
229   otp_unlock_data ? otp_unlock_data[otp_unlock_num_successes_column] : 0
230 end
otp_unlock_refresh_tag()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
207 def otp_unlock_refresh_tag
208   # RODAUTH3: Remove
209   "<meta http-equiv=\"refresh\" content=\"#{(otp_unlock_next_auth_attempt_after - Time.now).to_i + 1}\">"
210 end
otp_unlock_reset()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
250 def otp_unlock_reset
251   otp_unlock_ds.delete
252   @otp_unlock_data = nil
253 end
otp_unlock_success_cooldown_seconds()
[show source]
    # File lib/rodauth/features/otp_unlock.rb
246 def otp_unlock_success_cooldown_seconds
247   (_otp_interval+(otp_drift||0))*2
248 end
otp_update_last_use()
[show source]
    # File lib/rodauth/features/otp.rb
290 def otp_update_last_use
291   otp_key_ds.
292     where(Sequel.date_add(otp_keys_last_use_column, :seconds=>_otp_interval) < Sequel::CURRENT_TIMESTAMP).
293     update(otp_keys_last_use_column=>Sequel::CURRENT_TIMESTAMP) == 1
294 end
otp_user_key()
[show source]
    # File lib/rodauth/features/otp.rb
329 def otp_user_key
330   @otp_user_key ||= if otp_keys_use_hmac?
331     otp_hmac_secret(otp_key)
332   else
333     otp_key
334   end
335 end
otp_valid_code?(ot_pass)
[show source]
    # File lib/rodauth/features/otp.rb
253 def otp_valid_code?(ot_pass)
254   if _otp_valid_code?(ot_pass, otp)
255     true
256   elsif hmac_secret_rotation? && _otp_valid_code?(ot_pass, _otp_for_key(otp_hmac_old_secret(otp_key)))
257     _otp_valid_code_for_old_secret
258     true
259   else
260     false
261   end
262 end
otp_valid_key?(secret)
[show source]
    # File lib/rodauth/features/otp.rb
389 def otp_valid_key?(secret)
390   return false unless secret =~ /\A([a-z2-7]{16}|[a-z2-7]{32})\z/
391   if otp_keys_use_hmac?
392     # Purposely do not allow creating new OTPs with old secrets,
393     # since OTP rotation is difficult.  The user will get shown
394     # the same page with an updated secret, which they can submit
395     # to setup OTP.
396     timing_safe_eql?(otp_hmac_secret(param(otp_setup_raw_param)), secret)
397   else
398     true
399   end
400 end
over_max_bytesize_param_value(key, value)

Return nil by default for values over maximum bytesize.

[show source]
    # File lib/rodauth/features/base.rb
539 def over_max_bytesize_param_value(key, value)
540   nil
541 end
param(key)

Return a string for the parameter name. This will be an empty string if the parameter doesn’t exist.

[show source]
    # File lib/rodauth/features/base.rb
522 def param(key)
523   param_or_nil(key).to_s
524 end
param_or_nil(key)

Return a string for the parameter name, or nil if there is no parameter with that name.

[show source]
    # File lib/rodauth/features/base.rb
528 def param_or_nil(key)
529   value = raw_param(key)
530   unless value.nil?
531     value = value.to_s
532     value = over_max_bytesize_param_value(key, value) if max_param_bytesize && value.bytesize > max_param_bytesize
533     value = null_byte_parameter_value(key, value) if value && value.include?("\0")
534   end
535   value
536 end
password_confirm_label()
[show source]
   # File lib/rodauth/features/login_password_requirements_base.rb
53 def password_confirm_label
54   "Confirm #{password_label}"
55 end
password_does_not_contain_null_byte?(password)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
185 def password_does_not_contain_null_byte?(password)
186   return true unless password.include?("\0")
187   set_password_requirement_error_message(:password_contains_null_byte, contains_null_byte_message)
188   false
189 end
password_does_not_meet_requirements_message()
[show source]
   # File lib/rodauth/features/login_password_requirements_base.rb
90 def password_does_not_meet_requirements_message
91   "invalid password, does not meet requirements#{" (#{password_requirement_message})" if password_requirement_message}"
92 end
password_doesnt_match_previous_password?(password)
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
52 def password_doesnt_match_previous_password?(password)
53   match = if use_database_authentication_functions?
54     salts = previous_password_ds.
55       select_map([previous_password_id_column, Sequel.function(function_name(:rodauth_get_previous_salt), previous_password_id_column).as(:salt)])
56     return true if salts.empty?
57 
58     salts.any? do |hash_id, salt|
59       database_function_password_match?(:rodauth_previous_password_hash_match, hash_id, password, salt)
60     end
61   else
62     # :nocov:
63     previous_password_ds.select_map(previous_password_hash_column).any? do |hash|
64       password_hash_match?(hash, password)
65     end
66     # :nocov:
67   end
68 
69   return true unless match
70   set_password_requirement_error_message(:password_same_as_previous_password, password_same_as_previous_password_message)
71   false
72 end
password_expiration_ds()
[show source]
    # File lib/rodauth/features/password_expiration.rb
107 def password_expiration_ds
108   db[password_expiration_table].where(password_expiration_id_column=>account_id)
109 end
password_expired?()
[show source]
   # File lib/rodauth/features/password_expiration.rb
68 def password_expired?
69   if password_changed_at = session[password_changed_at_session_key]
70     return password_changed_at + require_password_change_after < Time.now.to_i
71   end
72 
73   account_from_session
74   if password_changed_at = get_password_changed_at
75     set_session_value(password_changed_at_session_key, password_changed_at.to_i)
76     password_changed_at + require_password_change_after < Time.now
77   else
78     set_session_value(password_changed_at_session_key, password_expiration_default ? 0 : 2147483647)
79     password_expiration_default
80   end
81 end
password_field_autocomplete_value()
[show source]
    # File lib/rodauth/features/base.rb
326 def password_field_autocomplete_value
327   @password_field_autocomplete_value || 'current-password'
328 end
password_has_enough_character_groups?(password)
[show source]
   # File lib/rodauth/features/password_complexity.rb
54 def password_has_enough_character_groups?(password)
55   return true if password.length > password_max_length_for_groups_check
56   return true if password_character_groups.select{|re| password =~ re}.length >= password_min_groups
57   set_password_requirement_error_message(:not_enough_character_groups_in_password, password_not_enough_character_groups_message)
58   false
59 end
password_has_no_invalid_pattern?(password)
[show source]
   # File lib/rodauth/features/password_complexity.rb
61 def password_has_no_invalid_pattern?(password)
62   return true unless password_invalid_pattern
63   return true if password !~ password_invalid_pattern
64   set_password_requirement_error_message(:invalid_password_pattern, password_invalid_pattern_message)
65   false
66 end
password_hash(password)
[show source]
   # File lib/rodauth/features/argon2.rb
21 def password_hash(password)
22   return super unless use_argon2?
23 
24   if secret = argon2_secret
25     argon2_params = Hash[password_hash_cost]
26     argon2_params[:secret] = secret
27   else
28     argon2_params = password_hash_cost
29   end
30   ::Argon2::Password.new(argon2_params).create(password)
31 end
password_hash_cost()
[show source]
   # File lib/rodauth/features/argon2.rb
47 def password_hash_cost
48   return super unless use_argon2?
49   argon2_hash_cost
50 end
password_hash_ds()
[show source]
    # File lib/rodauth/features/base.rb
895 def password_hash_ds
896   db[password_hash_table].where(password_hash_id_column=>account ? account_id : session_value)
897 end
password_hash_match?(hash, password)
[show source]
   # File lib/rodauth/features/argon2.rb
52 def password_hash_match?(hash, password)
53   return super unless argon2_hash_algorithm?(hash)
54   argon2_password_hash_match?(hash, password)
55 end
password_hash_using_salt(password, salt)
[show source]
   # File lib/rodauth/features/argon2.rb
57 def password_hash_using_salt(password, salt)
58   return super unless argon2_hash_algorithm?(salt)
59   argon2_password_hash_using_salt_and_secret(password, salt, argon2_secret)
60 end
password_match?(password)
[show source]
    # File lib/rodauth/features/base.rb
487 def password_match?(password)
488   if hash = get_password_hash
489     if account_password_hash_column || !use_database_authentication_functions?
490       password_hash_match?(hash, password)
491     else
492       database_function_password_match?(:rodauth_valid_password_hash, account_id, password, hash)
493     end 
494   end
495 end
password_meets_length_requirements?(password)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
170 def password_meets_length_requirements?(password)
171   if password_minimum_length > password.length
172     set_password_requirement_error_message(:password_too_short, password_too_short_message)
173     false
174   elsif password_maximum_length && password_maximum_length < password.length
175     set_password_requirement_error_message(:password_too_long, password_too_long_message)
176     false
177   elsif password_maximum_bytes && password_maximum_bytes < password.bytesize
178     set_password_requirement_error_message(:password_too_many_bytes, password_too_many_bytes_message)
179     false
180   else
181     true
182   end
183 end
password_meets_requirements?(password)
[show source]
   # File lib/rodauth/features/disallow_common_passwords.rb
13 def password_meets_requirements?(password)
14   super && password_not_one_of_the_most_common?(password)
15 end
password_not_in_dictionary?(password)
[show source]
   # File lib/rodauth/features/password_complexity.rb
75 def password_not_in_dictionary?(password)
76   return true unless dict = password_dictionary
77   return true unless password =~ /\A(?:\d*)([A-Za-z!@$+|][A-Za-z!@$+|0134578]+[A-Za-z!@$+|])(?:\d*)\z/
78   word = $1.downcase.tr('!@$+|0134578', 'iastloleastb')
79   return true if !dict.include?(word)
80   set_password_requirement_error_message(:password_in_dictionary, password_in_dictionary_message)
81   false
82 end
password_not_one_of_the_most_common?(password)
[show source]
   # File lib/rodauth/features/disallow_common_passwords.rb
33 def password_not_one_of_the_most_common?(password)
34   return true unless password_one_of_most_common?(password)
35   set_password_requirement_error_message(:password_is_one_of_the_most_common, password_is_one_of_the_most_common_message)
36   false
37 end
password_not_too_many_repeating_characters?(password)
[show source]
   # File lib/rodauth/features/password_complexity.rb
68 def password_not_too_many_repeating_characters?(password)
69   return true if password_max_repeating_characters < 2
70   return true if password !~ /(.)(\1){#{password_max_repeating_characters-1}}/ 
71   set_password_requirement_error_message(:too_many_repeating_characters_in_password, password_too_many_repeating_characters_message)
72   false
73 end
password_one_of_most_common?(password)
[show source]
   # File lib/rodauth/features/disallow_common_passwords.rb
27 def password_one_of_most_common?(password)
28   most_common_passwords.include?(password)
29 end
password_recently_entered?()
[show source]
   # File lib/rodauth/features/password_grace_period.rb
24 def password_recently_entered?
25   return false unless last_password_entry = session[last_password_entry_session_key]
26   last_password_entry + password_grace_period > Time.now.to_i
27 end
password_reset_ds(id=account_id)
[show source]
    # File lib/rodauth/features/reset_password.rb
254 def password_reset_ds(id=account_id)
255   db[reset_password_table].where(reset_password_id_column=>id)
256 end
password_too_long_message()
[show source]
   # File lib/rodauth/features/login_password_requirements_base.rb
94 def password_too_long_message
95   "maximum #{password_maximum_length} characters"
96 end
password_too_many_bytes_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
 98 def password_too_many_bytes_message
 99   "maximum #{password_maximum_bytes} bytes"
100 end
password_too_short_message()
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
102 def password_too_short_message
103   "minimum #{password_minimum_length} characters"
104 end
possible_authentication_methods()
[show source]
    # File lib/rodauth/features/base.rb
575 def possible_authentication_methods
576   has_password? ? ['password'] : []
577 end
post_configure()
[show source]
    # File lib/rodauth/features/base.rb
469 def post_configure
470   require 'bcrypt' if require_bcrypt?
471   db.extension :date_arithmetic if use_date_arithmetic?
472 
473   if method(:convert_token_id_to_integer?).owner == Rodauth::Base && (db rescue false) && db.table_exists?(accounts_table) && db.schema(accounts_table).find{|col, v| break v[:type] == :integer if col == account_id_column}
474     self.class.send(:define_method, :convert_token_id_to_integer?){true}
475   end
476 
477   route_hash= {}
478   self.class.routes.each do |meth|
479     route_meth = "#{meth.to_s.sub(/\Ahandle_/, '')}_route"
480     if route = send(route_meth)
481       route_hash["/#{route}"] = meth
482     end
483   end
484   self.class.route_hash = route_hash.freeze
485 end
previous_password_ds()
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
96 def previous_password_ds
97   db[previous_password_hash_table].where(previous_password_account_id_column=>account_id)
98 end
raises_uniqueness_violation?(&block)

In cases where retrying on uniqueness violations cannot work, this will detect whether a uniqueness violation is raised by the block and return the exception if so. This method should be used if you don’t care about the exception itself.

[show source]
    # File lib/rodauth/features/base.rb
924 def raises_uniqueness_violation?(&block)
925   transaction(:savepoint=>:only, &block)
926   false
927 rescue unique_constraint_violation_class => e
928   e
929 end
random_key()
[show source]
    # File lib/rodauth/features/base.rb
713 def random_key
714   SecureRandom.urlsafe_base64(32)
715 end
raw_param(key)
[show source]
    # File lib/rodauth/features/base.rb
557 def raw_param(key)
558   request.params[key]
559 end
recovery_code_match?(code)
[show source]
    # File lib/rodauth/features/recovery_codes.rb
162 def recovery_code_match?(code)
163   recovery_codes.each do |s|
164     if timing_safe_eql?(code, s)
165       recovery_codes_ds.where(recovery_codes_column=>code).delete
166       if recovery_codes_primary?
167         add_recovery_code
168       end
169       return true
170     end
171   end
172   false
173 end
recovery_codes_available?()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
198 def recovery_codes_available?
199   !recovery_codes_ds.empty?
200 end
recovery_codes_ds()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
266 def recovery_codes_ds
267   db[recovery_codes_table].where(recovery_codes_id_column=>session_value)
268 end
recovery_codes_primary?()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
246 def recovery_codes_primary?
247   (features & [:otp, :sms_codes, :webauthn]).empty?
248 end
recovery_codes_remove()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
158 def recovery_codes_remove
159   recovery_codes_ds.delete
160 end
redirect(path)
[show source]
    # File lib/rodauth/features/base.rb
690 def redirect(path)
691   request.redirect(path)
692 end
remember_key_ds(id=account_id)
[show source]
    # File lib/rodauth/features/remember.rb
263 def remember_key_ds(id=account_id)
264   db[remember_table].where(remember_id_column=>id)
265 end
remember_login()
[show source]
    # File lib/rodauth/features/remember.rb
134 def remember_login
135   get_remember_key
136   set_remember_cookie
137   set_session_value(remember_deadline_extended_session_key, Time.now.to_i) if extend_remember_deadline?
138 end
remembered_session_id()
[show source]
    # File lib/rodauth/features/remember.rb
 93 def remembered_session_id
 94   return unless cookie = _get_remember_cookie
 95   id, key = cookie.split('_', 2)
 96   return unless id && key
 97 
 98   actual, deadline = active_remember_key_ds(id).get([remember_key_column, remember_deadline_column])
 99   return unless actual
100 
101   if hmac_secret && !(valid = timing_safe_eql?(key, compute_hmac(actual)))
102     if hmac_secret_rotation? && (valid = timing_safe_eql?(key, compute_old_hmac(actual)))
103       _set_remember_cookie(id, actual, deadline)
104     elsif !(raw_remember_token_deadline && raw_remember_token_deadline > convert_timestamp(deadline))
105       return
106     end
107   end
108 
109   unless valid || timing_safe_eql?(key, actual)
110     return
111   end
112 
113   id
114 end
remove_active_session(session_id)
[show source]
   # File lib/rodauth/features/active_sessions.rb
94 def remove_active_session(session_id)
95   active_sessions_ds.where(active_sessions_session_id_column=>session_id).delete
96 end
remove_all_active_sessions()
[show source]
    # File lib/rodauth/features/active_sessions.rb
 98 def remove_all_active_sessions
 99   active_sessions_ds.delete
100 end
remove_all_active_sessions_except_current()
[show source]
    # File lib/rodauth/features/active_sessions.rb
106 def remove_all_active_sessions_except_current 
107   if session_id = session[session_id_session_key]
108     remove_all_active_sessions_except_for(session_id)
109   else
110     remove_all_active_sessions
111   end
112 end
remove_all_active_sessions_except_for(session_id)
[show source]
    # File lib/rodauth/features/active_sessions.rb
102 def remove_all_active_sessions_except_for(session_id)
103   active_sessions_ds.exclude(active_sessions_session_id_column=>compute_hmacs(session_id)).delete
104 end
remove_all_webauthn_keys_and_user_ids()
[show source]
    # File lib/rodauth/features/webauthn.rb
381 def remove_all_webauthn_keys_and_user_ids
382   webauthn_user_ids_ds.delete
383   webauthn_keys_ds.delete
384 end
remove_current_session()
[show source]
   # File lib/rodauth/features/active_sessions.rb
88 def remove_current_session
89   if session_id = session[session_id_session_key]
90     remove_active_session(compute_hmacs(session_id))
91   end
92 end
remove_email_auth_key()
[show source]
    # File lib/rodauth/features/email_auth.rb
129 def remove_email_auth_key
130   email_auth_ds.delete
131 end
remove_inactive_sessions()
[show source]
    # File lib/rodauth/features/active_sessions.rb
114 def remove_inactive_sessions
115   if cond = inactive_session_cond
116     active_sessions_ds.where(cond).delete
117   end
118 end
remove_jwt_refresh_token_key(token)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
187 def remove_jwt_refresh_token_key(token)
188   account_id, token_id, _ = _account_refresh_token_split(token)
189   jwt_refresh_token_account_token_ds(account_id, token_id).delete
190 end
remove_lockout_metadata()
[show source]
    # File lib/rodauth/features/lockout.rb
282 def remove_lockout_metadata
283   account_login_failures_ds.delete
284   account_lockouts_ds.delete
285 end
remove_remember_key(id=account_id)
[show source]
    # File lib/rodauth/features/remember.rb
172 def remove_remember_key(id=account_id)
173   remember_key_ds(id).delete
174 end
remove_reset_password_key()
[show source]
    # File lib/rodauth/features/reset_password.rb
182 def remove_reset_password_key
183   password_reset_ds.delete
184 end
remove_session_value(key)
[show source]
    # File lib/rodauth/features/base.rb
982 def remove_session_value(key)
983   session.delete(key)
984 end
remove_verify_account_key()
[show source]
    # File lib/rodauth/features/verify_account.rb
169 def remove_verify_account_key
170   verify_account_ds.delete
171 end
remove_verify_login_change_key()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
110 def remove_verify_login_change_key
111   verify_login_change_ds.delete
112 end
remove_webauthn_key(webauthn_id)
[show source]
    # File lib/rodauth/features/webauthn.rb
377 def remove_webauthn_key(webauthn_id)
378   webauthn_keys_ds.where(webauthn_keys_webauthn_id_column=>webauthn_id).delete == 1
379 end
render(page)
[show source]
    # File lib/rodauth/features/base.rb
461 def render(page)
462   _view(:render, page)
463 end
render_multi_phase_login_forms()
[show source]
    # File lib/rodauth/features/login.rb
148 def render_multi_phase_login_forms
149   multi_phase_login_forms.sort.map{|_, form, _| form}.join("\n")
150 end
request()
[show source]
    # File lib/rodauth/features/base.rb
182 def request
183   scope.request
184 end
require_account()
[show source]
    # File lib/rodauth/features/base.rb
406 def require_account
407   require_authentication
408   require_account_session
409 end
require_account_session()
[show source]
    # File lib/rodauth/features/base.rb
731 def require_account_session
732   unless account_from_session
733     clear_session
734     login_required
735   end
736 end
require_authentication()
[show source]
    # File lib/rodauth/features/base.rb
402 def require_authentication
403   require_login
404 end
require_current_password()
[show source]
   # File lib/rodauth/features/password_expiration.rb
61 def require_current_password
62   if authenticated? && password_expired? && password_change_needed_redirect != request.path_info
63     set_redirect_error_flash password_expiration_error_flash
64     redirect password_change_needed_redirect
65   end
66 end
require_http_basic_auth()
[show source]
   # File lib/rodauth/features/http_basic_auth.rb
29 def require_http_basic_auth
30   unless http_basic_auth
31     set_http_basic_auth_error_response
32     return_response
33   end
34 end
require_login()
[show source]
    # File lib/rodauth/features/base.rb
394 def require_login
395   login_required unless logged_in?
396 end
require_login_confirmation?()
[show source]
    # File lib/rodauth/features/verify_account.rb
161 def require_login_confirmation?
162   false
163 end
require_login_redirect()
[show source]
    # File lib/rodauth/features/login.rb
152 def require_login_redirect
153   login_path
154 end
require_otp_setup()
[show source]
    # File lib/rodauth/features/otp.rb
236 def require_otp_setup
237   unless otp_exists?
238     set_redirect_error_status(two_factor_not_setup_error_status)
239     set_error_reason :two_factor_not_setup
240     set_redirect_error_flash two_factor_not_setup_error_flash
241     redirect two_factor_need_setup_redirect
242   end
243 end
require_password_authentication()
[show source]
   # File lib/rodauth/features/confirm_password.rb
51 def require_password_authentication
52   require_login
53 
54   if require_password_authentication? && has_password?
55     set_redirect_error_status(password_authentication_required_error_status)
56     set_error_reason :password_authentication_required
57     set_redirect_error_flash password_authentication_required_error_flash
58     set_session_value(confirm_password_redirect_session_key, request.fullpath)
59     redirect password_authentication_required_redirect
60   end
61 end
require_password_authentication?()
[show source]
   # File lib/rodauth/features/confirm_password.rb
87 def require_password_authentication?
88   return true if defined?(super) && super
89   !authenticated_by.include?('password')
90 end
require_response(meth)
[show source]
    # File lib/rodauth/features/base.rb
973 def require_response(meth)
974   send(meth)
975   raise ConfigurationError, "#{meth.to_s.sub(/\A_/, '')} overridden without returning a response (should use redirect or request.halt)."
976 end
require_sms_available()
[show source]
    # File lib/rodauth/features/sms_codes.rb
338 def require_sms_available
339   require_sms_setup
340 
341   if sms_locked_out?
342     set_redirect_error_status(lockout_error_status)
343     set_error_reason :sms_locked_out
344     set_redirect_error_flash sms_lockout_error_flash
345     redirect sms_lockout_redirect
346   end
347 end
require_sms_not_setup()
[show source]
    # File lib/rodauth/features/sms_codes.rb
329 def require_sms_not_setup
330   if sms_setup?
331     set_redirect_error_status(sms_already_setup_error_status)
332     set_error_reason :sms_already_setup
333     set_redirect_error_flash sms_already_setup_error_flash
334     redirect sms_already_setup_redirect
335   end
336 end
require_sms_setup()
[show source]
    # File lib/rodauth/features/sms_codes.rb
320 def require_sms_setup
321   unless sms_setup?
322     set_redirect_error_status(two_factor_not_setup_error_status)
323     set_error_reason :sms_not_setup
324     set_redirect_error_flash sms_not_setup_error_flash
325     redirect sms_needs_setup_redirect
326   end
327 end
require_two_factor_authenticated()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
158 def require_two_factor_authenticated
159   unless two_factor_authenticated?
160     if two_factor_auth_return_to_requested_location?
161       set_session_value(two_factor_auth_redirect_session_key, request.fullpath)
162     end
163     set_redirect_error_status(two_factor_need_authentication_error_status)
164     set_error_reason :two_factor_need_authentication
165     set_redirect_error_flash two_factor_need_authentication_error_flash
166     redirect two_factor_auth_required_redirect
167   end
168 end
require_two_factor_not_authenticated(auth_type = nil)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
149 def require_two_factor_not_authenticated(auth_type = nil)
150   if two_factor_authenticated? || (auth_type && two_factor_login_type_match?(auth_type))
151     set_redirect_error_status(two_factor_already_authenticated_error_status)
152     set_error_reason :two_factor_already_authenticated
153     set_redirect_error_flash two_factor_already_authenticated_error_flash
154     redirect two_factor_already_authenticated_redirect
155   end
156 end
require_two_factor_setup()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
137 def require_two_factor_setup
138   # Avoid database query if already authenticated via 2nd factor
139   return if two_factor_authenticated?
140 
141   return if uses_two_factor_authentication?
142 
143   set_redirect_error_status(two_factor_not_setup_error_status)
144   set_error_reason :two_factor_not_setup
145   set_redirect_error_flash two_factor_not_setup_error_flash
146   redirect two_factor_need_setup_redirect
147 end
require_webauthn_setup()
[show source]
    # File lib/rodauth/features/webauthn.rb
390 def require_webauthn_setup
391   unless webauthn_setup?
392     set_redirect_error_status(webauthn_not_setup_error_status)
393     set_error_reason :webauthn_not_setup
394     set_redirect_error_flash webauthn_not_setup_error_flash
395     redirect two_factor_need_setup_redirect
396   end
397 end
rescue_jwt_payload(_)
[show source]
    # File lib/rodauth/features/jwt.rb
131 def rescue_jwt_payload(_)
132   @jwt_payload = false
133 end
reset_password_account_status_value()
[show source]
    # File lib/rodauth/features/reset_password.rb
262 def reset_password_account_status_value
263   account_open_status_value
264 end
reset_password_email_recently_sent?()
[show source]
    # File lib/rodauth/features/reset_password.rb
212 def reset_password_email_recently_sent?
213   (email_last_sent = get_reset_password_email_last_sent) && (Time.now - email_last_sent < reset_password_skip_resend_email_within)
214 end
reset_password_key_insert_hash()
[show source]
    # File lib/rodauth/features/reset_password.rb
248 def reset_password_key_insert_hash
249   hash = {reset_password_id_column=>account_id, reset_password_key_column=>reset_password_key_value}
250   set_deadline_value(hash, reset_password_deadline_column, reset_password_deadline_interval)
251   hash
252 end
reset_password_request_for_unverified_account()
[show source]
    # File lib/rodauth/features/reset_password.rb
178 def reset_password_request_for_unverified_account
179   throw_error_reason(:unverified_account, unopen_account_error_status, login_param, unverified_account_message)
180 end
reset_single_session_key()
[show source]
   # File lib/rodauth/features/single_session.rb
22 def reset_single_session_key
23   if logged_in?
24     single_session_ds.update(single_session_key_column=>random_key)
25   end
26 end
response()
[show source]
    # File lib/rodauth/features/base.rb
186 def response
187   scope.response
188 end
retry_on_uniqueness_violation(&block)

This is used to avoid race conditions when using the pattern of inserting when an update affects no rows. In such cases, if a row is inserted between the update and the insert, the insert will fail with a uniqueness error, but retrying will work. It is possible for it to fail again, but only if the row is deleted before the update and readded before the insert, which is very unlikely to happen. In such cases, raising an exception is acceptable.

[show source]
    # File lib/rodauth/features/base.rb
915 def retry_on_uniqueness_violation(&block)
916   if raises_uniqueness_violation?(&block)
917     yield
918   end
919 end
return_json_response()
[show source]
    # File lib/rodauth/features/json.rb
208 def return_json_response
209   _return_json_response
210 end
return_response(body=nil)
[show source]
    # File lib/rodauth/features/base.rb
694 def return_response(body=nil)
695   response.write(body) if body
696   request.halt
697 end
route!()
[show source]
    # File lib/rodauth/features/base.rb
198 def route!
199   if meth = self.class.route_hash[request.remaining_path]
200     send(meth)
201   end
202 
203   nil
204 end
route_path(route, opts={})
[show source]
    # File lib/rodauth/features/base.rb
699 def route_path(route, opts={})
700   path  = "#{prefix}/#{route}"
701   path += "?#{Rack::Utils.build_nested_query(opts)}" unless opts.empty?
702   path
703 end
route_url(route, opts={})
[show source]
    # File lib/rodauth/features/base.rb
705 def route_url(route, opts={})
706   "#{base_url}#{route_path(route, opts)}"
707 end
save_account()
[show source]
    # File lib/rodauth/features/create_account.rb
100 def save_account
101   id = nil
102   raised = raises_uniqueness_violation?{id = db[accounts_table].insert(account)}
103 
104   if raised
105     set_login_requirement_error_message(:already_an_account_with_this_login, already_an_account_with_this_login_message)
106   end
107 
108   if id
109     account[account_id_column] ||= id
110   end
111 
112   id && !raised
113 end
send_email(email)
[show source]
   # File lib/rodauth/features/email_base.rb
36 def send_email(email)
37   email.deliver!
38 end
send_verify_login_change_email(login)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
126 def send_verify_login_change_email(login)
127   send_email(create_verify_login_change_email(login))
128 end
serialize_audit_log_metadata(metadata)
[show source]
   # File lib/rodauth/features/audit_logging.rb
55 def serialize_audit_log_metadata(metadata)
56   metadata.to_json unless metadata.nil?
57 end
session()
[show source]
    # File lib/rodauth/features/base.rb
190 def session
191   scope.session
192 end
session_inactivity_deadline_condition()
[show source]
    # File lib/rodauth/features/active_sessions.rb
213 def session_inactivity_deadline_condition
214   if deadline = session_inactivity_deadline
215     Sequel[active_sessions_last_use_column] < Sequel.date_sub(Sequel::CURRENT_TIMESTAMP, seconds: deadline)
216   end
217 end
session_jwt()
[show source]
   # File lib/rodauth/features/jwt.rb
76 def session_jwt
77   JWT.encode(jwt_session_hash, jwt_secret, jwt_algorithm)
78 end
session_lifetime_deadline_condition()
[show source]
    # File lib/rodauth/features/active_sessions.rb
219 def session_lifetime_deadline_condition
220   if deadline = session_lifetime_deadline
221     Sequel[active_sessions_created_at_column] < Sequel.date_sub(Sequel::CURRENT_TIMESTAMP, seconds: deadline)
222   end
223 end
session_value()
[show source]
    # File lib/rodauth/features/base.rb
305 def session_value
306   session[session_key]
307 end
set_deadline_value(hash, column, interval)

This is needed on MySQL, which doesn’t support non constant defaults other than CURRENT_TIMESTAMP.

[show source]
    # File lib/rodauth/features/base.rb
951 def set_deadline_value(hash, column, interval)
952   if set_deadline_values?
953     # :nocov:
954     hash[column] = Sequel.date_add(Sequel::CURRENT_TIMESTAMP, interval)
955     # :nocov:
956   end
957 end
set_deadline_values?()
[show source]
    # File lib/rodauth/features/base.rb
781 def set_deadline_values?
782   db.database_type == :mysql
783 end
set_email_auth_email_last_sent()
[show source]
    # File lib/rodauth/features/email_auth.rb
117 def set_email_auth_email_last_sent
118    email_auth_ds.update(email_auth_email_last_sent_column=>Sequel::CURRENT_TIMESTAMP) if email_auth_email_last_sent_column
119 end
set_error_flash(message)
[show source]
    # File lib/rodauth/features/base.rb
378 def set_error_flash(message)
379   flash.now[flash_error_key] = message
380 end
set_error_reason(reason)
[show source]
    # File lib/rodauth/features/base.rb
769 def set_error_reason(reason)
770 end
set_expired()
[show source]
   # File lib/rodauth/features/account_expiration.rb
49 def set_expired
50   update_activity(account_id, account_activity_expired_column)
51   after_account_expiration
52 end
set_field_error(field, error)
[show source]
    # File lib/rodauth/features/base.rb
206 def set_field_error(field, error)
207   (@field_errors ||= {})[field] = error
208 end
set_http_basic_auth_error_response()
[show source]
   # File lib/rodauth/features/http_basic_auth.rb
79 def set_http_basic_auth_error_response
80   response.status = 401
81   set_response_header("www-authenticate", "Basic realm=\"#{http_basic_auth_realm}\"")
82 end
set_jwt()
[show source]
    # File lib/rodauth/features/jwt.rb
152 def set_jwt
153   set_jwt_token(session_jwt)
154 end
set_jwt_refresh_token_hmac_session_key(token)
[show source]
    # File lib/rodauth/features/jwt_refresh.rb
203 def set_jwt_refresh_token_hmac_session_key(token)
204   if allow_refresh_with_expired_jwt_access_token?
205     key = _account_refresh_token_split(token).last
206     data = random_key
207     set_session_value(jwt_refresh_token_data_session_key, data)
208     set_session_value(jwt_refresh_token_hmac_session_key, compute_hmac(data + key))
209   end
210 end
set_jwt_token(token)
[show source]
   # File lib/rodauth/features/jwt.rb
88 def set_jwt_token(token)
89   set_response_header('authorization', token)
90 end
set_last_password_entry()
[show source]
   # File lib/rodauth/features/password_grace_period.rb
46 def set_last_password_entry
47   set_session_value(last_password_entry_session_key, Time.now.to_i)
48 end
set_login_requirement_error_message(reason, message)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
127 def set_login_requirement_error_message(reason, message)
128   set_error_reason(reason)
129   @login_requirement_message = message
130 end
set_new_account_password(password)
[show source]
   # File lib/rodauth/features/create_account.rb
92 def set_new_account_password(password)
93   account[account_password_hash_column] = password_hash(password)
94 end
set_notice_flash(message)
[show source]
    # File lib/rodauth/features/base.rb
386 def set_notice_flash(message)
387   flash[flash_notice_key] = message
388 end
set_notice_now_flash(message)
[show source]
    # File lib/rodauth/features/base.rb
390 def set_notice_now_flash(message)
391   flash.now[flash_notice_key] = message
392 end
set_password(password)
[show source]
   # File lib/rodauth/features/disallow_password_reuse.rb
21 def set_password(password)
22   hash = super
23   add_previous_password_hash(hash)
24   hash
25 end
set_password_requirement_error_message(reason, message)
[show source]
    # File lib/rodauth/features/login_password_requirements_base.rb
106 def set_password_requirement_error_message(reason, message)
107   set_error_reason(reason)
108   @password_requirement_message = message
109 end
set_redirect_error_flash(message)
[show source]
    # File lib/rodauth/features/base.rb
382 def set_redirect_error_flash(message)
383   flash[flash_error_key] = message
384 end
set_redirect_error_status(status)

Don’t set an error status when redirecting in an error case, as a redirect status is needed.

[show source]
    # File lib/rodauth/features/base.rb
743 def set_redirect_error_status(status)
744 end
set_reset_password_email_last_sent()
[show source]
    # File lib/rodauth/features/reset_password.rb
200 def set_reset_password_email_last_sent
201    password_reset_ds.update(reset_password_email_last_sent_column=>Sequel::CURRENT_TIMESTAMP) if reset_password_email_last_sent_column
202 end
set_response_error_reason_status(reason, status)
[show source]
    # File lib/rodauth/features/base.rb
750 def set_response_error_reason_status(reason, status)
751   set_error_reason(reason)
752   set_response_error_status(status)
753 end
set_response_error_status(status)
[show source]
    # File lib/rodauth/features/base.rb
746 def set_response_error_status(status)
747   response.status = status
748 end
set_response_header(key, value)
[show source]
    # File lib/rodauth/features/base.rb
602 def set_response_header(key, value)
603   response.headers[key] = value
604 end
set_session_value(key, value)
[show source]
    # File lib/rodauth/features/base.rb
978 def set_session_value(key, value)
979   session[key] = value
980 end
set_single_session_key(data)
[show source]
    # File lib/rodauth/features/single_session.rb
 99 def set_single_session_key(data)
100   data = compute_hmac(data) if hmac_secret
101   set_session_value(single_session_session_key, data)
102 end
set_title(title)
[show source]
    # File lib/rodauth/features/base.rb
372 def set_title(title)
373   if title_instance_variable
374     scope.instance_variable_set(title_instance_variable, title)
375   end
376 end
set_unlock_account_email_last_sent()
[show source]
    # File lib/rodauth/features/lockout.rb
239 def set_unlock_account_email_last_sent
240   account_lockouts_ds.update(account_lockouts_email_last_sent_column=>Sequel::CURRENT_TIMESTAMP) if account_lockouts_email_last_sent_column
241 end
set_verify_account_email_last_sent()
[show source]
    # File lib/rodauth/features/verify_account.rb
227 def set_verify_account_email_last_sent
228    verify_account_ds.update(verify_account_email_last_sent_column=>Sequel::CURRENT_TIMESTAMP) if verify_account_email_last_sent_column
229 end
setup_account_verification()
[show source]
    # File lib/rodauth/features/verify_account.rb
239 def setup_account_verification
240   generate_verify_account_key_value
241   create_verify_account_key
242   send_verify_account_email
243 end
show_lockout_page()
[show source]
    # File lib/rodauth/features/lockout.rb
287 def show_lockout_page
288   set_response_error_reason_status(:account_locked_out, lockout_error_status)
289   set_error_flash login_lockout_error_flash
290   return_response unlock_account_request_view
291 end
single_session_ds(id=session_value)
[show source]
    # File lib/rodauth/features/single_session.rb
104 def single_session_ds(id=session_value)
105   db[single_session_table].
106     where(single_session_id_column=>id)
107 end
skip_login_field_on_login?()
[show source]
    # File lib/rodauth/features/login.rb
126 def skip_login_field_on_login?
127   return false unless use_multi_phase_login?
128   valid_login_entered?
129 end
skip_password_field_on_login?()
[show source]
    # File lib/rodauth/features/login.rb
131 def skip_password_field_on_login?
132   return false unless use_multi_phase_login?
133   !valid_login_entered?
134 end
skip_status_checks?()
[show source]
   # File lib/rodauth/features/close_account.rb
85 def skip_status_checks?
86   false
87 end
sms()
[show source]
    # File lib/rodauth/features/sms_codes.rb
525 def sms
526   case @sms
527   when nil
528     (@sms = sms_ds.first || false) || nil
529   when false
530     nil
531   else
532     @sms
533   end
534 end
sms_auth_message(code)
[show source]
    # File lib/rodauth/features/sms_codes.rb
400 def sms_auth_message(code)
401   "SMS authentication code for #{domain} is #{code}"
402 end
sms_available?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
453 def sms_available?
454   sms_setup? && !sms_locked_out?
455 end
sms_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
432 def sms_code
433   sms[sms_code_column]
434 end
sms_code_issued_at()
[show source]
    # File lib/rodauth/features/sms_codes.rb
436 def sms_code_issued_at
437   convert_timestamp(sms[sms_issued_at_column])
438 end
sms_code_match?(code)
[show source]
    # File lib/rodauth/features/sms_codes.rb
349 def sms_code_match?(code)
350   return false unless sms_current_auth?
351   timing_safe_eql?(code, sms_code)
352 end
sms_codes_primary?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
501 def sms_codes_primary?
502   (features & [:otp, :webauthn]).empty?
503 end
sms_confirm()
[show source]
    # File lib/rodauth/features/recovery_codes.rb
148 def sms_confirm
149   super if defined?(super)
150   auto_add_missing_recovery_codes
151 end
sms_confirm_failure()
[show source]
    # File lib/rodauth/features/sms_codes.rb
363 def sms_confirm_failure
364   sms_ds.delete
365 end
sms_confirm_message(code)
[show source]
    # File lib/rodauth/features/sms_codes.rb
404 def sms_confirm_message(code)
405   "SMS confirmation code for #{domain} is #{code}"
406 end
sms_confirmation_match?(code)
[show source]
    # File lib/rodauth/features/sms_codes.rb
354 def sms_confirmation_match?(code)
355   sms_needs_confirmation? && sms_code_match?(code)
356 end
sms_current_auth?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
461 def sms_current_auth?
462   sms_code && sms_code_issued_at + sms_code_allowed_seconds > Time.now
463 end
sms_disable()
[show source]
    # File lib/rodauth/features/sms_codes.rb
358 def sms_disable
359   sms_ds.delete
360   @sms = false
361 end
sms_ds()
[show source]
    # File lib/rodauth/features/sms_codes.rb
536 def sms_ds
537   db[sms_codes_table].where(sms_id_column=>session_value)
538 end
sms_failures()
[show source]
    # File lib/rodauth/features/sms_codes.rb
440 def sms_failures
441   sms[sms_failures_column]
442 end
sms_locked_out?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
457 def sms_locked_out?
458   sms_failures >= sms_failure_limit
459 end
sms_needs_confirmation?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
449 def sms_needs_confirmation?
450   sms && sms_failures.nil?
451 end
sms_needs_confirmation_notice_flash()
[show source]
    # File lib/rodauth/features/sms_codes.rb
408 def sms_needs_confirmation_notice_flash
409   sms_needs_confirmation_error_flash
410 end
sms_new_auth_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
509 def sms_new_auth_code
510   SecureRandom.random_number(10**sms_auth_code_length).to_s.rjust(sms_auth_code_length, "0")
511 end
sms_new_confirm_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
513 def sms_new_confirm_code
514   SecureRandom.random_number(10**sms_confirm_code_length).to_s.rjust(sms_confirm_code_length, "0")
515 end
sms_normalize_phone(phone)
[show source]
    # File lib/rodauth/features/sms_codes.rb
505 def sms_normalize_phone(phone)
506   phone.to_s.gsub(/\D+/, '')
507 end
sms_phone()
[show source]
    # File lib/rodauth/features/sms_codes.rb
428 def sms_phone
429   sms[sms_phone_column]
430 end
sms_record_failure()
[show source]
    # File lib/rodauth/features/sms_codes.rb
423 def sms_record_failure
424   update_sms(sms_failures_column=>Sequel.expr(sms_failures_column)+1)
425   sms[sms_failures_column] = sms_ds.get(sms_failures_column)
426 end
sms_remove_expired_confirm_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
416 def sms_remove_expired_confirm_code
417   db[sms_codes_table].
418     where(sms_id_column=>session_value, sms_failures_column => nil).
419     where(Sequel[sms_issued_at_column] < Sequel.date_sub(Sequel::CURRENT_TIMESTAMP, seconds: sms_confirm_deadline)).
420     delete
421 end
sms_remove_failures()
[show source]
    # File lib/rodauth/features/sms_codes.rb
374 def sms_remove_failures
375   return if sms_needs_confirmation?
376   update_hash_ds(sms, sms_ds.exclude(sms_failures_column => nil), sms_failures_column => 0, sms_code_column => nil)
377 end
sms_send(phone, message)
[show source]
    # File lib/rodauth/features/sms_codes.rb
517 def sms_send(phone, message)
518   raise ConfigurationError, "sms_send needs to be defined in the Rodauth configuration for SMS sending to work"
519 end
sms_send_auth_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
384 def sms_send_auth_code
385   code = sms_new_auth_code
386   sms_set_code(code)
387   sms_send(sms_phone, sms_auth_message(code))
388 end
sms_send_confirm_code()
[show source]
    # File lib/rodauth/features/sms_codes.rb
390 def sms_send_confirm_code
391   code = sms_new_confirm_code
392   sms_set_code(code)
393   sms_send(sms_phone, sms_confirm_message(code))
394 end
sms_set_code(code)
[show source]
    # File lib/rodauth/features/sms_codes.rb
412 def sms_set_code(code)
413  update_sms(sms_code_column=>code, sms_issued_at_column=>Sequel::CURRENT_TIMESTAMP)
414 end
sms_setup(phone_number)
[show source]
    # File lib/rodauth/features/sms_codes.rb
367 def sms_setup(phone_number)
368   # Cannot handle uniqueness violation here, as the phone number given may not match the
369   # one in the table.
370   sms_ds.insert(sms_id_column=>session_value, sms_phone_column=>phone_number, sms_failures_column => nil)
371   @sms = nil
372 end
sms_setup?()
[show source]
    # File lib/rodauth/features/sms_codes.rb
444 def sms_setup?
445   return false unless sms
446   !sms_needs_confirmation?
447 end
sms_valid_phone?(phone)
[show source]
    # File lib/rodauth/features/sms_codes.rb
396 def sms_valid_phone?(phone)
397   phone.length >= sms_phone_min_length
398 end
split_token(token)
[show source]
    # File lib/rodauth/features/base.rb
667 def split_token(token)
668   token.split(token_separator, 2)
669 end
template_path(page)
[show source]
    # File lib/rodauth/features/base.rb
880 def template_path(page)
881   File.join(File.dirname(__FILE__), '../../../templates', "#{page}.str")
882 end
throw_basic_auth_error(*args)
[show source]
   # File lib/rodauth/features/http_basic_auth.rb
84 def throw_basic_auth_error(*args)
85   set_http_basic_auth_error_response
86   throw_error(*args) 
87 end
throw_error(field, error)
[show source]
    # File lib/rodauth/features/base.rb
759 def throw_error(field, error)
760   set_field_error(field, error)
761   throw_rodauth_error
762 end
throw_error_reason(reason, status, field, message)
[show source]
    # File lib/rodauth/features/base.rb
772 def throw_error_reason(reason, status, field, message)
773   set_error_reason(reason)
774   throw_error_status(status, field, message)
775 end
throw_error_status(status, field, error)
[show source]
    # File lib/rodauth/features/base.rb
764 def throw_error_status(status, field, error)
765   set_response_error_status(status)
766   throw_error(field, error)
767 end
throw_rodauth_error()
[show source]
    # File lib/rodauth/features/base.rb
755 def throw_rodauth_error
756   throw :rodauth_error
757 end
timing_safe_eql?(provided, actual)
[show source]
    # File lib/rodauth/features/base.rb
726 def timing_safe_eql?(provided, actual)
727   provided = provided.to_s
728   Rack::Utils.secure_compare(provided.ljust(actual.length), actual) && provided.length == actual.length
729 end
token_param_value(key)
[show source]
   # File lib/rodauth/features/email_base.rb
57 def token_param_value(key)
58   "#{account_id}#{token_separator}#{convert_email_token_key(key)}"
59 end
transaction(opts={}, &block)
[show source]
    # File lib/rodauth/features/base.rb
709 def transaction(opts={}, &block)
710   db.transaction(opts, &block)
711 end
translate(_key, default)
[show source]
    # File lib/rodauth/features/base.rb
273 def translate(_key, default)
274   # do not attempt to translate by default
275   default
276 end
two_factor_authenticate(type)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
243 def two_factor_authenticate(type)
244   two_factor_update_session(type)
245   two_factor_remove_auth_failures
246   after_two_factor_authentication
247   require_response(:_two_factor_auth_response)
248 end
two_factor_authenticated?()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
186 def two_factor_authenticated?
187   authenticated_by && authenticated_by.length >= 2
188 end
two_factor_authentication_setup?()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
190 def two_factor_authentication_setup?
191   possible_authentication_methods.length >= 2
192 end
two_factor_login_type_match?(type)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
200 def two_factor_login_type_match?(type)
201   authenticated_by && authenticated_by.include?(type)
202 end
two_factor_modifications_require_password?()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
124 def two_factor_modifications_require_password?
125   modifications_require_password?
126 end
two_factor_partially_authenticated?()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
182 def two_factor_partially_authenticated?
183   logged_in? && !two_factor_authenticated? && uses_two_factor_authentication?
184 end
two_factor_password_match?(password)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
174 def two_factor_password_match?(password)
175   if two_factor_modifications_require_password?
176     password_match?(password)
177   else
178     true
179   end
180 end
two_factor_remove()
[show source]
    # File lib/rodauth/features/otp.rb
226 def two_factor_remove
227   super
228   otp_remove
229 end
two_factor_remove_auth_failures()
[show source]
    # File lib/rodauth/features/otp.rb
231 def two_factor_remove_auth_failures
232   super
233   otp_remove_auth_failures
234 end
two_factor_remove_session(type)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
256 def two_factor_remove_session(type)
257   authenticated_by.delete(type)
258   remove_session_value(two_factor_setup_session_key)
259   if authenticated_by.empty?
260     clear_session
261   end
262 end
two_factor_update_session(auth_type)
[show source]
    # File lib/rodauth/features/two_factor_base.rb
264 def two_factor_update_session(auth_type)
265   authenticated_by << auth_type
266   set_session_value(two_factor_setup_session_key, true)
267 end
unique_constraint_violation_class()

Work around jdbc/sqlite issue where it only raises ConstraintViolation and not UniqueConstraintViolation.

[show source]
    # File lib/rodauth/features/base.rb
933 def unique_constraint_violation_class
934   if db.adapter_scheme == :jdbc && db.database_type == :sqlite
935     # :nocov:
936     Sequel::ConstraintViolation
937     # :nocov:
938   else
939     Sequel::UniqueConstraintViolation
940   end
941 end
unlock_account()
[show source]
    # File lib/rodauth/features/lockout.rb
162 def unlock_account
163   transaction do
164     remove_lockout_metadata
165   end
166 end
unlock_account_email_recently_sent?()
[show source]
    # File lib/rodauth/features/lockout.rb
243 def unlock_account_email_recently_sent?
244   (email_last_sent = get_unlock_account_email_last_sent) && (Time.now - email_last_sent < unlock_account_skip_resend_email_within)
245 end
unverified_grace_period_expired?()
[show source]
   # File lib/rodauth/features/verify_account_grace_period.rb
92 def unverified_grace_period_expired?
93   return false unless expires_at = session[unverified_account_session_key]
94   expires_at.is_a?(Integer) && Time.now.to_i > expires_at
95 end
update_account(values, ds=account_ds)
[show source]
    # File lib/rodauth/features/base.rb
996 def update_account(values, ds=account_ds)
997   update_hash_ds(account, ds, values)
998 end
update_activity(account_id, *columns)
[show source]
    # File lib/rodauth/features/account_expiration.rb
113 def update_activity(account_id, *columns)
114   ds = account_activity_ds(account_id)
115   hash = {}
116   columns.each do |c|
117     hash[c] = Sequel::CURRENT_TIMESTAMP
118   end
119   if ds.update(hash) == 0
120     hash[account_activity_id_column] = account_id
121     hash[account_activity_last_activity_column] ||= Sequel::CURRENT_TIMESTAMP
122     hash[account_activity_last_login_column] ||= Sequel::CURRENT_TIMESTAMP
123     # It is safe to ignore uniqueness violations here, as a concurrent insert would also use current timestamps.
124     ignore_uniqueness_violation{ds.insert(hash)}
125   end
126 end
update_current_session?()
[show source]
    # File lib/rodauth/features/active_sessions.rb
232 def update_current_session?
233   !!session_inactivity_deadline
234 end
update_hash_ds(hash, ds, values)
[show source]
    # File lib/rodauth/features/base.rb
986 def update_hash_ds(hash, ds, values)
987   num = ds.update(values)
988   if num == 1
989     values.each do |k, v|
990       hash[k] = Sequel::CURRENT_TIMESTAMP == v ? Time.now : v
991     end
992   end
993   num
994 end
update_last_activity()
[show source]
   # File lib/rodauth/features/account_expiration.rb
43 def update_last_activity
44   if session_value
45     update_activity(session_value, account_activity_last_activity_column)
46   end
47 end
update_last_login()
[show source]
   # File lib/rodauth/features/account_expiration.rb
39 def update_last_login
40   update_activity(account_id, account_activity_last_login_column, account_activity_last_activity_column)
41 end
update_login(login)
[show source]
   # File lib/rodauth/features/change_login.rb
79 def update_login(login)
80   _update_login(login)
81 end
update_password_changed_at()
[show source]
   # File lib/rodauth/features/password_expiration.rb
52 def update_password_changed_at
53   ds = password_expiration_ds
54   if ds.update(password_expiration_changed_at_column=>Sequel::CURRENT_TIMESTAMP) == 0
55     # Ignoring the violation is safe here, since a concurrent insert would also set it to the
56     # current timestamp.
57     ignore_uniqueness_violation{ds.insert(password_expiration_id_column=>account_id)}
58   end
59 end
update_password_hash?()
[show source]
   # File lib/rodauth/features/update_password_hash.rb
20 def update_password_hash?
21   password_hash_cost != @current_password_hash_cost || @update_password_hash
22 end
update_session()
[show source]
   # File lib/rodauth/features/account_expiration.rb
72 def update_session
73   check_account_expiration
74   super
75 end
update_single_session_key()
[show source]
   # File lib/rodauth/features/single_session.rb
66 def update_single_session_key
67   key = random_key
68   set_single_session_key(key)
69   if single_session_ds.update(single_session_key_column=>key) == 0
70     # Don't handle uniqueness violations here.  While we could get the stored key from the
71     # database, it could lead to two sessions sharing the same key, which this feature is
72     # designed to prevent.
73     single_session_ds.insert(single_session_id_column=>session_value, single_session_key_column=>key)
74   end
75 end
update_sms(values)
[show source]
    # File lib/rodauth/features/sms_codes.rb
521 def update_sms(values)
522   update_hash_ds(sms, sms_ds, values)
523 end
use_database_authentication_functions?()
[show source]
    # File lib/rodauth/features/base.rb
785 def use_database_authentication_functions?
786   case db.database_type
787   when :postgres, :mysql, :mssql
788     true
789   else
790     # :nocov:
791     false
792     # :nocov:
793   end
794 end
use_date_arithmetic?()
[show source]
    # File lib/rodauth/features/active_sessions.rb
241 def use_date_arithmetic?
242   true
243 end
use_json?()
[show source]
   # File lib/rodauth/features/json.rb
62 def use_json?
63   json_request? || only_json?
64 end
use_jwt?()
[show source]
   # File lib/rodauth/features/jwt.rb
92 def use_jwt?
93   use_json?
94 end
use_multi_phase_login?()
[show source]
    # File lib/rodauth/features/email_auth.rb
158 def use_multi_phase_login?
159   true
160 end
use_request_specific_csrf_tokens?()
[show source]
    # File lib/rodauth/features/base.rb
796 def use_request_specific_csrf_tokens?
797   scope.opts[:rodauth_route_csrf] && scope.use_request_specific_csrf_tokens?
798 end
use_scope_clear_session?()
[show source]
    # File lib/rodauth/features/base.rb
969 def use_scope_clear_session?
970   scope.respond_to?(:clear_session)
971 end
uses_instance_variables(*ivs)
[show source]
    # File lib/rodauth/features/base.rb
154 def uses_instance_variables(*ivs)
155   auth.define_singleton_method(:instance_variables_used) do
156     super() + ivs
157   end
158 end
uses_two_factor_authentication?()
[show source]
    # File lib/rodauth/features/two_factor_base.rb
194 def uses_two_factor_authentication?
195   return false unless logged_in?
196   set_session_value(two_factor_setup_session_key, two_factor_authentication_setup?) unless session.has_key?(two_factor_setup_session_key)
197   session[two_factor_setup_session_key]
198 end
valid_jwt?()
[show source]
    # File lib/rodauth/features/jwt.rb
100 def valid_jwt?
101   !!(jwt_token && jwt_payload)
102 end
valid_login_entered?()
[show source]
    # File lib/rodauth/features/login.rb
136 def valid_login_entered?
137   @valid_login_entered
138 end
valid_new_webauthn_credential?(webauthn_credential)
[show source]
    # File lib/rodauth/features/webauthn.rb
314 def valid_new_webauthn_credential?(webauthn_credential)
315   _override_webauthn_credential_response_verify(webauthn_credential)
316   (challenge = param_or_nil(webauthn_setup_challenge_param)) &&
317     (hmac = param_or_nil(webauthn_setup_challenge_hmac_param)) &&
318     (timing_safe_eql?(compute_hmac(challenge), hmac) || (hmac_secret_rotation? && timing_safe_eql?(compute_old_hmac(challenge), hmac))) &&
319     webauthn_credential.verify(challenge)
320 end
valid_webauthn_credential_auth?(webauthn_credential)
[show source]
    # File lib/rodauth/features/webauthn.rb
362 def valid_webauthn_credential_auth?(webauthn_credential)
363   ds = webauthn_keys_ds.where(webauthn_keys_webauthn_id_column => webauthn_credential.id)
364   pub_key, sign_count = ds.get([webauthn_keys_public_key_column, webauthn_keys_sign_count_column])
365 
366   _override_webauthn_credential_response_verify(webauthn_credential)
367   (challenge = param_or_nil(webauthn_auth_challenge_param)) &&
368     (hmac = param_or_nil(webauthn_auth_challenge_hmac_param)) &&
369     (timing_safe_eql?(compute_hmac(challenge), hmac) || (hmac_secret_rotation? && timing_safe_eql?(compute_old_hmac(challenge), hmac))) &&
370     webauthn_credential.verify(challenge, public_key: pub_key, sign_count: sign_count) &&
371     ds.update(
372       webauthn_keys_sign_count_column => Integer(webauthn_credential.sign_count),
373       webauthn_keys_last_use_column => Sequel::CURRENT_TIMESTAMP
374     ) == 1
375 end
verified_account?()
[show source]
   # File lib/rodauth/features/verify_account_grace_period.rb
17 def verified_account?
18   logged_in? && !session[unverified_account_session_key]
19 end
verify_account()
[show source]
    # File lib/rodauth/features/verify_account.rb
173 def verify_account
174   update_account(account_status_column=>account_open_status_value) == 1
175 end
verify_account_check_already_logged_in()
[show source]
    # File lib/rodauth/features/verify_account.rb
280 def verify_account_check_already_logged_in
281   check_already_logged_in
282 end
verify_account_ds(id=account_id)
[show source]
    # File lib/rodauth/features/verify_account.rb
305 def verify_account_ds(id=account_id)
306   db[verify_account_table].where(verify_account_id_column=>id)
307 end
verify_account_email_recently_sent?()
[show source]
    # File lib/rodauth/features/verify_account.rb
245 def verify_account_email_recently_sent?
246   account && (email_last_sent = get_verify_account_email_last_sent) && (Time.now - email_last_sent < verify_account_skip_resend_email_within)
247 end
verify_account_email_resend()
[show source]
    # File lib/rodauth/features/verify_account.rb
177 def verify_account_email_resend
178   if @verify_account_key_value = get_verify_account_key(account_id)
179     set_verify_account_email_last_sent
180     send_verify_account_email
181     true
182   end
183 end
verify_account_key_insert_hash()
[show source]
    # File lib/rodauth/features/verify_account.rb
301 def verify_account_key_insert_hash
302   {verify_account_id_column=>account_id, verify_account_key_column=>verify_account_key_value}
303 end
verify_account_set_password?()
[show source]
   # File lib/rodauth/features/verify_account_grace_period.rb
29 def verify_account_set_password?
30   false
31 end
verify_account_view()
[show source]
   # File lib/rodauth/features/webauthn_verify_account.rb
 9 def verify_account_view
10   webauthn_setup_view
11 end
verify_login_change()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
114 def verify_login_change
115   unless res = _update_login(verify_login_change_new_login)
116     remove_verify_login_change_key
117   end
118 
119   res
120 end
verify_login_change_ds(id=account_id)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
208 def verify_login_change_ds(id=account_id)
209   db[verify_login_change_table].where(verify_login_change_id_column=>id)
210 end
verify_login_change_email_body()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
204 def verify_login_change_email_body
205   render('verify-login-change-email')
206 end
verify_login_change_key_insert_hash(login)
[show source]
    # File lib/rodauth/features/verify_login_change.rb
194 def verify_login_change_key_insert_hash(login)
195   hash = {verify_login_change_id_column=>account_id, verify_login_change_key_column=>verify_login_change_key_value, verify_login_change_login_column=>login}
196   set_deadline_value(hash, verify_login_change_deadline_column, verify_login_change_deadline_interval)
197   hash
198 end
verify_login_change_old_login()
[show source]
    # File lib/rodauth/features/verify_login_change.rb
142 def verify_login_change_old_login
143   account_ds.get(login_column)
144 end
view(page, title)
[show source]
    # File lib/rodauth/features/base.rb
456 def view(page, title)
457   set_title(title)
458   _view(:view, page)
459 end
webauthn_account_id()
[show source]
    # File lib/rodauth/features/webauthn.rb
503 def webauthn_account_id
504   session_value
505 end
webauthn_allow()
[show source]
    # File lib/rodauth/features/webauthn.rb
340 def webauthn_allow
341   account_webauthn_ids
342 end
webauthn_auth_additional_form_tags()
[show source]
   # File lib/rodauth/features/webauthn_login.rb
45 def webauthn_auth_additional_form_tags
46   if @webauthn_login
47     super.to_s + login_hidden_field
48   else
49     super
50   end
51 end
webauthn_auth_credential_from_form_submission()
[show source]
    # File lib/rodauth/features/webauthn.rb
515 def webauthn_auth_credential_from_form_submission
516   begin
517     webauthn_credential = webauthn_form_submission_call(:from_get, webauthn_auth_data)
518 
519     unless valid_webauthn_credential_auth?(webauthn_credential)
520       throw_error_reason(:invalid_webauthn_auth_param, invalid_key_error_status, webauthn_auth_param, webauthn_invalid_auth_param_message)
521     end
522   rescue WebAuthn::SignCountVerificationError
523     handle_webauthn_sign_count_verification_error
524   rescue WebAuthn::Error, RuntimeError, NoMethodError
525     throw_error_reason(:invalid_webauthn_auth_param, invalid_field_error_status, webauthn_auth_param, webauthn_invalid_auth_param_message) 
526   end
527 
528   webauthn_credential
529 end
webauthn_auth_data()
[show source]
    # File lib/rodauth/features/webauthn.rb
531 def webauthn_auth_data
532   case auth_data = raw_param(webauthn_auth_param)
533   when String
534     begin
535       JSON.parse(auth_data)
536     rescue
537       throw_error_reason(:invalid_webauthn_auth_param, invalid_field_error_status, webauthn_auth_param, webauthn_invalid_auth_param_message) 
538     end
539   when Hash
540     auth_data
541   else
542     throw_error_reason(:invalid_webauthn_auth_param, invalid_field_error_status, webauthn_auth_param, webauthn_invalid_auth_param_message)
543   end
544 end
webauthn_auth_form_path()
[show source]
    # File lib/rodauth/features/webauthn.rb
247 def webauthn_auth_form_path
248   webauthn_auth_path
249 end
webauthn_authenticator_selection()
[show source]
    # File lib/rodauth/features/webauthn.rb
263 def webauthn_authenticator_selection
264   {'requireResidentKey' => false, 'userVerification' => webauthn_user_verification}
265 end
webauthn_create_relying_party_opts()
[show source]
    # File lib/rodauth/features/webauthn.rb
434 def webauthn_create_relying_party_opts
435   { :relying_party => webauthn_relying_party }
436 end
webauthn_credential_options_for_get()
[show source]
    # File lib/rodauth/features/webauthn.rb
322 def webauthn_credential_options_for_get
323   WebAuthn::Credential.options_for_get(
324     :allow => webauthn_allow,
325     :timeout => webauthn_auth_timeout,
326     :user_verification => webauthn_user_verification,
327     :extensions => webauthn_extensions,
328     **webauthn_get_relying_party_opts
329   )
330 end
webauthn_extensions()
[show source]
    # File lib/rodauth/features/webauthn.rb
267 def webauthn_extensions
268   {}
269 end
webauthn_form_submission_call(meth, arg)
[show source]
    # File lib/rodauth/features/webauthn.rb
439 def webauthn_form_submission_call(meth, arg)
440   WebAuthn::Credential.public_send(meth, arg, :relying_party => webauthn_relying_party)
441 end
webauthn_key_insert_hash(webauthn_credential)
[show source]
    # File lib/rodauth/features/webauthn.rb
494 def webauthn_key_insert_hash(webauthn_credential)
495   {
496     webauthn_keys_account_id_column => webauthn_account_id,
497     webauthn_keys_webauthn_id_column => webauthn_credential.id,
498     webauthn_keys_public_key_column => webauthn_credential.public_key,
499     webauthn_keys_sign_count_column => Integer(webauthn_credential.sign_count)
500   }
501 end
webauthn_keys_ds()
[show source]
    # File lib/rodauth/features/webauthn.rb
511 def webauthn_keys_ds
512   db[webauthn_keys_table].where(webauthn_keys_account_id_column => webauthn_account_id)
513 end
webauthn_login_options?()
[show source]
   # File lib/rodauth/features/webauthn_autofill.rb
60 def webauthn_login_options?
61   return true unless param_or_nil(login_param)
62   super
63 end
webauthn_login_verification_factor?(webauthn_credential)
[show source]
   # File lib/rodauth/features/webauthn_login.rb
72 def webauthn_login_verification_factor?(webauthn_credential)
73   webauthn_login_user_verification_additional_factor? &&
74     webauthn_credential.response.authenticator_data.user_verified? &&
75     uses_two_factor_authentication?
76 end
webauthn_origin()
[show source]
    # File lib/rodauth/features/webauthn.rb
336 def webauthn_origin
337   base_url
338 end
webauthn_relying_party()
[show source]
    # File lib/rodauth/features/webauthn.rb
414 def webauthn_relying_party
415   # No need to memoize, only called once per request
416   WebAuthn::RelyingParty.new(
417     allowed_origins: [webauthn_origin],
418     id: webauthn_rp_id,
419     name: webauthn_rp_name,
420   )
421 end
webauthn_remove_authenticated_session()
[show source]
    # File lib/rodauth/features/webauthn.rb
255 def webauthn_remove_authenticated_session
256   remove_session_value(authenticated_webauthn_id_session_key)
257 end
webauthn_rp_id()
[show source]
    # File lib/rodauth/features/webauthn.rb
344 def webauthn_rp_id
345   webauthn_origin.sub(/\Ahttps?:\/\//, '').sub(/:\d+\z/, '')
346 end
webauthn_rp_name()
[show source]
    # File lib/rodauth/features/webauthn.rb
348 def webauthn_rp_name
349   webauthn_rp_id
350 end
webauthn_setup?()
[show source]
    # File lib/rodauth/features/webauthn.rb
386 def webauthn_setup?
387   !webauthn_keys_ds.empty?
388 end
webauthn_setup_credential_from_form_submission()
[show source]
    # File lib/rodauth/features/webauthn.rb
546 def webauthn_setup_credential_from_form_submission
547   unless two_factor_password_match?(param(password_param))
548     throw_error_reason(:invalid_password, invalid_password_error_status, password_param, invalid_password_message)
549   end
550 
551   begin
552     webauthn_credential = webauthn_form_submission_call(:from_create, webauthn_setup_data)
553 
554     unless valid_new_webauthn_credential?(webauthn_credential)
555       throw_error_reason(:invalid_webauthn_setup_param, invalid_field_error_status, webauthn_setup_param, webauthn_invalid_setup_param_message) 
556     end
557   rescue WebAuthn::Error, RuntimeError, NoMethodError
558     throw_error_reason(:invalid_webauthn_setup_param, invalid_field_error_status, webauthn_setup_param, webauthn_invalid_setup_param_message) 
559   end
560 
561   webauthn_credential
562 end
webauthn_setup_data()
[show source]
    # File lib/rodauth/features/webauthn.rb
564 def webauthn_setup_data
565   case setup_data = raw_param(webauthn_setup_param)
566   when String
567     begin
568       JSON.parse(setup_data)
569     rescue
570       throw_error_reason(:invalid_webauthn_setup_param, invalid_field_error_status, webauthn_setup_param, webauthn_invalid_setup_param_message) 
571     end
572   when Hash
573     setup_data
574   else
575     throw_error_reason(:invalid_webauthn_setup_param, invalid_field_error_status, webauthn_setup_param, webauthn_invalid_setup_param_message)
576   end
577 end
webauthn_update_session(webauthn_id)
[show source]
    # File lib/rodauth/features/webauthn.rb
259 def webauthn_update_session(webauthn_id)
260   set_session_value(authenticated_webauthn_id_session_key, webauthn_id)
261 end
webauthn_user_ids_ds()
[show source]
    # File lib/rodauth/features/webauthn.rb
507 def webauthn_user_ids_ds
508   db[webauthn_user_ids_table].where(webauthn_user_ids_account_id_column => webauthn_account_id)
509 end
webauthn_user_name()
[show source]
    # File lib/rodauth/features/webauthn.rb
332 def webauthn_user_name
333   account![login_column]
334 end
webauthn_user_verification()
[show source]
   # File lib/rodauth/features/webauthn_autofill.rb
25 def webauthn_user_verification
26   'preferred'
27 end