sequel 5.20.0 → 5.49.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (511) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG +398 -1922
  3. data/MIT-LICENSE +1 -1
  4. data/README.rdoc +7 -7
  5. data/doc/advanced_associations.rdoc +4 -4
  6. data/doc/association_basics.rdoc +80 -16
  7. data/doc/cheat_sheet.rdoc +6 -5
  8. data/doc/code_order.rdoc +10 -12
  9. data/doc/dataset_filtering.rdoc +17 -2
  10. data/doc/fork_safety.rdoc +84 -0
  11. data/doc/migration.rdoc +11 -5
  12. data/doc/model_dataset_method_design.rdoc +1 -1
  13. data/doc/model_plugins.rdoc +1 -1
  14. data/doc/opening_databases.rdoc +10 -2
  15. data/doc/postgresql.rdoc +82 -3
  16. data/doc/querying.rdoc +4 -4
  17. data/doc/release_notes/5.21.0.txt +87 -0
  18. data/doc/release_notes/5.22.0.txt +48 -0
  19. data/doc/release_notes/5.23.0.txt +56 -0
  20. data/doc/release_notes/5.24.0.txt +56 -0
  21. data/doc/release_notes/5.25.0.txt +32 -0
  22. data/doc/release_notes/5.26.0.txt +35 -0
  23. data/doc/release_notes/5.27.0.txt +21 -0
  24. data/doc/release_notes/5.28.0.txt +16 -0
  25. data/doc/release_notes/5.29.0.txt +22 -0
  26. data/doc/release_notes/5.30.0.txt +20 -0
  27. data/doc/release_notes/5.31.0.txt +148 -0
  28. data/doc/release_notes/5.32.0.txt +46 -0
  29. data/doc/release_notes/5.33.0.txt +24 -0
  30. data/doc/release_notes/5.34.0.txt +40 -0
  31. data/doc/release_notes/5.35.0.txt +56 -0
  32. data/doc/release_notes/5.36.0.txt +60 -0
  33. data/doc/release_notes/5.37.0.txt +30 -0
  34. data/doc/release_notes/5.38.0.txt +28 -0
  35. data/doc/release_notes/5.39.0.txt +19 -0
  36. data/doc/release_notes/5.40.0.txt +40 -0
  37. data/doc/release_notes/5.41.0.txt +25 -0
  38. data/doc/release_notes/5.42.0.txt +136 -0
  39. data/doc/release_notes/5.43.0.txt +98 -0
  40. data/doc/release_notes/5.44.0.txt +32 -0
  41. data/doc/release_notes/5.45.0.txt +34 -0
  42. data/doc/release_notes/5.46.0.txt +87 -0
  43. data/doc/release_notes/5.47.0.txt +59 -0
  44. data/doc/release_notes/5.48.0.txt +14 -0
  45. data/doc/release_notes/5.49.0.txt +59 -0
  46. data/doc/sharding.rdoc +2 -0
  47. data/doc/sql.rdoc +13 -1
  48. data/doc/testing.rdoc +20 -7
  49. data/doc/transactions.rdoc +0 -8
  50. data/doc/validations.rdoc +1 -1
  51. data/doc/virtual_rows.rdoc +1 -1
  52. data/lib/sequel/adapters/ado/access.rb +1 -1
  53. data/lib/sequel/adapters/ado.rb +43 -35
  54. data/lib/sequel/adapters/ibmdb.rb +2 -2
  55. data/lib/sequel/adapters/jdbc/mysql.rb +6 -6
  56. data/lib/sequel/adapters/jdbc/postgresql.rb +11 -17
  57. data/lib/sequel/adapters/jdbc/sqlite.rb +29 -0
  58. data/lib/sequel/adapters/jdbc.rb +24 -6
  59. data/lib/sequel/adapters/mysql.rb +1 -1
  60. data/lib/sequel/adapters/mysql2.rb +2 -3
  61. data/lib/sequel/adapters/odbc.rb +8 -6
  62. data/lib/sequel/adapters/oracle.rb +5 -4
  63. data/lib/sequel/adapters/postgres.rb +15 -9
  64. data/lib/sequel/adapters/shared/access.rb +6 -6
  65. data/lib/sequel/adapters/shared/mssql.rb +66 -21
  66. data/lib/sequel/adapters/shared/mysql.rb +27 -10
  67. data/lib/sequel/adapters/shared/oracle.rb +29 -23
  68. data/lib/sequel/adapters/shared/postgres.rb +271 -32
  69. data/lib/sequel/adapters/shared/sqlanywhere.rb +9 -9
  70. data/lib/sequel/adapters/shared/sqlite.rb +161 -19
  71. data/lib/sequel/adapters/sqlanywhere.rb +1 -1
  72. data/lib/sequel/adapters/sqlite.rb +1 -1
  73. data/lib/sequel/adapters/tinytds.rb +15 -2
  74. data/lib/sequel/adapters/utils/mysql_mysql2.rb +4 -1
  75. data/lib/sequel/ast_transformer.rb +6 -0
  76. data/lib/sequel/connection_pool/sharded_single.rb +4 -1
  77. data/lib/sequel/connection_pool/sharded_threaded.rb +12 -12
  78. data/lib/sequel/connection_pool/single.rb +1 -1
  79. data/lib/sequel/connection_pool/threaded.rb +2 -2
  80. data/lib/sequel/core.rb +333 -319
  81. data/lib/sequel/database/connecting.rb +3 -4
  82. data/lib/sequel/database/logging.rb +7 -1
  83. data/lib/sequel/database/misc.rb +31 -12
  84. data/lib/sequel/database/query.rb +3 -1
  85. data/lib/sequel/database/schema_generator.rb +53 -51
  86. data/lib/sequel/database/schema_methods.rb +38 -23
  87. data/lib/sequel/database/transactions.rb +17 -18
  88. data/lib/sequel/dataset/actions.rb +14 -9
  89. data/lib/sequel/dataset/features.rb +16 -0
  90. data/lib/sequel/dataset/misc.rb +2 -2
  91. data/lib/sequel/dataset/placeholder_literalizer.rb +3 -7
  92. data/lib/sequel/dataset/prepared_statements.rb +2 -0
  93. data/lib/sequel/dataset/query.rb +26 -9
  94. data/lib/sequel/dataset/sql.rb +76 -25
  95. data/lib/sequel/dataset.rb +4 -2
  96. data/lib/sequel/deprecated.rb +3 -1
  97. data/lib/sequel/exceptions.rb +2 -0
  98. data/lib/sequel/extensions/_pretty_table.rb +1 -2
  99. data/lib/sequel/extensions/any_not_empty.rb +45 -0
  100. data/lib/sequel/extensions/async_thread_pool.rb +438 -0
  101. data/lib/sequel/extensions/blank.rb +8 -0
  102. data/lib/sequel/extensions/columns_introspection.rb +1 -2
  103. data/lib/sequel/extensions/connection_expiration.rb +2 -2
  104. data/lib/sequel/extensions/connection_validator.rb +2 -2
  105. data/lib/sequel/extensions/core_refinements.rb +2 -0
  106. data/lib/sequel/extensions/date_arithmetic.rb +36 -24
  107. data/lib/sequel/extensions/duplicate_columns_handler.rb +3 -1
  108. data/lib/sequel/extensions/eval_inspect.rb +2 -0
  109. data/lib/sequel/extensions/exclude_or_null.rb +68 -0
  110. data/lib/sequel/extensions/fiber_concurrency.rb +24 -0
  111. data/lib/sequel/extensions/index_caching.rb +9 -7
  112. data/lib/sequel/extensions/inflector.rb +9 -1
  113. data/lib/sequel/extensions/integer64.rb +2 -0
  114. data/lib/sequel/extensions/migration.rb +11 -3
  115. data/lib/sequel/extensions/named_timezones.rb +56 -8
  116. data/lib/sequel/extensions/pagination.rb +1 -1
  117. data/lib/sequel/extensions/pg_array.rb +5 -0
  118. data/lib/sequel/extensions/pg_array_ops.rb +14 -6
  119. data/lib/sequel/extensions/pg_enum.rb +11 -3
  120. data/lib/sequel/extensions/pg_extended_date_support.rb +2 -2
  121. data/lib/sequel/extensions/pg_hstore.rb +6 -0
  122. data/lib/sequel/extensions/pg_hstore_ops.rb +54 -2
  123. data/lib/sequel/extensions/pg_inet.rb +15 -5
  124. data/lib/sequel/extensions/pg_interval.rb +36 -8
  125. data/lib/sequel/extensions/pg_json.rb +387 -123
  126. data/lib/sequel/extensions/pg_json_ops.rb +238 -0
  127. data/lib/sequel/extensions/pg_loose_count.rb +3 -1
  128. data/lib/sequel/extensions/pg_range.rb +17 -9
  129. data/lib/sequel/extensions/pg_range_ops.rb +2 -0
  130. data/lib/sequel/extensions/pg_row.rb +4 -2
  131. data/lib/sequel/extensions/pg_row_ops.rb +24 -0
  132. data/lib/sequel/extensions/pg_timestamptz.rb +2 -0
  133. data/lib/sequel/extensions/query.rb +3 -0
  134. data/lib/sequel/extensions/run_transaction_hooks.rb +72 -0
  135. data/lib/sequel/extensions/s.rb +2 -0
  136. data/lib/sequel/extensions/schema_dumper.rb +24 -7
  137. data/lib/sequel/extensions/server_block.rb +18 -7
  138. data/lib/sequel/extensions/sql_comments.rb +2 -2
  139. data/lib/sequel/extensions/string_agg.rb +1 -1
  140. data/lib/sequel/extensions/symbol_aref_refinement.rb +2 -0
  141. data/lib/sequel/extensions/symbol_as_refinement.rb +2 -0
  142. data/lib/sequel/extensions/to_dot.rb +9 -3
  143. data/lib/sequel/model/associations.rb +356 -117
  144. data/lib/sequel/model/base.rb +107 -68
  145. data/lib/sequel/model/errors.rb +10 -1
  146. data/lib/sequel/model/inflections.rb +1 -1
  147. data/lib/sequel/model/plugins.rb +9 -3
  148. data/lib/sequel/model.rb +3 -1
  149. data/lib/sequel/plugins/association_lazy_eager_option.rb +66 -0
  150. data/lib/sequel/plugins/association_multi_add_remove.rb +85 -0
  151. data/lib/sequel/plugins/association_pks.rb +60 -18
  152. data/lib/sequel/plugins/association_proxies.rb +8 -2
  153. data/lib/sequel/plugins/async_thread_pool.rb +39 -0
  154. data/lib/sequel/plugins/auto_validations.rb +39 -5
  155. data/lib/sequel/plugins/auto_validations_constraint_validations_presence_message.rb +68 -0
  156. data/lib/sequel/plugins/blacklist_security.rb +1 -2
  157. data/lib/sequel/plugins/boolean_subsets.rb +4 -1
  158. data/lib/sequel/plugins/caching.rb +3 -0
  159. data/lib/sequel/plugins/class_table_inheritance.rb +33 -28
  160. data/lib/sequel/plugins/column_encryption.rb +728 -0
  161. data/lib/sequel/plugins/composition.rb +7 -2
  162. data/lib/sequel/plugins/concurrent_eager_loading.rb +174 -0
  163. data/lib/sequel/plugins/constraint_validations.rb +2 -1
  164. data/lib/sequel/plugins/csv_serializer.rb +28 -9
  165. data/lib/sequel/plugins/dataset_associations.rb +4 -1
  166. data/lib/sequel/plugins/dirty.rb +60 -22
  167. data/lib/sequel/plugins/empty_failure_backtraces.rb +38 -0
  168. data/lib/sequel/plugins/forbid_lazy_load.rb +216 -0
  169. data/lib/sequel/plugins/insert_conflict.rb +72 -0
  170. data/lib/sequel/plugins/instance_specific_default.rb +113 -0
  171. data/lib/sequel/plugins/json_serializer.rb +57 -35
  172. data/lib/sequel/plugins/lazy_attributes.rb +1 -1
  173. data/lib/sequel/plugins/many_through_many.rb +108 -9
  174. data/lib/sequel/plugins/nested_attributes.rb +15 -3
  175. data/lib/sequel/plugins/pg_array_associations.rb +58 -41
  176. data/lib/sequel/plugins/pg_auto_constraint_validations.rb +91 -30
  177. data/lib/sequel/plugins/prepared_statements.rb +15 -12
  178. data/lib/sequel/plugins/prepared_statements_safe.rb +1 -3
  179. data/lib/sequel/plugins/rcte_tree.rb +43 -35
  180. data/lib/sequel/plugins/serialization.rb +8 -3
  181. data/lib/sequel/plugins/serialization_modification_detection.rb +1 -1
  182. data/lib/sequel/plugins/sharding.rb +11 -5
  183. data/lib/sequel/plugins/single_table_inheritance.rb +22 -15
  184. data/lib/sequel/plugins/skip_saving_columns.rb +108 -0
  185. data/lib/sequel/plugins/static_cache.rb +9 -4
  186. data/lib/sequel/plugins/static_cache_cache.rb +53 -0
  187. data/lib/sequel/plugins/string_stripper.rb +1 -1
  188. data/lib/sequel/plugins/subclasses.rb +2 -0
  189. data/lib/sequel/plugins/throw_failures.rb +1 -1
  190. data/lib/sequel/plugins/timestamps.rb +1 -1
  191. data/lib/sequel/plugins/tree.rb +9 -4
  192. data/lib/sequel/plugins/typecast_on_load.rb +3 -2
  193. data/lib/sequel/plugins/unused_associations.rb +521 -0
  194. data/lib/sequel/plugins/update_or_create.rb +1 -1
  195. data/lib/sequel/plugins/validation_class_methods.rb +5 -1
  196. data/lib/sequel/plugins/validation_helpers.rb +18 -11
  197. data/lib/sequel/plugins/xml_serializer.rb +1 -1
  198. data/lib/sequel/sql.rb +20 -5
  199. data/lib/sequel/timezones.rb +63 -17
  200. data/lib/sequel/version.rb +1 -1
  201. metadata +113 -381
  202. data/Rakefile +0 -151
  203. data/doc/release_notes/4.0.0.txt +0 -262
  204. data/doc/release_notes/4.1.0.txt +0 -85
  205. data/doc/release_notes/4.10.0.txt +0 -226
  206. data/doc/release_notes/4.11.0.txt +0 -147
  207. data/doc/release_notes/4.12.0.txt +0 -105
  208. data/doc/release_notes/4.13.0.txt +0 -169
  209. data/doc/release_notes/4.14.0.txt +0 -68
  210. data/doc/release_notes/4.15.0.txt +0 -56
  211. data/doc/release_notes/4.16.0.txt +0 -36
  212. data/doc/release_notes/4.17.0.txt +0 -38
  213. data/doc/release_notes/4.18.0.txt +0 -36
  214. data/doc/release_notes/4.19.0.txt +0 -45
  215. data/doc/release_notes/4.2.0.txt +0 -129
  216. data/doc/release_notes/4.20.0.txt +0 -79
  217. data/doc/release_notes/4.21.0.txt +0 -94
  218. data/doc/release_notes/4.22.0.txt +0 -72
  219. data/doc/release_notes/4.23.0.txt +0 -65
  220. data/doc/release_notes/4.24.0.txt +0 -99
  221. data/doc/release_notes/4.25.0.txt +0 -181
  222. data/doc/release_notes/4.26.0.txt +0 -44
  223. data/doc/release_notes/4.27.0.txt +0 -78
  224. data/doc/release_notes/4.28.0.txt +0 -57
  225. data/doc/release_notes/4.29.0.txt +0 -41
  226. data/doc/release_notes/4.3.0.txt +0 -40
  227. data/doc/release_notes/4.30.0.txt +0 -37
  228. data/doc/release_notes/4.31.0.txt +0 -57
  229. data/doc/release_notes/4.32.0.txt +0 -132
  230. data/doc/release_notes/4.33.0.txt +0 -88
  231. data/doc/release_notes/4.34.0.txt +0 -86
  232. data/doc/release_notes/4.35.0.txt +0 -130
  233. data/doc/release_notes/4.36.0.txt +0 -116
  234. data/doc/release_notes/4.37.0.txt +0 -50
  235. data/doc/release_notes/4.38.0.txt +0 -67
  236. data/doc/release_notes/4.39.0.txt +0 -127
  237. data/doc/release_notes/4.4.0.txt +0 -92
  238. data/doc/release_notes/4.40.0.txt +0 -179
  239. data/doc/release_notes/4.41.0.txt +0 -77
  240. data/doc/release_notes/4.42.0.txt +0 -221
  241. data/doc/release_notes/4.43.0.txt +0 -87
  242. data/doc/release_notes/4.44.0.txt +0 -125
  243. data/doc/release_notes/4.45.0.txt +0 -370
  244. data/doc/release_notes/4.46.0.txt +0 -404
  245. data/doc/release_notes/4.47.0.txt +0 -56
  246. data/doc/release_notes/4.48.0.txt +0 -293
  247. data/doc/release_notes/4.49.0.txt +0 -222
  248. data/doc/release_notes/4.5.0.txt +0 -34
  249. data/doc/release_notes/4.6.0.txt +0 -30
  250. data/doc/release_notes/4.7.0.txt +0 -103
  251. data/doc/release_notes/4.8.0.txt +0 -175
  252. data/doc/release_notes/4.9.0.txt +0 -190
  253. data/spec/adapter_spec.rb +0 -4
  254. data/spec/adapters/db2_spec.rb +0 -170
  255. data/spec/adapters/mssql_spec.rb +0 -804
  256. data/spec/adapters/mysql_spec.rb +0 -1065
  257. data/spec/adapters/oracle_spec.rb +0 -371
  258. data/spec/adapters/postgres_spec.rb +0 -4125
  259. data/spec/adapters/spec_helper.rb +0 -44
  260. data/spec/adapters/sqlanywhere_spec.rb +0 -97
  261. data/spec/adapters/sqlite_spec.rb +0 -652
  262. data/spec/bin_spec.rb +0 -278
  263. data/spec/core/connection_pool_spec.rb +0 -1250
  264. data/spec/core/database_spec.rb +0 -2865
  265. data/spec/core/dataset_spec.rb +0 -5515
  266. data/spec/core/deprecated_spec.rb +0 -70
  267. data/spec/core/expression_filters_spec.rb +0 -1455
  268. data/spec/core/mock_adapter_spec.rb +0 -722
  269. data/spec/core/object_graph_spec.rb +0 -336
  270. data/spec/core/placeholder_literalizer_spec.rb +0 -166
  271. data/spec/core/schema_generator_spec.rb +0 -214
  272. data/spec/core/schema_spec.rb +0 -1826
  273. data/spec/core/spec_helper.rb +0 -24
  274. data/spec/core/version_spec.rb +0 -14
  275. data/spec/core_extensions_spec.rb +0 -763
  276. data/spec/core_model_spec.rb +0 -2
  277. data/spec/core_spec.rb +0 -1
  278. data/spec/deprecation_helper.rb +0 -30
  279. data/spec/extensions/accessed_columns_spec.rb +0 -51
  280. data/spec/extensions/active_model_spec.rb +0 -99
  281. data/spec/extensions/after_initialize_spec.rb +0 -28
  282. data/spec/extensions/arbitrary_servers_spec.rb +0 -109
  283. data/spec/extensions/association_dependencies_spec.rb +0 -125
  284. data/spec/extensions/association_pks_spec.rb +0 -423
  285. data/spec/extensions/association_proxies_spec.rb +0 -100
  286. data/spec/extensions/auto_literal_strings_spec.rb +0 -205
  287. data/spec/extensions/auto_validations_spec.rb +0 -229
  288. data/spec/extensions/blacklist_security_spec.rb +0 -95
  289. data/spec/extensions/blank_spec.rb +0 -69
  290. data/spec/extensions/boolean_readers_spec.rb +0 -93
  291. data/spec/extensions/boolean_subsets_spec.rb +0 -47
  292. data/spec/extensions/caching_spec.rb +0 -273
  293. data/spec/extensions/caller_logging_spec.rb +0 -52
  294. data/spec/extensions/class_table_inheritance_spec.rb +0 -750
  295. data/spec/extensions/column_conflicts_spec.rb +0 -75
  296. data/spec/extensions/column_select_spec.rb +0 -129
  297. data/spec/extensions/columns_introspection_spec.rb +0 -90
  298. data/spec/extensions/columns_updated_spec.rb +0 -35
  299. data/spec/extensions/composition_spec.rb +0 -248
  300. data/spec/extensions/connection_expiration_spec.rb +0 -151
  301. data/spec/extensions/connection_validator_spec.rb +0 -144
  302. data/spec/extensions/constant_sql_override_spec.rb +0 -24
  303. data/spec/extensions/constraint_validations_plugin_spec.rb +0 -300
  304. data/spec/extensions/constraint_validations_spec.rb +0 -439
  305. data/spec/extensions/core_refinements_spec.rb +0 -528
  306. data/spec/extensions/csv_serializer_spec.rb +0 -183
  307. data/spec/extensions/current_datetime_timestamp_spec.rb +0 -27
  308. data/spec/extensions/dataset_associations_spec.rb +0 -365
  309. data/spec/extensions/dataset_source_alias_spec.rb +0 -51
  310. data/spec/extensions/date_arithmetic_spec.rb +0 -181
  311. data/spec/extensions/datetime_parse_to_time_spec.rb +0 -169
  312. data/spec/extensions/def_dataset_method_spec.rb +0 -100
  313. data/spec/extensions/defaults_setter_spec.rb +0 -150
  314. data/spec/extensions/delay_add_association_spec.rb +0 -73
  315. data/spec/extensions/dirty_spec.rb +0 -189
  316. data/spec/extensions/duplicate_columns_handler_spec.rb +0 -104
  317. data/spec/extensions/eager_each_spec.rb +0 -62
  318. data/spec/extensions/eager_graph_eager_spec.rb +0 -100
  319. data/spec/extensions/empty_array_consider_nulls_spec.rb +0 -24
  320. data/spec/extensions/error_splitter_spec.rb +0 -18
  321. data/spec/extensions/error_sql_spec.rb +0 -20
  322. data/spec/extensions/escaped_like_spec.rb +0 -40
  323. data/spec/extensions/eval_inspect_spec.rb +0 -81
  324. data/spec/extensions/finder_spec.rb +0 -260
  325. data/spec/extensions/force_encoding_spec.rb +0 -126
  326. data/spec/extensions/freeze_datasets_spec.rb +0 -31
  327. data/spec/extensions/graph_each_spec.rb +0 -113
  328. data/spec/extensions/hook_class_methods_spec.rb +0 -402
  329. data/spec/extensions/identifier_mangling_spec.rb +0 -201
  330. data/spec/extensions/implicit_subquery_spec.rb +0 -58
  331. data/spec/extensions/index_caching_spec.rb +0 -66
  332. data/spec/extensions/inflector_spec.rb +0 -183
  333. data/spec/extensions/input_transformer_spec.rb +0 -69
  334. data/spec/extensions/insert_returning_select_spec.rb +0 -72
  335. data/spec/extensions/instance_filters_spec.rb +0 -79
  336. data/spec/extensions/instance_hooks_spec.rb +0 -246
  337. data/spec/extensions/integer64_spec.rb +0 -22
  338. data/spec/extensions/inverted_subsets_spec.rb +0 -33
  339. data/spec/extensions/json_serializer_spec.rb +0 -336
  340. data/spec/extensions/lazy_attributes_spec.rb +0 -183
  341. data/spec/extensions/list_spec.rb +0 -291
  342. data/spec/extensions/looser_typecasting_spec.rb +0 -43
  343. data/spec/extensions/many_through_many_spec.rb +0 -2177
  344. data/spec/extensions/migration_spec.rb +0 -864
  345. data/spec/extensions/modification_detection_spec.rb +0 -93
  346. data/spec/extensions/mssql_optimistic_locking_spec.rb +0 -92
  347. data/spec/extensions/named_timezones_spec.rb +0 -111
  348. data/spec/extensions/nested_attributes_spec.rb +0 -767
  349. data/spec/extensions/null_dataset_spec.rb +0 -85
  350. data/spec/extensions/optimistic_locking_spec.rb +0 -127
  351. data/spec/extensions/pagination_spec.rb +0 -116
  352. data/spec/extensions/pg_array_associations_spec.rb +0 -802
  353. data/spec/extensions/pg_array_ops_spec.rb +0 -144
  354. data/spec/extensions/pg_array_spec.rb +0 -398
  355. data/spec/extensions/pg_auto_constraint_validations_spec.rb +0 -172
  356. data/spec/extensions/pg_enum_spec.rb +0 -118
  357. data/spec/extensions/pg_extended_date_support_spec.rb +0 -126
  358. data/spec/extensions/pg_hstore_ops_spec.rb +0 -238
  359. data/spec/extensions/pg_hstore_spec.rb +0 -219
  360. data/spec/extensions/pg_inet_ops_spec.rb +0 -102
  361. data/spec/extensions/pg_inet_spec.rb +0 -72
  362. data/spec/extensions/pg_interval_spec.rb +0 -103
  363. data/spec/extensions/pg_json_ops_spec.rb +0 -289
  364. data/spec/extensions/pg_json_spec.rb +0 -262
  365. data/spec/extensions/pg_loose_count_spec.rb +0 -23
  366. data/spec/extensions/pg_range_ops_spec.rb +0 -60
  367. data/spec/extensions/pg_range_spec.rb +0 -519
  368. data/spec/extensions/pg_row_ops_spec.rb +0 -61
  369. data/spec/extensions/pg_row_plugin_spec.rb +0 -60
  370. data/spec/extensions/pg_row_spec.rb +0 -363
  371. data/spec/extensions/pg_static_cache_updater_spec.rb +0 -93
  372. data/spec/extensions/pg_timestamptz_spec.rb +0 -17
  373. data/spec/extensions/prepared_statements_safe_spec.rb +0 -66
  374. data/spec/extensions/prepared_statements_spec.rb +0 -177
  375. data/spec/extensions/pretty_table_spec.rb +0 -123
  376. data/spec/extensions/query_spec.rb +0 -94
  377. data/spec/extensions/rcte_tree_spec.rb +0 -381
  378. data/spec/extensions/round_timestamps_spec.rb +0 -39
  379. data/spec/extensions/s_spec.rb +0 -60
  380. data/spec/extensions/schema_caching_spec.rb +0 -64
  381. data/spec/extensions/schema_dumper_spec.rb +0 -870
  382. data/spec/extensions/select_remove_spec.rb +0 -38
  383. data/spec/extensions/sequel_4_dataset_methods_spec.rb +0 -121
  384. data/spec/extensions/serialization_modification_detection_spec.rb +0 -98
  385. data/spec/extensions/serialization_spec.rb +0 -365
  386. data/spec/extensions/server_block_spec.rb +0 -97
  387. data/spec/extensions/server_logging_spec.rb +0 -45
  388. data/spec/extensions/sharding_spec.rb +0 -189
  389. data/spec/extensions/shared_caching_spec.rb +0 -151
  390. data/spec/extensions/single_table_inheritance_spec.rb +0 -347
  391. data/spec/extensions/singular_table_names_spec.rb +0 -22
  392. data/spec/extensions/skip_create_refresh_spec.rb +0 -18
  393. data/spec/extensions/spec_helper.rb +0 -63
  394. data/spec/extensions/split_array_nil_spec.rb +0 -24
  395. data/spec/extensions/split_values_spec.rb +0 -57
  396. data/spec/extensions/sql_comments_spec.rb +0 -33
  397. data/spec/extensions/sql_expr_spec.rb +0 -59
  398. data/spec/extensions/static_cache_spec.rb +0 -471
  399. data/spec/extensions/string_agg_spec.rb +0 -90
  400. data/spec/extensions/string_date_time_spec.rb +0 -95
  401. data/spec/extensions/string_stripper_spec.rb +0 -68
  402. data/spec/extensions/subclasses_spec.rb +0 -79
  403. data/spec/extensions/subset_conditions_spec.rb +0 -38
  404. data/spec/extensions/symbol_aref_refinement_spec.rb +0 -28
  405. data/spec/extensions/symbol_as_refinement_spec.rb +0 -21
  406. data/spec/extensions/synchronize_sql_spec.rb +0 -124
  407. data/spec/extensions/table_select_spec.rb +0 -83
  408. data/spec/extensions/tactical_eager_loading_spec.rb +0 -402
  409. data/spec/extensions/thread_local_timezones_spec.rb +0 -67
  410. data/spec/extensions/throw_failures_spec.rb +0 -74
  411. data/spec/extensions/timestamps_spec.rb +0 -209
  412. data/spec/extensions/to_dot_spec.rb +0 -153
  413. data/spec/extensions/touch_spec.rb +0 -226
  414. data/spec/extensions/tree_spec.rb +0 -334
  415. data/spec/extensions/typecast_on_load_spec.rb +0 -86
  416. data/spec/extensions/unlimited_update_spec.rb +0 -21
  417. data/spec/extensions/update_or_create_spec.rb +0 -83
  418. data/spec/extensions/update_primary_key_spec.rb +0 -105
  419. data/spec/extensions/update_refresh_spec.rb +0 -59
  420. data/spec/extensions/uuid_spec.rb +0 -101
  421. data/spec/extensions/validate_associated_spec.rb +0 -52
  422. data/spec/extensions/validation_class_methods_spec.rb +0 -1040
  423. data/spec/extensions/validation_contexts_spec.rb +0 -31
  424. data/spec/extensions/validation_helpers_spec.rb +0 -525
  425. data/spec/extensions/whitelist_security_spec.rb +0 -157
  426. data/spec/extensions/xml_serializer_spec.rb +0 -213
  427. data/spec/files/bad_down_migration/001_create_alt_basic.rb +0 -4
  428. data/spec/files/bad_down_migration/002_create_alt_advanced.rb +0 -4
  429. data/spec/files/bad_timestamped_migrations/1273253849_create_sessions.rb +0 -9
  430. data/spec/files/bad_timestamped_migrations/1273253851_create_nodes.rb +0 -9
  431. data/spec/files/bad_timestamped_migrations/1273253853_3_create_users.rb +0 -3
  432. data/spec/files/bad_up_migration/001_create_alt_basic.rb +0 -4
  433. data/spec/files/bad_up_migration/002_create_alt_advanced.rb +0 -3
  434. data/spec/files/convert_to_timestamp_migrations/001_create_sessions.rb +0 -9
  435. data/spec/files/convert_to_timestamp_migrations/002_create_nodes.rb +0 -9
  436. data/spec/files/convert_to_timestamp_migrations/003_3_create_users.rb +0 -4
  437. data/spec/files/convert_to_timestamp_migrations/1273253850_create_artists.rb +0 -9
  438. data/spec/files/convert_to_timestamp_migrations/1273253852_create_albums.rb +0 -9
  439. data/spec/files/double_migration/001_create_sessions.rb +0 -9
  440. data/spec/files/double_migration/002_create_nodes.rb +0 -19
  441. data/spec/files/double_migration/003_3_create_users.rb +0 -4
  442. data/spec/files/duplicate_integer_migrations/001_create_alt_advanced.rb +0 -4
  443. data/spec/files/duplicate_integer_migrations/001_create_alt_basic.rb +0 -4
  444. data/spec/files/duplicate_timestamped_migrations/1273253849_create_sessions.rb +0 -9
  445. data/spec/files/duplicate_timestamped_migrations/1273253853_create_nodes.rb +0 -9
  446. data/spec/files/duplicate_timestamped_migrations/1273253853_create_users.rb +0 -4
  447. data/spec/files/empty_migration/001_create_sessions.rb +0 -9
  448. data/spec/files/empty_migration/002_create_nodes.rb +0 -0
  449. data/spec/files/empty_migration/003_3_create_users.rb +0 -4
  450. data/spec/files/integer_migrations/001_create_sessions.rb +0 -9
  451. data/spec/files/integer_migrations/002_create_nodes.rb +0 -9
  452. data/spec/files/integer_migrations/003_3_create_users.rb +0 -4
  453. data/spec/files/interleaved_timestamped_migrations/1273253849_create_sessions.rb +0 -9
  454. data/spec/files/interleaved_timestamped_migrations/1273253850_create_artists.rb +0 -9
  455. data/spec/files/interleaved_timestamped_migrations/1273253851_create_nodes.rb +0 -9
  456. data/spec/files/interleaved_timestamped_migrations/1273253852_create_albums.rb +0 -9
  457. data/spec/files/interleaved_timestamped_migrations/1273253853_3_create_users.rb +0 -4
  458. data/spec/files/missing_integer_migrations/001_create_alt_basic.rb +0 -4
  459. data/spec/files/missing_integer_migrations/003_create_alt_advanced.rb +0 -4
  460. data/spec/files/missing_timestamped_migrations/1273253849_create_sessions.rb +0 -9
  461. data/spec/files/missing_timestamped_migrations/1273253853_3_create_users.rb +0 -4
  462. data/spec/files/reversible_migrations/001_reversible.rb +0 -5
  463. data/spec/files/reversible_migrations/002_reversible.rb +0 -5
  464. data/spec/files/reversible_migrations/003_reversible.rb +0 -5
  465. data/spec/files/reversible_migrations/004_reversible.rb +0 -5
  466. data/spec/files/reversible_migrations/005_reversible.rb +0 -10
  467. data/spec/files/reversible_migrations/006_reversible.rb +0 -10
  468. data/spec/files/reversible_migrations/007_reversible.rb +0 -10
  469. data/spec/files/timestamped_migrations/1273253849_create_sessions.rb +0 -9
  470. data/spec/files/timestamped_migrations/1273253851_create_nodes.rb +0 -9
  471. data/spec/files/timestamped_migrations/1273253853_3_create_users.rb +0 -4
  472. data/spec/files/transaction_specified_migrations/001_create_alt_basic.rb +0 -4
  473. data/spec/files/transaction_specified_migrations/002_create_basic.rb +0 -4
  474. data/spec/files/transaction_unspecified_migrations/001_create_alt_basic.rb +0 -3
  475. data/spec/files/transaction_unspecified_migrations/002_create_basic.rb +0 -3
  476. data/spec/files/uppercase_timestamped_migrations/1273253849_CREATE_SESSIONS.RB +0 -9
  477. data/spec/files/uppercase_timestamped_migrations/1273253851_CREATE_NODES.RB +0 -9
  478. data/spec/files/uppercase_timestamped_migrations/1273253853_3_CREATE_USERS.RB +0 -4
  479. data/spec/guards_helper.rb +0 -59
  480. data/spec/integration/associations_test.rb +0 -2597
  481. data/spec/integration/database_test.rb +0 -113
  482. data/spec/integration/dataset_test.rb +0 -1981
  483. data/spec/integration/eager_loader_test.rb +0 -687
  484. data/spec/integration/migrator_test.rb +0 -262
  485. data/spec/integration/model_test.rb +0 -203
  486. data/spec/integration/plugin_test.rb +0 -2396
  487. data/spec/integration/prepared_statement_test.rb +0 -405
  488. data/spec/integration/schema_test.rb +0 -889
  489. data/spec/integration/spec_helper.rb +0 -65
  490. data/spec/integration/timezone_test.rb +0 -86
  491. data/spec/integration/transaction_test.rb +0 -603
  492. data/spec/integration/type_test.rb +0 -127
  493. data/spec/model/association_reflection_spec.rb +0 -803
  494. data/spec/model/associations_spec.rb +0 -4738
  495. data/spec/model/base_spec.rb +0 -875
  496. data/spec/model/class_dataset_methods_spec.rb +0 -146
  497. data/spec/model/dataset_methods_spec.rb +0 -198
  498. data/spec/model/eager_loading_spec.rb +0 -2377
  499. data/spec/model/hooks_spec.rb +0 -370
  500. data/spec/model/inflector_spec.rb +0 -26
  501. data/spec/model/model_spec.rb +0 -956
  502. data/spec/model/plugins_spec.rb +0 -429
  503. data/spec/model/record_spec.rb +0 -2118
  504. data/spec/model/spec_helper.rb +0 -46
  505. data/spec/model/validations_spec.rb +0 -220
  506. data/spec/model_no_assoc_spec.rb +0 -1
  507. data/spec/model_spec.rb +0 -1
  508. data/spec/plugin_spec.rb +0 -1
  509. data/spec/sequel_coverage.rb +0 -15
  510. data/spec/sequel_warning.rb +0 -4
  511. data/spec/spec_config.rb +0 -12
@@ -0,0 +1,728 @@
1
+ # frozen-string-literal: true
2
+
3
+ # :nocov:
4
+ raise(Sequel::Error, "Sequel column_encryption plugin requires ruby 2.3 or greater") unless RUBY_VERSION >= '2.3'
5
+ # :nocov:
6
+
7
+ require 'openssl'
8
+
9
+ begin
10
+ # Test cipher actually works
11
+ cipher = OpenSSL::Cipher.new("aes-256-gcm")
12
+ cipher.encrypt
13
+ cipher.key = '1'*32
14
+ cipher_iv = cipher.random_iv
15
+ cipher.auth_data = ''
16
+ cipher_text = cipher.update('2') << cipher.final
17
+ auth_tag = cipher.auth_tag
18
+
19
+ cipher = OpenSSL::Cipher.new("aes-256-gcm")
20
+ cipher.decrypt
21
+ cipher.iv = cipher_iv
22
+ cipher.key = '1'*32
23
+ cipher.auth_data = ''
24
+ cipher.auth_tag = auth_tag
25
+ # :nocov:
26
+ unless (cipher.update(cipher_text) << cipher.final) == '2'
27
+ raise OpenSSL::Cipher::CipherError
28
+ end
29
+ rescue RuntimeError, OpenSSL::Cipher::CipherError
30
+ raise LoadError, "Sequel column_encryption plugin requires a working aes-256-gcm cipher"
31
+ # :nocov:
32
+ end
33
+
34
+ require 'base64'
35
+ require 'securerandom'
36
+
37
+ module Sequel
38
+ module Plugins
39
+ # The column_encryption plugin adds support for encrypting the content of individual
40
+ # columns in a table.
41
+ #
42
+ # Column values are encrypted with AES-256-GCM using a per-value cipher key derived from
43
+ # a key provided in the configuration using HMAC-SHA256.
44
+ #
45
+ # = Usage
46
+ #
47
+ # If you would like to support encryption of columns in more than one model, you should
48
+ # probably load the plugin into the parent class of your models and specify the keys:
49
+ #
50
+ # Sequel::Model.plugin :column_encryption do |enc|
51
+ # enc.key 0, ENV["SEQUEL_COLUMN_ENCRYPTION_KEY"]
52
+ # end
53
+ #
54
+ # This specifies a single master encryption key. Unless you are actively rotating keys,
55
+ # it is best to use a single master key. Rotation of encryption keys will be discussed
56
+ # in a later section.
57
+ #
58
+ # In the above call, <tt>0</tt> is the id of the key, and the
59
+ # <tt>ENV["SEQUEL_COLUMN_ENCRYPTION_KEY"]</tt> is the content of the key, which must be
60
+ # a string with exactly 32 bytes. As indicated, this key should not be hardcoded or
61
+ # otherwise committed to the source control repository.
62
+ #
63
+ # For models that need encrypted columns, you load the plugin again, but specify the
64
+ # columns to encrypt:
65
+ #
66
+ # ConfidentialModel.plugin :column_encryption do |enc|
67
+ # enc.column :encrypted_column_name
68
+ # enc.column :searchable_column_name, searchable: true
69
+ # enc.column :ci_searchable_column_name, searchable: :case_insensitive
70
+ # end
71
+ #
72
+ # With this, all three specified columns (+encrypted_column_name+, +searchable_column_name+,
73
+ # and +ci_searchable_column_name+) will be marked as encrypted columns. When you run the
74
+ # following code:
75
+ #
76
+ # ConfidentialModel.create(
77
+ # encrypted_column_name: 'These',
78
+ # searchable_column_name: 'will be',
79
+ # ci_searchable_column_name: 'Encrypted'
80
+ # )
81
+ #
82
+ # It will save encrypted versions to the database. +encrypted_column_name+ will not be
83
+ # searchable, +searchable_column_name+ will be searchable with an exact match, and
84
+ # +ci_searchable_column_name+ will be searchable with a case insensitive match. See section
85
+ # below for details on searching.
86
+ #
87
+ # It is possible to have model-specific keys by specifying both the +key+ and +column+ methods
88
+ # in the model:
89
+ #
90
+ # ConfidentialModel.plugin :column_encryption do |enc|
91
+ # enc.key 0, ENV["SEQUEL_MODEL_SPECIFIC_ENCRYPTION_KEY"]
92
+ #
93
+ # enc.column :encrypted_column_name
94
+ # enc.column :searchable_column_name, searchable: true
95
+ # enc.column :ci_searchable_column_name, searchable: :case_insensitive
96
+ # end
97
+ #
98
+ # When the +key+ method is called inside the plugin block, previous keys are ignored,
99
+ # and only the new keys specified will be used. This approach would allow the
100
+ # +ConfidentialModel+ to use the model specific encryption keys, and other models
101
+ # to use the default keys specified in the parent class.
102
+ #
103
+ # The +key+ and +column+ methods inside the plugin block support additional options.
104
+ # The +key+ method supports the following options:
105
+ #
106
+ # :auth_data :: The authentication data to use for the AES-256-GCM cipher. Defaults
107
+ # to the empty string.
108
+ # :padding :: The number of padding bytes to use. For security, data is padded so that
109
+ # a database administrator cannot determine the exact size of the
110
+ # unencrypted data. By default, this value is 8, which means that
111
+ # unencrypted data will be padded to a multiple of 8 bytes. Up to twice as
112
+ # much padding as specified will be used, as the number of padding bytes
113
+ # is partially randomized.
114
+ #
115
+ # The +column+ method supports the following options:
116
+ #
117
+ # :searchable :: Whether the column is searchable. This should not be used unless
118
+ # searchability is needed, as it can allow the database administrator
119
+ # to determine whether two distinct rows have the same unencrypted
120
+ # data (but not what that data is). This can be set to +true+ to allow
121
+ # searching with an exact match, or +:case_insensitive+ for a case
122
+ # insensitive match.
123
+ # :search_both :: This should only be used if you have previously switched the
124
+ # +:searchable+ option from +true+ to +:case_insensitive+ or vice-versa,
125
+ # and would like the search to return values that have not yet been
126
+ # reencrypted. Note that switching from +true+ to +:case_insensitive+
127
+ # isn't a problem, but switching from +:case_insensitive+ to +true+ and
128
+ # using this option can cause the search to return values that are
129
+ # not an exact match. You should manually filter those objects
130
+ # after decrypting if you want to ensure an exact match.
131
+ # :format :: The format of the column, if you want to perform serialization before
132
+ # encryption and deserialization after decryption. Can be either a
133
+ # symbol registered with the serialization plugin or an array of two
134
+ # callables, the first for serialization and the second for deserialization.
135
+ #
136
+ # The +column+ method also supports a block for column-specific keys:
137
+ #
138
+ # ConfidentialModel.plugin :column_encryption do |enc|
139
+ # enc.column :encrypted_column_name do |cenc|
140
+ # cenc.key 0, ENV["SEQUEL_COLUMN_SPECIFIC_ENCRYPTION_KEY"]
141
+ # end
142
+ #
143
+ # enc.column :searchable_column_name, searchable: true
144
+ # enc.column :ci_searchable_column_name, searchable: :case_insensitive
145
+ # end
146
+ #
147
+ # In this case, the <tt>ENV["SEQUEL_COLUMN_SPECIFIC_ENCRYPTION_KEY"]</tt> key will
148
+ # only be used for the +:encrypted_column_name+ column, and not the other columns.
149
+ #
150
+ # Note that there isn't a security reason to prefer either model-specific or
151
+ # column-specific keys, as the actual cipher key used is unique per column value.
152
+ #
153
+ # Note that changing the key_id, key string, or auth_data for an existing key will
154
+ # break decryption of values encrypted with that key. If you would like to change
155
+ # any aspect of the key, add a new key, rotate to the new encryption key, and then
156
+ # remove the previous key, as described in the section below on key rotation.
157
+ #
158
+ # = Searching Encrypted Values
159
+ #
160
+ # To search searchable encrypted columns, use +with_encrypted_value+. This example
161
+ # code will return the model instance created in the code example in the previous
162
+ # section:
163
+ #
164
+ # ConfidentialModel.
165
+ # with_encrypted_value(:searchable_column_name, "will be")
166
+ # with_encrypted_value(:ci_searchable_column_name, "encrypted").
167
+ # first
168
+ #
169
+ # = Encryption Key Rotation
170
+ #
171
+ # To rotate encryption keys, add a new key above the existing key, with a new key ID:
172
+ #
173
+ # Sequel::Model.plugin :column_encryption do |enc|
174
+ # enc.key 1, ENV["SEQUEL_COLUMN_ENCRYPTION_KEY"]
175
+ # enc.key 0, ENV["SEQUEL_OLD_COLUMN_ENCRYPTION_KEY"]
176
+ # end
177
+ #
178
+ # Newly encrypted data will then use the new key. Records encrypted with the older key
179
+ # will still be decrypted correctly.
180
+ #
181
+ # To force reencryption for existing records that are using the older key, you can use
182
+ # the +needing_reencryption+ dataset method and the +reencrypt+ instance method. For a
183
+ # small number of records, you can probably do:
184
+ #
185
+ # ConfidentialModel.needing_reencryption.all(&:reencrypt)
186
+ #
187
+ # With more than a small number of records, you'll want to do this in batches. It's
188
+ # possible you could use an approach such as:
189
+ #
190
+ # ds = ConfidentialModel.needing_reencryption.limit(100)
191
+ # true until ds.all(&:reencrypt).empty?
192
+ #
193
+ # After all values have been reencrypted for all models, and no models use the older
194
+ # encryption key, you can remove it from the configuration:
195
+ #
196
+ # Sequel::Model.plugin :column_encryption do |enc|
197
+ # enc.key 1, ENV["SEQUEL_COLUMN_ENCRYPTION_KEY"]
198
+ # end
199
+ #
200
+ # Once an encryption key has been removed, after no data uses it, it is safe to reuse
201
+ # the same key id for a new key. This approach allows for up to 256 concurrent keys
202
+ # in the same configuration.
203
+ #
204
+ # = Encrypting Additional Formats
205
+ #
206
+ # By default, the column_encryption plugin assumes that the decrypted data should be
207
+ # returned as a string, and a string will be passed to encrypt. However, using the
208
+ # +:format+ option, you can specify an alternate format. For example, if you want to
209
+ # encrypt a JSON representation of the object, so that you can deal with an array/hash
210
+ # and automatically have it serialized with JSON and then encrypted when saving, and
211
+ # then deserialized with JSON after decryption when it is retrieved:
212
+ #
213
+ # require 'json'
214
+ # ConfidentialModel.plugin :column_encryption do |enc|
215
+ # enc.key 0, ENV["SEQUEL_MODEL_SPECIFIC_ENCRYPTION_KEY"]
216
+ #
217
+ # enc.column :encrypted_column_name
218
+ # enc.column :searchable_column_name, searchable: true
219
+ # enc.column :ci_searchable_column_name, searchable: :case_insensitive
220
+ # enc.column :encrypted_json_column_name, format: :json
221
+ # end
222
+ #
223
+ # The values of the +:format+ are the same values you can pass as the first argument
224
+ # to +serialize_attributes+ (in the serialization plugin). You can pass an array
225
+ # with the serializer and deserializer for custom support.
226
+ #
227
+ # You can use both +:searchable+ and +:format+ together for searchable encrypted
228
+ # serialized columns. However, note that this allows only exact searches of the
229
+ # serialized version of the data. So for JSON, a search for <tt>{'a'=>1, 'b'=>2}</tt>
230
+ # would not match <tt>{'b'=>2, 'a'=>1}</tt> even though the objects are considered
231
+ # equal. If this is an issue, make sure you use a serialization format where all
232
+ # equal objects are serialized to the same string.
233
+ #
234
+ # = Enforcing Uniqueness
235
+ #
236
+ # You cannot enforce uniqueness of unencrypted data at the database level
237
+ # if you also want to support key rotation. However, absent key rotation, a
238
+ # unique index on the first 48 characters of the encrypted column can enforce uniqueness,
239
+ # as long as the column is searchable. If the encrypted column is case-insensitive
240
+ # searchable, the uniqueness is case insensitive as well.
241
+ #
242
+ # = Column Value Cryptography/Format
243
+ #
244
+ # Column values used by this plugin use the following format (+key+ is specified
245
+ # in the plugin configuration and must be exactly 32 bytes):
246
+ #
247
+ # column_value :: urlsafe_base64(flags + NUL + key_id + NUL + search_data + key_data +
248
+ # cipher_iv + cipher_auth_tag + encrypted_data)
249
+ # flags :: 1 byte, the type of record (0: not searchable, 1: searchable, 2: lowercase searchable)
250
+ # NUL :: 1 byte, ASCII NUL
251
+ # key_id :: 1 byte, the key id, supporting 256 concurrently active keys (0 - 255)
252
+ # search_data :: 0 bytes if flags is 0, 32 bytes if flags is 1 or 2.
253
+ # Format is HMAC-SHA256(key, unencrypted_data).
254
+ # Ignored on decryption, only used for searching.
255
+ # key_data :: 32 bytes random data used to construct cipher key
256
+ # cipher_iv :: 12 bytes, AES-256-GCM cipher random initialization vector
257
+ # cipher_auth_tag :: 16 bytes, AES-256-GCM cipher authentication tag
258
+ # encrypted_data :: AES-256-GCM(HMAC-SHA256(key, key_data),
259
+ # padding_size + padding + unencrypted_data)
260
+ # padding_size :: 1 byte, with the amount of padding (0-255 bytes of padding allowed)
261
+ # padding :: number of bytes specified by padding size, ignored on decryption
262
+ # unencrypted_data :: actual column value
263
+ #
264
+ # The reason for <tt>flags + NUL + key_id + NUL</tt> (4 bytes) as the header is to allow for
265
+ # an easy way to search for values needing reencryption using a database index. It takes
266
+ # the first three bytes and converts them to base64, and looks for values less than that value
267
+ # or greater than that value with 'B' appended. The NUL byte in the fourth byte of the header
268
+ # ensures that after base64 encoding, the fifth byte in the column will be 'A'.
269
+ #
270
+ # The reason for <tt>search_data</tt> (32 bytes) directly after is that for searchable values,
271
+ # after base64 encoding of the header and search data, it is 48 bytes and can be used directly
272
+ # as a prefix search on the column, which can be supported by the same database index. This is
273
+ # more efficient than a full column value search for large values, and allows for case-insensitive
274
+ # searching without a separate column, by having the search_data be based on the lowercase value
275
+ # while the unencrypted data is original case.
276
+ #
277
+ # The reason for the padding is so that a database administrator cannot be sure exactly how
278
+ # many bytes are in the column. It is stored encrypted because otherwise the database
279
+ # administrator could calculate it by decoding the base64 data.
280
+ #
281
+ # = Unsupported Features
282
+ #
283
+ # The following features are delibrately not supported:
284
+ #
285
+ # == Compression
286
+ #
287
+ # Allowing compression with encryption is inviting security issues later.
288
+ # While padding can reduce the risk of compression with encryption, it does not
289
+ # eliminate it entirely. Users that must have compression with encryption can use
290
+ # the +:format+ option with a serializer that compresses and a deserializer that
291
+ # decompresses.
292
+ #
293
+ # == Mixing Encrypted/Unencrypted Data
294
+ #
295
+ # Mixing encrypted and unencrypted data increases the complexity and security risk, since there
296
+ # is a chance unencrypted data could look like encrypted data in the pathologic case.
297
+ # If you have existing unencrypted data that would like to encrypt, create a new column for
298
+ # the encrypted data, and then migrate the data from the unencrypted column to the encrypted
299
+ # column. After all unencrypted values have been migrated, drop the unencrypted column.
300
+ #
301
+ # == Arbitrary Encryption Schemes
302
+ #
303
+ # Supporting arbitrary encryption schemes increases the complexity risk.
304
+ # If in the future AES-256-GCM is not considered a secure enough cipher, it is possible to
305
+ # extend the current format using the reserved values in the first two bytes of the header.
306
+ #
307
+ # = Caveats
308
+ #
309
+ # As column_encryption is a model plugin, it only works with using model instance methods.
310
+ # If you directly modify the database using a dataset or an external program that modifies
311
+ # the contents of the encrypted columns, you will probably corrupt the data. To make data
312
+ # corruption less likely, it is best to have a CHECK constraints on the encrypted column
313
+ # with a basic format and length check:
314
+ #
315
+ # DB.alter_table(:table_name) do
316
+ # c = Sequel[:encrypted_column_name]
317
+ # add_constraint(:encrypted_column_name_format,
318
+ # c.like('AA__A%') | c.like('Ag__A%') | c.like('AQ__A%'))
319
+ # add_constraint(:encrypted_column_name_length, Sequel.char_length(c) >= 88)
320
+ # end
321
+ #
322
+ # If possible, it's also best to check that the column is valid urlsafe base64 data of
323
+ # sufficient length. This can be done on PostgreSQL using a combination of octet_length,
324
+ # decode, and regexp_replace:
325
+ #
326
+ # DB.alter_table(:ce_test) do
327
+ # c = Sequel[:encrypted_column_name]
328
+ # add_constraint(:enc_base64) do
329
+ # octet_length(decode(regexp_replace(regexp_replace(c, '_', '/', 'g'), '-', '+', 'g'), 'base64')) >= 65}
330
+ # end
331
+ # end
332
+ #
333
+ # Such constraints will probably be sufficient to protect against most unintentional corruption of
334
+ # encrypted columns.
335
+ #
336
+ # If the database supports transparent data encryption and you trust the database administrator,
337
+ # using the database support is probably a better approach.
338
+ #
339
+ # The column_encryption plugin is only supported on Ruby 2.3+ and when the Ruby openssl standard
340
+ # library supports the AES-256-GCM cipher.
341
+ module ColumnEncryption
342
+ # Cryptor handles the encryption and decryption of rows for a key set.
343
+ # It also provides methods that return search prefixes, which datasets
344
+ # use in queries.
345
+ #
346
+ # The same cryptor can support non-searchable, searchable, and case-insensitive
347
+ # searchable columns.
348
+ class Cryptor # :nodoc:
349
+ # Flags
350
+ NOT_SEARCHABLE = 0
351
+ SEARCHABLE = 1
352
+ LOWERCASE_SEARCHABLE = 2
353
+
354
+ # This is the default padding, but up to 2x the padding can be used for a record.
355
+ DEFAULT_PADDING = 8
356
+
357
+ # Keys should be an array of arrays containing key_id, key string, auth_data, and padding.
358
+ def initialize(keys)
359
+ if keys.empty?
360
+ raise Error, "Cannot initialize encryptor without encryption key"
361
+ end
362
+
363
+ # First key is used for encryption
364
+ @key_id, @key, @auth_data, @padding = keys[0]
365
+
366
+ # All keys are candidates for decryption
367
+ @key_map = {}
368
+ keys.each do |key_id, key, auth_data, padding|
369
+ @key_map[key_id] = [key, auth_data, padding].freeze
370
+ end
371
+
372
+ freeze
373
+ end
374
+
375
+ # Decrypt using any supported format and any available key.
376
+ def decrypt(data)
377
+ begin
378
+ data = Base64.urlsafe_decode64(data)
379
+ rescue ArgumentError
380
+ raise Error, "Unable to decode encrypted column: invalid base64"
381
+ end
382
+
383
+ unless data.getbyte(1) == 0 && data.getbyte(3) == 0
384
+ raise Error, "Unable to decode encrypted column: invalid format"
385
+ end
386
+
387
+ flags = data.getbyte(0)
388
+
389
+ key, auth_data = @key_map[data.getbyte(2)]
390
+ unless key
391
+ raise Error, "Unable to decode encrypted column: invalid key id"
392
+ end
393
+
394
+ case flags
395
+ when NOT_SEARCHABLE
396
+ if data.bytesize < 65
397
+ raise Error, "Decoded encrypted column smaller than minimum size"
398
+ end
399
+
400
+ data.slice!(0, 4)
401
+ when SEARCHABLE, LOWERCASE_SEARCHABLE
402
+ if data.bytesize < 97
403
+ raise Error, "Decoded encrypted column smaller than minimum size"
404
+ end
405
+
406
+ data.slice!(0, 36)
407
+ else
408
+ raise Error, "Unable to decode encrypted column: invalid flags"
409
+ end
410
+
411
+ key_part = data.slice!(0, 32)
412
+ cipher_iv = data.slice!(0, 12)
413
+ auth_tag = data.slice!(0, 16)
414
+
415
+ cipher = OpenSSL::Cipher.new("aes-256-gcm")
416
+ cipher.decrypt
417
+ cipher.iv = cipher_iv
418
+ cipher.key = OpenSSL::HMAC.digest(OpenSSL::Digest::SHA256.new, key, key_part)
419
+ cipher.auth_data = auth_data
420
+ cipher.auth_tag = auth_tag
421
+ begin
422
+ decrypted_data = cipher.update(data) << cipher.final
423
+ rescue OpenSSL::Cipher::CipherError => e
424
+ raise Error, "Unable to decrypt encrypted column: #{e.class} (probably due to encryption key or auth data mismatch or corrupt data)"
425
+ end
426
+
427
+ # Remove padding
428
+ decrypted_data.slice!(0, decrypted_data.getbyte(0) + 1)
429
+
430
+ decrypted_data
431
+ end
432
+
433
+ # Encrypt in not searchable format with the first configured encryption key.
434
+ def encrypt(data)
435
+ _encrypt(data, "#{NOT_SEARCHABLE.chr}\0#{@key_id.chr}\0")
436
+ end
437
+
438
+ # Encrypt in searchable format with the first configured encryption key.
439
+ def searchable_encrypt(data)
440
+ _encrypt(data, _search_prefix(data, SEARCHABLE, @key_id, @key))
441
+ end
442
+
443
+ # Encrypt in case insensitive searchable format with the first configured encryption key.
444
+ def case_insensitive_searchable_encrypt(data)
445
+ _encrypt(data, _search_prefix(data.downcase, LOWERCASE_SEARCHABLE, @key_id, @key))
446
+ end
447
+
448
+ # The prefix string of columns for the given search type and the first configured encryption key.
449
+ # Used to find values that do not use this prefix in order to perform reencryption.
450
+ def current_key_prefix(search_type)
451
+ Base64.urlsafe_encode64("#{search_type.chr}\0#{@key_id.chr}")
452
+ end
453
+
454
+ # The prefix values to search for the given data (an array of strings), assuming the column uses
455
+ # the searchable format.
456
+ def search_prefixes(data)
457
+ _search_prefixes(data, SEARCHABLE)
458
+ end
459
+
460
+ # The prefix values to search for the given data (an array of strings), assuming the column uses
461
+ # the case insensitive searchable format.
462
+ def lowercase_search_prefixes(data)
463
+ _search_prefixes(data.downcase, LOWERCASE_SEARCHABLE)
464
+ end
465
+
466
+ # The prefix values to search for the given data (an array of strings), assuming the column uses
467
+ # either the searchable or the case insensitive searchable format. Should be used only when
468
+ # transitioning between formats (used by the :search_both option when encrypting columns).
469
+ def regular_and_lowercase_search_prefixes(data)
470
+ search_prefixes(data) + lowercase_search_prefixes(data)
471
+ end
472
+
473
+ private
474
+
475
+ # An array of strings, one for each configured encryption key, to find encypted values matching
476
+ # the given data and search format.
477
+ def _search_prefixes(data, search_type)
478
+ @key_map.map do |key_id, (key, _)|
479
+ Base64.urlsafe_encode64(_search_prefix(data, search_type, key_id, key))
480
+ end
481
+ end
482
+
483
+ # The prefix to use for searchable data, including the HMAC-SHA256(key, data).
484
+ def _search_prefix(data, search_type, key_id, key)
485
+ "#{search_type.chr}\0#{key_id.chr}\0#{OpenSSL::HMAC.digest(OpenSSL::Digest::SHA256.new, key, data)}"
486
+ end
487
+
488
+ # Encrypt the data using AES-256-GCM, with the given prefix.
489
+ def _encrypt(data, prefix)
490
+ padding = @padding
491
+ random_data = SecureRandom.random_bytes(32)
492
+ cipher = OpenSSL::Cipher.new("aes-256-gcm")
493
+ cipher.encrypt
494
+ cipher.key = OpenSSL::HMAC.digest(OpenSSL::Digest::SHA256.new, @key, random_data)
495
+ cipher_iv = cipher.random_iv
496
+ cipher.auth_data = @auth_data
497
+
498
+ cipher_text = String.new
499
+ data_size = data.bytesize
500
+
501
+ padding_size = if padding
502
+ (padding * rand(1)) + padding - (data.bytesize % padding)
503
+ else
504
+ 0
505
+ end
506
+
507
+ cipher_text << cipher.update(padding_size.chr)
508
+ cipher_text << cipher.update(SecureRandom.random_bytes(padding_size)) if padding_size > 0
509
+ cipher_text << cipher.update(data) if data_size > 0
510
+ cipher_text << cipher.final
511
+
512
+ Base64.urlsafe_encode64("#{prefix}#{random_data}#{cipher_iv}#{cipher.auth_tag}#{cipher_text}")
513
+ end
514
+ end
515
+
516
+ # The object type yielded to blocks passed to the +column+ method inside
517
+ # <tt>plugin :column_encryption</tt> blocks. This is used to configure custom
518
+ # per-column keys.
519
+ class ColumnDSL # :nodoc:
520
+ # An array of arrays for the data for the keys configured inside the block.
521
+ attr_reader :keys
522
+
523
+ def initialize
524
+ @keys = []
525
+ end
526
+
527
+ # Verify that the key_id, key, and options are value.
528
+ def key(key_id, key, opts=OPTS)
529
+ unless key_id.is_a?(Integer) && key_id >= 0 && key_id <= 255
530
+ raise Error, "invalid key_id argument, must be integer between 0 and 255"
531
+ end
532
+
533
+ unless key.is_a?(String) && key.bytesize == 32
534
+ raise Error, "invalid key argument, must be string with exactly 32 bytes"
535
+ end
536
+
537
+ if opts.has_key?(:padding)
538
+ if padding = opts[:padding]
539
+ unless padding.is_a?(Integer) && padding >= 1 && padding <= 120
540
+ raise Error, "invalid :padding option, must be between 1 and 120"
541
+ end
542
+ end
543
+ else
544
+ padding = Cryptor::DEFAULT_PADDING
545
+ end
546
+
547
+ @keys << [key_id, key, opts[:auth_data].to_s, padding].freeze
548
+ end
549
+ end
550
+
551
+ # The object type yielded to <tt>plugin :column_encryption</tt> blocks,
552
+ # used to configure encryption keys and encrypted columns.
553
+ class DSL < ColumnDSL # :nodoc:
554
+ # An array of arrays of data for the columns configured inside the block.
555
+ attr_reader :columns
556
+
557
+ def initialize
558
+ super
559
+ @columns = []
560
+ end
561
+
562
+ # Store the column information.
563
+ def column(column, opts=OPTS, &block)
564
+ @columns << [column, opts, block].freeze
565
+ end
566
+ end
567
+
568
+ def self.apply(model, opts=OPTS)
569
+ model.plugin :serialization
570
+ end
571
+
572
+ def self.configure(model)
573
+ dsl = DSL.new
574
+ yield dsl
575
+
576
+ model.instance_exec do
577
+ unless dsl.keys.empty?
578
+ @column_encryption_keys = dsl.keys.freeze
579
+ @column_encryption_cryptor = nil
580
+ end
581
+
582
+ @column_encryption_metadata = Hash[@column_encryption_metadata || {}]
583
+
584
+ dsl.columns.each do |column, opts, block|
585
+ _encrypt_column(column, opts, &block)
586
+ end
587
+
588
+ @column_encryption_metadata.freeze
589
+ end
590
+ end
591
+
592
+ # This stores four callables for handling encyption, decryption, data searching,
593
+ # and key searching. One of these is created for each encrypted column.
594
+ ColumnEncryptionMetadata = Struct.new(:encryptor, :decryptor, :data_searcher, :key_searcher) # :nodoc:
595
+
596
+ module ClassMethods
597
+ private
598
+
599
+ # A hash with column symbol keys and ColumnEncryptionMetadata values for each
600
+ # encrypted column.
601
+ attr_reader :column_encryption_metadata
602
+
603
+ # The default Cryptor to use for encrypted columns. This is only overridden if
604
+ # per-column keys are used.
605
+ def column_encryption_cryptor
606
+ @column_encryption_cryptor ||= Cryptor.new(@column_encryption_keys)
607
+ end
608
+
609
+ # Setup encryption for the given column.
610
+ def _encrypt_column(column, opts)
611
+ cryptor ||= if defined?(yield)
612
+ dsl = ColumnDSL.new
613
+ yield dsl
614
+ Cryptor.new(dsl.keys)
615
+ else
616
+ column_encryption_cryptor
617
+ end
618
+
619
+ encrypt_method, search_prefixes_method, search_type = case searchable = opts[:searchable]
620
+ when nil, false
621
+ [:encrypt, nil, Cryptor::NOT_SEARCHABLE]
622
+ when true
623
+ [:searchable_encrypt, :search_prefixes, Cryptor::SEARCHABLE]
624
+ when :case_insensitive
625
+ [:case_insensitive_searchable_encrypt, :lowercase_search_prefixes, Cryptor::LOWERCASE_SEARCHABLE]
626
+ else
627
+ raise Error, "invalid :searchable option for encrypted column: #{searchable.inspect}"
628
+ end
629
+
630
+ if searchable && opts[:search_both]
631
+ search_prefixes_method = :regular_and_lowercase_search_prefixes
632
+ end
633
+
634
+ # Setup the callables used in the metadata.
635
+ encryptor = cryptor.method(encrypt_method)
636
+ decryptor = cryptor.method(:decrypt)
637
+ data_searcher = cryptor.method(search_prefixes_method) if search_prefixes_method
638
+ key_searcher = lambda{cryptor.current_key_prefix(search_type)}
639
+
640
+ if format = opts[:format]
641
+ if format.is_a?(Symbol)
642
+ unless format = Sequel.synchronize{Serialization::REGISTERED_FORMATS[format]}
643
+ raise(Error, "Unsupported serialization format: #{format} (valid formats: #{Sequel.synchronize{Serialization::REGISTERED_FORMATS.keys}.inspect})")
644
+ end
645
+ end
646
+
647
+ # If a custom serialization format is used, override the
648
+ # callables to handle serialization and deserialization.
649
+ serializer, deserializer = format
650
+ enc, dec, data_s = encryptor, decryptor, data_searcher
651
+ encryptor = lambda do |data|
652
+ enc.call(serializer.call(data))
653
+ end
654
+ decryptor = lambda do |data|
655
+ deserializer.call(dec.call(data))
656
+ end
657
+ data_searcher = lambda do |data|
658
+ data_s.call(serializer.call(data))
659
+ end
660
+ end
661
+
662
+ # Setup the setter and getter methods to do encryption and decryption using
663
+ # the serialization plugin.
664
+ serialize_attributes([encryptor, decryptor], column)
665
+
666
+ column_encryption_metadata[column] = ColumnEncryptionMetadata.new(encryptor, decryptor, data_searcher, key_searcher).freeze
667
+
668
+ nil
669
+ end
670
+ end
671
+
672
+ module ClassMethods
673
+ Plugins.def_dataset_methods(self, [:with_encrypted_value, :needing_reencryption])
674
+
675
+ Plugins.inherited_instance_variables(self,
676
+ :@column_encryption_cryptor=>nil,
677
+ :@column_encryption_keys=>nil,
678
+ :@column_encryption_metadata=>nil,
679
+ )
680
+ end
681
+
682
+ module InstanceMethods
683
+ # Reencrypt the model if needed. Looks at all of the models encrypted columns
684
+ # and if any were encypted with older keys or a different format, reencrypt
685
+ # with the current key and format and save the object. Returns the object
686
+ # if reencryption was needed, or nil if reencryption was not needed.
687
+ def reencrypt
688
+ do_save = false
689
+
690
+ model.send(:column_encryption_metadata).each do |column, metadata|
691
+ if (value = values[column]) && !value.start_with?(metadata.key_searcher.call)
692
+ do_save = true
693
+ values[column] = metadata.encryptor.call(metadata.decryptor.call(value))
694
+ end
695
+ end
696
+
697
+ save if do_save
698
+ end
699
+ end
700
+
701
+ module DatasetMethods
702
+ # Filter the dataset to only match rows where the column contains an encrypted version
703
+ # of value. Only works on searchable encrypted columns.
704
+ def with_encrypted_value(column, value)
705
+ metadata = model.send(:column_encryption_metadata)[column]
706
+
707
+ unless metadata && metadata.data_searcher
708
+ raise Error, "lookup for encrypted column #{column.inspect} is not supported"
709
+ end
710
+
711
+ prefixes = metadata.data_searcher.call(value)
712
+ where(Sequel.|(*prefixes.map{|v| Sequel.like(column, "#{escape_like(v)}%")}))
713
+ end
714
+
715
+ # Filter the dataset to exclude rows where all encrypted columns are already encrypted
716
+ # with the current key and format.
717
+ def needing_reencryption
718
+ incorrect_column_prefixes = model.send(:column_encryption_metadata).map do |column, metadata|
719
+ prefix = metadata.key_searcher.call
720
+ (Sequel[column] < prefix) | (Sequel[column] > prefix + 'B')
721
+ end
722
+
723
+ where(Sequel.|(*incorrect_column_prefixes))
724
+ end
725
+ end
726
+ end
727
+ end
728
+ end