moneta 1.0.0 → 1.5.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (488) hide show
  1. checksums.yaml +5 -5
  2. data/.github/workflows/ruby.yml +425 -0
  3. data/.gitignore +3 -0
  4. data/.rspec +4 -0
  5. data/.rubocop.yml +194 -0
  6. data/.yardopts +6 -1
  7. data/CHANGES +88 -0
  8. data/CONTRIBUTORS +10 -2
  9. data/Gemfile +198 -76
  10. data/LICENSE +1 -1
  11. data/README.md +181 -83
  12. data/SPEC.md +70 -0
  13. data/feature_matrix.yaml +227 -0
  14. data/lib/action_dispatch/middleware/session/moneta_store.rb +1 -0
  15. data/lib/active_support/cache/moneta_store.rb +102 -6
  16. data/lib/moneta/adapter.rb +52 -0
  17. data/lib/moneta/adapters/activerecord.rb +284 -99
  18. data/lib/moneta/adapters/activesupportcache.rb +120 -0
  19. data/lib/moneta/adapters/cassandra.rb +303 -51
  20. data/lib/moneta/adapters/client.rb +72 -23
  21. data/lib/moneta/adapters/couch.rb +353 -49
  22. data/lib/moneta/adapters/datamapper.rb +11 -5
  23. data/lib/moneta/adapters/daybreak.rb +31 -20
  24. data/lib/moneta/adapters/dbm.rb +10 -16
  25. data/lib/moneta/adapters/file.rb +34 -13
  26. data/lib/moneta/adapters/fog.rb +5 -6
  27. data/lib/moneta/adapters/gdbm.rb +10 -16
  28. data/lib/moneta/adapters/hbase.rb +11 -13
  29. data/lib/moneta/adapters/kyotocabinet.rb +78 -17
  30. data/lib/moneta/adapters/leveldb.rb +35 -15
  31. data/lib/moneta/adapters/lmdb.rb +47 -23
  32. data/lib/moneta/adapters/localmemcache.rb +7 -13
  33. data/lib/moneta/adapters/lruhash.rb +45 -48
  34. data/lib/moneta/adapters/memcached/dalli.rb +67 -30
  35. data/lib/moneta/adapters/memcached/native.rb +33 -30
  36. data/lib/moneta/adapters/memcached.rb +1 -0
  37. data/lib/moneta/adapters/memory.rb +7 -7
  38. data/lib/moneta/adapters/mongo.rb +265 -6
  39. data/lib/moneta/adapters/null.rb +1 -2
  40. data/lib/moneta/adapters/pstore.rb +68 -25
  41. data/lib/moneta/adapters/redis.rb +101 -32
  42. data/lib/moneta/adapters/restclient.rb +17 -16
  43. data/lib/moneta/adapters/riak.rb +10 -11
  44. data/lib/moneta/adapters/sdbm.rb +10 -16
  45. data/lib/moneta/adapters/sequel/mysql.rb +66 -0
  46. data/lib/moneta/adapters/sequel/postgres.rb +80 -0
  47. data/lib/moneta/adapters/sequel/postgres_hstore.rb +240 -0
  48. data/lib/moneta/adapters/sequel/sqlite.rb +57 -0
  49. data/lib/moneta/adapters/sequel.rb +253 -52
  50. data/lib/moneta/adapters/sqlite.rb +103 -23
  51. data/lib/moneta/adapters/tdb.rb +21 -12
  52. data/lib/moneta/adapters/tokyocabinet.rb +24 -16
  53. data/lib/moneta/adapters/tokyotyrant.rb +49 -28
  54. data/lib/moneta/adapters/yaml.rb +1 -5
  55. data/lib/moneta/builder.rb +2 -3
  56. data/lib/moneta/cache.rb +11 -1
  57. data/lib/moneta/config.rb +101 -0
  58. data/lib/moneta/create_support.rb +21 -0
  59. data/lib/moneta/dbm_adapter.rb +31 -0
  60. data/lib/moneta/defaults.rb +320 -0
  61. data/lib/moneta/each_key_support.rb +27 -0
  62. data/lib/moneta/enumerable.rb +38 -0
  63. data/lib/moneta/expires.rb +112 -11
  64. data/lib/moneta/expires_support.rb +59 -0
  65. data/lib/moneta/fallback.rb +84 -0
  66. data/lib/moneta/hash_adapter.rb +68 -0
  67. data/lib/moneta/increment_support.rb +16 -0
  68. data/lib/moneta/lock.rb +26 -2
  69. data/lib/moneta/logger.rb +2 -2
  70. data/lib/moneta/nil_values.rb +35 -0
  71. data/lib/moneta/option_support.rb +51 -0
  72. data/lib/moneta/optionmerger.rb +0 -1
  73. data/lib/moneta/pool.rb +339 -28
  74. data/lib/moneta/proxy.rb +75 -2
  75. data/lib/moneta/server.rb +230 -68
  76. data/lib/moneta/shared.rb +36 -14
  77. data/lib/moneta/stack.rb +7 -7
  78. data/lib/moneta/synchronize.rb +5 -11
  79. data/lib/moneta/transformer/config.rb +64 -42
  80. data/lib/moneta/transformer/helper/bson.rb +7 -14
  81. data/lib/moneta/transformer/helper.rb +5 -5
  82. data/lib/moneta/transformer.rb +217 -23
  83. data/lib/moneta/utils.rb +3 -9
  84. data/lib/moneta/version.rb +1 -1
  85. data/lib/moneta/weak_each_key.rb +72 -0
  86. data/lib/moneta/wrapper.rb +30 -0
  87. data/lib/moneta.rb +29 -10
  88. data/lib/rack/cache/moneta.rb +13 -11
  89. data/lib/rack/moneta_rest.rb +2 -2
  90. data/lib/rack/session/moneta.rb +3 -4
  91. data/moneta.gemspec +19 -4
  92. data/script/benchmarks +311 -100
  93. data/script/contributors +11 -6
  94. data/script/parallel-tests +58 -83
  95. data/script/start-couchdb +27 -0
  96. data/script/start-hbase +47 -0
  97. data/script/start-services +5 -6
  98. data/script/update-feature-matrix +148 -0
  99. data/spec/active_support/cache_moneta_store_spec.rb +129 -106
  100. data/spec/features/concurrent_create.rb +42 -0
  101. data/spec/features/concurrent_increment.rb +39 -0
  102. data/spec/features/create.rb +25 -0
  103. data/spec/features/create_expires.rb +19 -0
  104. data/spec/features/default_expires.rb +14 -0
  105. data/spec/features/each_key.rb +119 -0
  106. data/spec/features/expires.rb +296 -0
  107. data/spec/features/features.rb +16 -0
  108. data/spec/features/increment.rb +98 -0
  109. data/spec/features/marshallable_key.rb +43 -0
  110. data/spec/features/marshallable_value.rb +7 -0
  111. data/spec/features/multiprocess.rb +8 -0
  112. data/spec/features/not_create.rb +7 -0
  113. data/spec/features/not_each_key.rb +7 -0
  114. data/spec/features/not_increment.rb +13 -0
  115. data/spec/features/not_persist.rb +9 -0
  116. data/spec/features/null.rb +66 -0
  117. data/spec/features/persist.rb +14 -0
  118. data/spec/features/returndifferent.rb +9 -0
  119. data/spec/features/returnsame.rb +9 -0
  120. data/spec/features/store.rb +261 -0
  121. data/spec/features/store_large.rb +13 -0
  122. data/spec/features/transform_value.rb +44 -0
  123. data/spec/features/transform_value_expires.rb +41 -0
  124. data/spec/helper.rb +345 -78
  125. data/spec/moneta/adapters/activerecord/adapter_activerecord_existing_connection_spec.rb +61 -0
  126. data/spec/moneta/adapters/activerecord/adapter_activerecord_spec.rb +126 -0
  127. data/spec/moneta/adapters/activerecord/standard_activerecord_spec.rb +21 -0
  128. data/spec/moneta/adapters/activerecord/standard_activerecord_with_expires_spec.rb +27 -0
  129. data/spec/moneta/adapters/activesupportcache/adapter_activesupportcache_spec.rb +54 -0
  130. data/spec/moneta/adapters/activesupportcache/adapter_activesupportcache_with_default_expires_spec.rb +54 -0
  131. data/spec/moneta/adapters/activesupportcache/standard_activesupportcache_spec.rb +14 -0
  132. data/spec/moneta/adapters/cassandra/adapter_cassandra_spec.rb +17 -0
  133. data/spec/moneta/adapters/cassandra/adapter_cassandra_with_default_expires_spec.rb +18 -0
  134. data/spec/moneta/adapters/cassandra/helper.rb +18 -0
  135. data/spec/moneta/adapters/cassandra/standard_cassandra_spec.rb +18 -0
  136. data/spec/moneta/adapters/client/adapter_client_spec.rb +11 -0
  137. data/spec/moneta/adapters/client/client_helper.rb +25 -0
  138. data/spec/moneta/adapters/client/standard_client_tcp_spec.rb +23 -0
  139. data/spec/moneta/adapters/client/standard_client_unix_spec.rb +28 -0
  140. data/spec/moneta/adapters/cookie/adapter_cookie_spec.rb +7 -0
  141. data/spec/moneta/adapters/couch/adapter_couch_spec.rb +204 -0
  142. data/spec/moneta/adapters/couch/standard_couch_spec.rb +15 -0
  143. data/spec/moneta/adapters/couch/standard_couch_with_expires_spec.rb +19 -0
  144. data/spec/moneta/adapters/datamapper/adapter_datamapper_spec.rb +60 -0
  145. data/spec/moneta/adapters/datamapper/standard_datamapper_spec.rb +21 -0
  146. data/spec/moneta/adapters/datamapper/standard_datamapper_with_expires_spec.rb +26 -0
  147. data/spec/moneta/adapters/datamapper/standard_datamapper_with_repository_spec.rb +22 -0
  148. data/spec/moneta/adapters/daybreak/adapter_daybreak_spec.rb +7 -0
  149. data/spec/moneta/adapters/daybreak/standard_daybreak_spec.rb +7 -0
  150. data/spec/moneta/adapters/daybreak/standard_daybreak_with_expires_spec.rb +11 -0
  151. data/spec/moneta/adapters/dbm/adapter_dbm_spec.rb +7 -0
  152. data/spec/moneta/adapters/dbm/standard_dbm_spec.rb +7 -0
  153. data/spec/moneta/adapters/dbm/standard_dbm_with_expires_spec.rb +12 -0
  154. data/spec/moneta/adapters/faraday_helper.rb +10 -0
  155. data/spec/moneta/adapters/file/adapter_file_spec.rb +7 -0
  156. data/spec/moneta/adapters/file/standard_file_spec.rb +7 -0
  157. data/spec/moneta/adapters/file/standard_file_with_expires_spec.rb +11 -0
  158. data/spec/moneta/adapters/fog/adapter_fog_spec.rb +16 -0
  159. data/spec/moneta/adapters/fog/standard_fog_spec.rb +15 -0
  160. data/spec/moneta/adapters/fog/standard_fog_with_expires_spec.rb +20 -0
  161. data/spec/moneta/adapters/gdbm/adapter_gdbm_spec.rb +7 -0
  162. data/spec/moneta/adapters/gdbm/standard_gdbm_spec.rb +7 -0
  163. data/spec/moneta/adapters/gdbm/standard_gdbm_with_expires_spec.rb +11 -0
  164. data/spec/moneta/adapters/hashfile/standard_hashfile_spec.rb +7 -0
  165. data/spec/moneta/adapters/hashfile/standard_hashfile_with_expires_spec.rb +12 -0
  166. data/spec/moneta/adapters/hbase/adapter_hbase_spec.rb +7 -0
  167. data/spec/moneta/adapters/hbase/standard_hbase_spec.rb +4 -0
  168. data/spec/moneta/adapters/hbase/standard_hbase_with_expires_spec.rb +8 -0
  169. data/spec/moneta/adapters/kyotocabinet/adapter_kyotocabinet_spec.rb +7 -0
  170. data/spec/moneta/adapters/kyotocabinet/standard_kyotocabinet_spec.rb +7 -0
  171. data/spec/moneta/adapters/kyotocabinet/standard_kyotocabinet_with_expires_spec.rb +14 -0
  172. data/spec/moneta/adapters/leveldb/adapter_leveldb_spec.rb +7 -0
  173. data/spec/moneta/adapters/leveldb/standard_leveldb_spec.rb +7 -0
  174. data/spec/moneta/adapters/leveldb/standard_leveldb_with_expires_spec.rb +11 -0
  175. data/spec/moneta/adapters/lmdb/adapter_lmdb_spec.rb +7 -0
  176. data/spec/moneta/adapters/lmdb/adapter_lmdb_with_db_spec.rb +7 -0
  177. data/spec/moneta/adapters/lmdb/standard_lmdb_spec.rb +7 -0
  178. data/spec/moneta/adapters/lmdb/standard_lmdb_with_expires_spec.rb +11 -0
  179. data/spec/moneta/adapters/localmemcache/adapter_localmemcache_spec.rb +7 -0
  180. data/spec/moneta/adapters/localmemcache/standard_localmemcache_spec.rb +7 -0
  181. data/spec/moneta/adapters/localmemcache/standard_localmemcache_with_expires_spec.rb +11 -0
  182. data/spec/moneta/{adapter_lruhash_spec.rb → adapters/lruhash/adapter_lruhash_spec.rb} +15 -33
  183. data/spec/moneta/adapters/lruhash/standard_lruhash_spec.rb +4 -0
  184. data/spec/moneta/adapters/lruhash/standard_lruhash_with_expires_spec.rb +8 -0
  185. data/spec/moneta/adapters/memcached/adapter_memcached_spec.rb +13 -0
  186. data/spec/moneta/adapters/memcached/dalli/adapter_memcached_dalli_spec.rb +25 -0
  187. data/spec/moneta/adapters/memcached/dalli/standard_memcached_dalli_spec.rb +11 -0
  188. data/spec/moneta/adapters/memcached/native/adapter_memcached_native_spec.rb +25 -0
  189. data/spec/moneta/adapters/memcached/native/standard_memcached_native_spec.rb +11 -0
  190. data/spec/moneta/adapters/memcached/standard_memcached_spec.rb +17 -0
  191. data/spec/moneta/adapters/memcached_helper.rb +20 -0
  192. data/spec/moneta/adapters/memory/adapter_memory_spec.rb +7 -0
  193. data/spec/moneta/adapters/memory/standard_memory_spec.rb +4 -0
  194. data/spec/moneta/adapters/memory/standard_memory_with_compress_spec.rb +9 -0
  195. data/spec/moneta/adapters/memory/standard_memory_with_expires_spec.rb +8 -0
  196. data/spec/moneta/adapters/memory/standard_memory_with_json_key_serializer_spec.rb +4 -0
  197. data/spec/moneta/adapters/memory/standard_memory_with_json_serializer_spec.rb +9 -0
  198. data/spec/moneta/adapters/memory/standard_memory_with_json_value_serializer_spec.rb +9 -0
  199. data/spec/moneta/adapters/memory/standard_memory_with_prefix_spec.rb +41 -0
  200. data/spec/moneta/adapters/memory/standard_memory_with_snappy_compress_spec.rb +9 -0
  201. data/spec/moneta/adapters/mongo/adapter_mongo_spec.rb +41 -0
  202. data/spec/moneta/adapters/mongo/adapter_mongo_with_default_expires_spec.rb +14 -0
  203. data/spec/moneta/adapters/mongo/standard_mongo_spec.rb +7 -0
  204. data/spec/moneta/adapters/null/null_adapter_spec.rb +7 -0
  205. data/spec/moneta/adapters/null/standard_null_spec.rb +4 -0
  206. data/spec/moneta/adapters/pstore/adapter_pstore_spec.rb +7 -0
  207. data/spec/moneta/adapters/pstore/standard_pstore_spec.rb +9 -0
  208. data/spec/moneta/adapters/pstore/standard_pstore_with_expires_spec.rb +13 -0
  209. data/spec/moneta/adapters/redis/adapter_redis_spec.rb +20 -0
  210. data/spec/moneta/adapters/redis/standard_redis_spec.rb +14 -0
  211. data/spec/moneta/adapters/restclient/adapter_restclient_spec.rb +13 -0
  212. data/spec/moneta/adapters/restclient/helper.rb +12 -0
  213. data/spec/moneta/adapters/restclient/standard_restclient_spec.rb +13 -0
  214. data/spec/moneta/adapters/riak/adapter_riak_spec.rb +14 -0
  215. data/spec/moneta/adapters/riak/standard_riak_spec.rb +10 -0
  216. data/spec/moneta/adapters/riak/standard_riak_with_expires_spec.rb +14 -0
  217. data/spec/moneta/adapters/sdbm/adapter_sdbm_spec.rb +7 -0
  218. data/spec/moneta/adapters/sdbm/standard_sdbm_spec.rb +7 -0
  219. data/spec/moneta/adapters/sdbm/standard_sdbm_with_expires_spec.rb +11 -0
  220. data/spec/moneta/adapters/sequel/adapter_sequel_spec.rb +133 -0
  221. data/spec/moneta/adapters/sequel/helper.rb +80 -0
  222. data/spec/moneta/adapters/sequel/standard_sequel_spec.rb +8 -0
  223. data/spec/moneta/adapters/sequel/standard_sequel_with_expires_spec.rb +18 -0
  224. data/spec/moneta/adapters/sqlite/adapter_sqlite_spec.rb +7 -0
  225. data/spec/moneta/adapters/sqlite/standard_sqlite_spec.rb +7 -0
  226. data/spec/moneta/adapters/sqlite/standard_sqlite_with_expires_spec.rb +15 -0
  227. data/spec/moneta/adapters/tdb/adapter_tdb_spec.rb +7 -0
  228. data/spec/moneta/adapters/tdb/standard_tdb_spec.rb +7 -0
  229. data/spec/moneta/adapters/tdb/standard_tdb_with_expires_spec.rb +11 -0
  230. data/spec/moneta/adapters/tokyocabinet/adapter_tokyocabinet_bdb_spec.rb +7 -0
  231. data/spec/moneta/adapters/tokyocabinet/adapter_tokyocabinet_hdb_spec.rb +7 -0
  232. data/spec/moneta/adapters/tokyocabinet/standard_tokyocabinet_spec.rb +7 -0
  233. data/spec/moneta/adapters/tokyocabinet/standard_tokyocabinet_with_expires_spec.rb +11 -0
  234. data/spec/moneta/adapters/tokyotyrant/adapter_tokyotyrant_spec.rb +11 -0
  235. data/spec/moneta/adapters/tokyotyrant/helper.rb +12 -0
  236. data/spec/moneta/adapters/tokyotyrant/standard_tokyotyrant_spec.rb +7 -0
  237. data/spec/moneta/adapters/tokyotyrant/standard_tokyotyrant_with_expires_spec.rb +12 -0
  238. data/spec/moneta/adapters/yaml/adapter_yaml_spec.rb +7 -0
  239. data/spec/moneta/adapters/yaml/standard_yaml_spec.rb +9 -0
  240. data/spec/moneta/adapters/yaml/standard_yaml_with_expires_spec.rb +13 -0
  241. data/spec/moneta/builder_spec.rb +22 -1
  242. data/spec/moneta/config_spec.rb +219 -0
  243. data/spec/moneta/mutex_spec.rb +6 -23
  244. data/spec/moneta/proxies/cache/cache_file_memory_spec.rb +24 -0
  245. data/spec/moneta/proxies/cache/cache_memory_null_spec.rb +12 -0
  246. data/spec/moneta/proxies/enumerable/enumerable_spec.rb +26 -0
  247. data/spec/moneta/proxies/expires/expires_file_spec.rb +28 -0
  248. data/spec/moneta/proxies/expires/expires_memory_spec.rb +15 -0
  249. data/spec/moneta/proxies/expires/expires_memory_with_default_expires_spec.rb +16 -0
  250. data/spec/moneta/proxies/fallback/fallback_spec.rb +42 -0
  251. data/spec/moneta/proxies/lock/lock_spec.rb +10 -0
  252. data/spec/moneta/{optionmerger_spec.rb → proxies/optionmerger/optionmerger_spec.rb} +2 -18
  253. data/spec/moneta/proxies/pool/pool_spec.rb +353 -0
  254. data/spec/moneta/proxies/proxy/proxy_expires_memory_spec.rb +16 -0
  255. data/spec/moneta/proxies/shared/shared_tcp_spec.rb +59 -0
  256. data/spec/moneta/proxies/shared/shared_unix_spec.rb +58 -0
  257. data/spec/moneta/proxies/transformer/transformer_bencode_spec.rb +19 -0
  258. data/spec/moneta/proxies/transformer/transformer_bert_spec.rb +19 -0
  259. data/spec/moneta/proxies/transformer/transformer_bson_spec.rb +25 -0
  260. data/spec/moneta/proxies/transformer/transformer_bzip2_spec.rb +19 -0
  261. data/spec/moneta/proxies/transformer/transformer_json_spec.rb +19 -0
  262. data/spec/moneta/proxies/transformer/transformer_key_inspect_spec.rb +17 -0
  263. data/spec/moneta/proxies/transformer/transformer_key_marshal_spec.rb +17 -0
  264. data/spec/moneta/proxies/transformer/transformer_key_to_s_spec.rb +17 -0
  265. data/spec/moneta/proxies/transformer/transformer_key_yaml_spec.rb +17 -0
  266. data/spec/moneta/proxies/transformer/transformer_lz4_spec.rb +19 -0
  267. data/spec/moneta/proxies/transformer/transformer_lzma_spec.rb +19 -0
  268. data/spec/moneta/proxies/transformer/transformer_lzo_spec.rb +19 -0
  269. data/spec/moneta/proxies/transformer/transformer_marshal_base64_spec.rb +20 -0
  270. data/spec/moneta/proxies/transformer/transformer_marshal_city128_spec.rb +15 -0
  271. data/spec/moneta/proxies/transformer/transformer_marshal_city32_spec.rb +15 -0
  272. data/spec/moneta/proxies/transformer/transformer_marshal_city64_spec.rb +15 -0
  273. data/spec/moneta/proxies/transformer/transformer_marshal_escape_spec.rb +21 -0
  274. data/spec/moneta/proxies/transformer/transformer_marshal_hex_spec.rb +20 -0
  275. data/spec/moneta/proxies/transformer/transformer_marshal_hmac_spec.rb +19 -0
  276. data/spec/moneta/proxies/transformer/transformer_marshal_md5_spec.rb +15 -0
  277. data/spec/moneta/proxies/transformer/transformer_marshal_md5_spread_spec.rb +15 -0
  278. data/spec/moneta/proxies/transformer/transformer_marshal_prefix_base64_spec.rb +33 -0
  279. data/spec/moneta/proxies/transformer/transformer_marshal_prefix_spec.rb +15 -0
  280. data/spec/moneta/proxies/transformer/transformer_marshal_qp_spec.rb +20 -0
  281. data/spec/moneta/proxies/transformer/transformer_marshal_rmd160_spec.rb +15 -0
  282. data/spec/moneta/proxies/transformer/transformer_marshal_sha1_spec.rb +15 -0
  283. data/spec/moneta/proxies/transformer/transformer_marshal_sha256_spec.rb +15 -0
  284. data/spec/moneta/proxies/transformer/transformer_marshal_sha384_spec.rb +15 -0
  285. data/spec/moneta/proxies/transformer/transformer_marshal_sha512_spec.rb +15 -0
  286. data/spec/moneta/proxies/transformer/transformer_marshal_spec.rb +19 -0
  287. data/spec/moneta/proxies/transformer/transformer_marshal_truncate_spec.rb +15 -0
  288. data/spec/moneta/proxies/transformer/transformer_marshal_urlsafe_base64_spec.rb +20 -0
  289. data/spec/moneta/proxies/transformer/transformer_marshal_uuencode_spec.rb +19 -0
  290. data/spec/moneta/proxies/transformer/transformer_msgpack_spec.rb +19 -0
  291. data/spec/moneta/proxies/transformer/transformer_ox_spec.rb +19 -0
  292. data/spec/moneta/proxies/transformer/transformer_php_spec.rb +19 -0
  293. data/spec/moneta/proxies/transformer/transformer_quicklz_spec.rb +19 -0
  294. data/spec/moneta/proxies/transformer/transformer_snappy_spec.rb +19 -0
  295. data/spec/moneta/proxies/transformer/transformer_tnet_spec.rb +19 -0
  296. data/spec/moneta/proxies/transformer/transformer_value_marshal_spec.rb +19 -0
  297. data/spec/moneta/proxies/transformer/transformer_value_yaml_spec.rb +19 -0
  298. data/spec/moneta/proxies/transformer/transformer_yaml_spec.rb +19 -0
  299. data/spec/moneta/proxies/transformer/transformer_zlib_spec.rb +19 -0
  300. data/spec/moneta/proxies/weak_create/weak_create_spec.rb +21 -0
  301. data/spec/moneta/proxies/weak_each_key/weak_each_key_spec.rb +22 -0
  302. data/spec/moneta/proxies/weak_increment/weak_increment_spec.rb +21 -0
  303. data/spec/moneta/semaphore_spec.rb +6 -23
  304. data/spec/moneta/stack_file_memory_spec.rb +5 -29
  305. data/spec/moneta/stack_memory_file_spec.rb +5 -31
  306. data/spec/rack/session_moneta_spec.rb +44 -25
  307. data/spec/restserver.rb +44 -0
  308. data/{spec → test}/action_dispatch/fixtures/session_autoload_test/foo.rb +0 -0
  309. data/{spec/action_dispatch/session_moneta_store_spec.rb → test/action_dispatch/session_moneta_store_test.rb} +13 -5
  310. metadata +545 -369
  311. data/.travis.yml +0 -55
  312. data/lib/moneta/adapters/mongo/base.rb +0 -78
  313. data/lib/moneta/adapters/mongo/moped.rb +0 -107
  314. data/lib/moneta/adapters/mongo/official.rb +0 -117
  315. data/lib/moneta/mixins.rb +0 -343
  316. data/script/generate-specs +0 -2696
  317. data/script/install-kyotocabinet +0 -17
  318. data/spec/moneta/adapter_activerecord_exisiting_connection_spec.rb +0 -36
  319. data/spec/moneta/adapter_activerecord_spec.rb +0 -65
  320. data/spec/moneta/adapter_cassandra_spec.rb +0 -33
  321. data/spec/moneta/adapter_cassandra_with_default_expires_spec.rb +0 -34
  322. data/spec/moneta/adapter_client_spec.rb +0 -35
  323. data/spec/moneta/adapter_cookie_spec.rb +0 -30
  324. data/spec/moneta/adapter_couch_spec.rb +0 -36
  325. data/spec/moneta/adapter_datamapper_spec.rb +0 -64
  326. data/spec/moneta/adapter_daybreak_spec.rb +0 -31
  327. data/spec/moneta/adapter_dbm_spec.rb +0 -31
  328. data/spec/moneta/adapter_file_spec.rb +0 -34
  329. data/spec/moneta/adapter_fog_spec.rb +0 -37
  330. data/spec/moneta/adapter_gdbm_spec.rb +0 -31
  331. data/spec/moneta/adapter_hbase_spec.rb +0 -33
  332. data/spec/moneta/adapter_kyotocabinet_spec.rb +0 -31
  333. data/spec/moneta/adapter_leveldb_spec.rb +0 -31
  334. data/spec/moneta/adapter_lmdb_spec.rb +0 -32
  335. data/spec/moneta/adapter_lmdb_with_db_spec.rb +0 -32
  336. data/spec/moneta/adapter_localmemcache_spec.rb +0 -32
  337. data/spec/moneta/adapter_memcached_dalli_spec.rb +0 -36
  338. data/spec/moneta/adapter_memcached_dalli_with_default_expires_spec.rb +0 -37
  339. data/spec/moneta/adapter_memcached_native_spec.rb +0 -36
  340. data/spec/moneta/adapter_memcached_native_with_default_expires_spec.rb +0 -37
  341. data/spec/moneta/adapter_memcached_spec.rb +0 -36
  342. data/spec/moneta/adapter_memcached_with_default_expires_spec.rb +0 -37
  343. data/spec/moneta/adapter_memory_spec.rb +0 -184
  344. data/spec/moneta/adapter_mongo_moped_spec.rb +0 -64
  345. data/spec/moneta/adapter_mongo_moped_with_default_expires_spec.rb +0 -53
  346. data/spec/moneta/adapter_mongo_official_spec.rb +0 -64
  347. data/spec/moneta/adapter_mongo_official_with_default_expires_spec.rb +0 -53
  348. data/spec/moneta/adapter_mongo_spec.rb +0 -51
  349. data/spec/moneta/adapter_mongo_with_default_expires_spec.rb +0 -53
  350. data/spec/moneta/adapter_pstore_spec.rb +0 -250
  351. data/spec/moneta/adapter_redis_spec.rb +0 -36
  352. data/spec/moneta/adapter_redis_with_default_expires_spec.rb +0 -37
  353. data/spec/moneta/adapter_restclient_spec.rb +0 -33
  354. data/spec/moneta/adapter_riak_spec.rb +0 -36
  355. data/spec/moneta/adapter_sdbm_spec.rb +0 -30
  356. data/spec/moneta/adapter_sequel_spec.rb +0 -34
  357. data/spec/moneta/adapter_sqlite_spec.rb +0 -32
  358. data/spec/moneta/adapter_tdb_spec.rb +0 -31
  359. data/spec/moneta/adapter_tokyocabinet_bdb_spec.rb +0 -31
  360. data/spec/moneta/adapter_tokyocabinet_hdb_spec.rb +0 -31
  361. data/spec/moneta/adapter_tokyotyrant_spec.rb +0 -34
  362. data/spec/moneta/adapter_yaml_spec.rb +0 -57
  363. data/spec/moneta/cache_file_memory_spec.rb +0 -50
  364. data/spec/moneta/cache_memory_null_spec.rb +0 -35
  365. data/spec/moneta/expires_file_spec.rb +0 -78
  366. data/spec/moneta/expires_memory_spec.rb +0 -189
  367. data/spec/moneta/expires_memory_with_default_expires_spec.rb +0 -190
  368. data/spec/moneta/lock_spec.rb +0 -187
  369. data/spec/moneta/null_adapter_spec.rb +0 -86
  370. data/spec/moneta/pool_spec.rb +0 -38
  371. data/spec/moneta/proxy_expires_memory_spec.rb +0 -191
  372. data/spec/moneta/proxy_redis_spec.rb +0 -40
  373. data/spec/moneta/shared_tcp_spec.rb +0 -46
  374. data/spec/moneta/shared_unix_spec.rb +0 -46
  375. data/spec/moneta/standard_activerecord_spec.rb +0 -253
  376. data/spec/moneta/standard_activerecord_with_expires_spec.rb +0 -255
  377. data/spec/moneta/standard_cassandra_spec.rb +0 -252
  378. data/spec/moneta/standard_client_tcp_spec.rb +0 -269
  379. data/spec/moneta/standard_client_unix_spec.rb +0 -254
  380. data/spec/moneta/standard_couch_spec.rb +0 -252
  381. data/spec/moneta/standard_couch_with_expires_spec.rb +0 -254
  382. data/spec/moneta/standard_datamapper_spec.rb +0 -254
  383. data/spec/moneta/standard_datamapper_with_expires_spec.rb +0 -256
  384. data/spec/moneta/standard_datamapper_with_repository_spec.rb +0 -254
  385. data/spec/moneta/standard_daybreak_spec.rb +0 -250
  386. data/spec/moneta/standard_daybreak_with_expires_spec.rb +0 -252
  387. data/spec/moneta/standard_dbm_spec.rb +0 -250
  388. data/spec/moneta/standard_dbm_with_expires_spec.rb +0 -252
  389. data/spec/moneta/standard_file_spec.rb +0 -253
  390. data/spec/moneta/standard_file_with_expires_spec.rb +0 -255
  391. data/spec/moneta/standard_fog_spec.rb +0 -256
  392. data/spec/moneta/standard_fog_with_expires_spec.rb +0 -258
  393. data/spec/moneta/standard_gdbm_spec.rb +0 -250
  394. data/spec/moneta/standard_gdbm_with_expires_spec.rb +0 -252
  395. data/spec/moneta/standard_hashfile_spec.rb +0 -253
  396. data/spec/moneta/standard_hashfile_with_expires_spec.rb +0 -255
  397. data/spec/moneta/standard_hbase_spec.rb +0 -252
  398. data/spec/moneta/standard_hbase_with_expires_spec.rb +0 -255
  399. data/spec/moneta/standard_kyotocabinet_spec.rb +0 -250
  400. data/spec/moneta/standard_kyotocabinet_with_expires_spec.rb +0 -252
  401. data/spec/moneta/standard_leveldb_spec.rb +0 -250
  402. data/spec/moneta/standard_leveldb_with_expires_spec.rb +0 -252
  403. data/spec/moneta/standard_lmdb_spec.rb +0 -251
  404. data/spec/moneta/standard_lmdb_with_expires_spec.rb +0 -253
  405. data/spec/moneta/standard_localmemcache_spec.rb +0 -251
  406. data/spec/moneta/standard_localmemcache_with_expires_spec.rb +0 -252
  407. data/spec/moneta/standard_lruhash_spec.rb +0 -187
  408. data/spec/moneta/standard_lruhash_with_expires_spec.rb +0 -189
  409. data/spec/moneta/standard_memcached_dalli_spec.rb +0 -255
  410. data/spec/moneta/standard_memcached_native_spec.rb +0 -255
  411. data/spec/moneta/standard_memcached_spec.rb +0 -255
  412. data/spec/moneta/standard_memory_spec.rb +0 -187
  413. data/spec/moneta/standard_memory_with_compress_spec.rb +0 -187
  414. data/spec/moneta/standard_memory_with_expires_spec.rb +0 -189
  415. data/spec/moneta/standard_memory_with_json_key_serializer_spec.rb +0 -86
  416. data/spec/moneta/standard_memory_with_json_serializer_spec.rb +0 -49
  417. data/spec/moneta/standard_memory_with_json_value_serializer_spec.rb +0 -90
  418. data/spec/moneta/standard_memory_with_prefix_spec.rb +0 -187
  419. data/spec/moneta/standard_memory_with_snappy_compress_spec.rb +0 -187
  420. data/spec/moneta/standard_mongo_moped_spec.rb +0 -255
  421. data/spec/moneta/standard_mongo_official_spec.rb +0 -255
  422. data/spec/moneta/standard_mongo_spec.rb +0 -255
  423. data/spec/moneta/standard_null_spec.rb +0 -120
  424. data/spec/moneta/standard_pstore_spec.rb +0 -253
  425. data/spec/moneta/standard_pstore_with_expires_spec.rb +0 -255
  426. data/spec/moneta/standard_redis_spec.rb +0 -255
  427. data/spec/moneta/standard_restclient_spec.rb +0 -252
  428. data/spec/moneta/standard_riak_spec.rb +0 -255
  429. data/spec/moneta/standard_riak_with_expires_spec.rb +0 -256
  430. data/spec/moneta/standard_sdbm_spec.rb +0 -249
  431. data/spec/moneta/standard_sdbm_with_expires_spec.rb +0 -251
  432. data/spec/moneta/standard_sequel_spec.rb +0 -253
  433. data/spec/moneta/standard_sequel_with_expires_spec.rb +0 -255
  434. data/spec/moneta/standard_sqlite_spec.rb +0 -251
  435. data/spec/moneta/standard_sqlite_with_expires_spec.rb +0 -253
  436. data/spec/moneta/standard_tdb_spec.rb +0 -250
  437. data/spec/moneta/standard_tdb_with_expires_spec.rb +0 -252
  438. data/spec/moneta/standard_tokyocabinet_spec.rb +0 -250
  439. data/spec/moneta/standard_tokyocabinet_with_expires_spec.rb +0 -252
  440. data/spec/moneta/standard_tokyotyrant_spec.rb +0 -253
  441. data/spec/moneta/standard_tokyotyrant_with_expires_spec.rb +0 -255
  442. data/spec/moneta/standard_yaml_spec.rb +0 -250
  443. data/spec/moneta/standard_yaml_with_expires_spec.rb +0 -252
  444. data/spec/moneta/transformer_bencode_spec.rb +0 -55
  445. data/spec/moneta/transformer_bert_spec.rb +0 -55
  446. data/spec/moneta/transformer_bson_spec.rb +0 -60
  447. data/spec/moneta/transformer_bzip2_spec.rb +0 -55
  448. data/spec/moneta/transformer_json_spec.rb +0 -55
  449. data/spec/moneta/transformer_key_inspect_spec.rb +0 -91
  450. data/spec/moneta/transformer_key_marshal_spec.rb +0 -191
  451. data/spec/moneta/transformer_key_to_s_spec.rb +0 -91
  452. data/spec/moneta/transformer_key_yaml_spec.rb +0 -191
  453. data/spec/moneta/transformer_lz4_spec.rb +0 -55
  454. data/spec/moneta/transformer_lzma_spec.rb +0 -55
  455. data/spec/moneta/transformer_lzo_spec.rb +0 -55
  456. data/spec/moneta/transformer_marshal_base64_spec.rb +0 -194
  457. data/spec/moneta/transformer_marshal_city128_spec.rb +0 -194
  458. data/spec/moneta/transformer_marshal_city32_spec.rb +0 -194
  459. data/spec/moneta/transformer_marshal_city64_spec.rb +0 -194
  460. data/spec/moneta/transformer_marshal_escape_spec.rb +0 -194
  461. data/spec/moneta/transformer_marshal_hex_spec.rb +0 -194
  462. data/spec/moneta/transformer_marshal_hmac_spec.rb +0 -194
  463. data/spec/moneta/transformer_marshal_md5_spec.rb +0 -194
  464. data/spec/moneta/transformer_marshal_md5_spread_spec.rb +0 -194
  465. data/spec/moneta/transformer_marshal_prefix_spec.rb +0 -194
  466. data/spec/moneta/transformer_marshal_qp_spec.rb +0 -194
  467. data/spec/moneta/transformer_marshal_rmd160_spec.rb +0 -194
  468. data/spec/moneta/transformer_marshal_sha1_spec.rb +0 -194
  469. data/spec/moneta/transformer_marshal_sha256_spec.rb +0 -194
  470. data/spec/moneta/transformer_marshal_sha384_spec.rb +0 -194
  471. data/spec/moneta/transformer_marshal_sha512_spec.rb +0 -194
  472. data/spec/moneta/transformer_marshal_spec.rb +0 -191
  473. data/spec/moneta/transformer_marshal_truncate_spec.rb +0 -194
  474. data/spec/moneta/transformer_marshal_uuencode_spec.rb +0 -194
  475. data/spec/moneta/transformer_msgpack_spec.rb +0 -55
  476. data/spec/moneta/transformer_ox_spec.rb +0 -191
  477. data/spec/moneta/transformer_php_spec.rb +0 -55
  478. data/spec/moneta/transformer_quicklz_spec.rb +0 -55
  479. data/spec/moneta/transformer_snappy_spec.rb +0 -55
  480. data/spec/moneta/transformer_tnet_spec.rb +0 -55
  481. data/spec/moneta/transformer_value_marshal_spec.rb +0 -191
  482. data/spec/moneta/transformer_value_yaml_spec.rb +0 -191
  483. data/spec/moneta/transformer_yaml_spec.rb +0 -191
  484. data/spec/moneta/transformer_zlib_spec.rb +0 -55
  485. data/spec/moneta/weak_create_spec.rb +0 -41
  486. data/spec/moneta/weak_increment_spec.rb +0 -41
  487. data/spec/monetaspecs.rb +0 -51267
  488. data/spec/quality_spec.rb +0 -51
@@ -4,61 +4,89 @@ module Moneta
4
4
  module Adapters
5
5
  # Sequel backend
6
6
  # @api public
7
- class Sequel
8
- include Defaults
7
+ class Sequel < Adapter
8
+ autoload :MySQL, 'moneta/adapters/sequel/mysql'
9
+ autoload :Postgres, 'moneta/adapters/sequel/postgres'
10
+ autoload :PostgresHStore, 'moneta/adapters/sequel/postgres_hstore'
11
+ autoload :SQLite, 'moneta/adapters/sequel/sqlite'
9
12
 
10
- # Sequel::UniqueConstraintViolation is defined since sequel 3.44.0
11
- # older versions raise a Sequel::DatabaseError.
12
- UniqueConstraintViolation = defined?(::Sequel::UniqueConstraintViolation) ? ::Sequel::UniqueConstraintViolation : ::Sequel::DatabaseError
13
+ supports :create, :increment, :each_key
13
14
 
14
- supports :create, :increment
15
- attr_reader :backend
15
+ config :table, default: :moneta, coerce: :to_sym
16
+ config :optimize, default: true
17
+ config :create_table, default: true
18
+ config :key_column, default: :k
19
+ config :value_column, default: :v
20
+ config :hstore, coerce: :to_s
21
+ config :each_key_server
22
+
23
+ backend do |db:, extensions: [], connection_validation_timeout: nil, **options|
24
+ ::Sequel.connect(db, options).tap do |backend|
25
+ extensions.map(&:to_sym).each(&backend.method(:extension))
26
+
27
+ if connection_validation_timeout
28
+ backend.pool.connection_validation_timeout = connection_validation_timeout
29
+ end
30
+ end
31
+ end
16
32
 
17
33
  # @param [Hash] options
18
34
  # @option options [String] :db Sequel database
19
- # @option options [String/Symbol] :table (:moneta) Table name
35
+ # @option options [String, Symbol] :table (:moneta) Table name
20
36
  # @option options [Array] :extensions ([]) List of Sequel extensions
21
37
  # @option options [Integer] :connection_validation_timeout (nil) Sequel connection_validation_timeout
38
+ # @option options [Sequel::Database] :backend Use existing backend instance
39
+ # @option options [Boolean] :optimize (true) Set to false to prevent database-specific optimisations
40
+ # @option options [Proc, Boolean] :create_table (true) Provide a Proc for creating the table, or
41
+ # set to false to disable table creation all together. If a Proc is given, it will be
42
+ # called regardless of whether the table exists already.
43
+ # @option options [Symbol] :key_column (:k) The name of the key column
44
+ # @option options [Symbol] :value_column (:v) The name of the value column
45
+ # @option options [String] :hstore If using Postgres, keys and values are stored in a single
46
+ # row of the table in the value_column using the hstore format. The row to use is
47
+ # the one where the value_column is equal to the value of this option, and will be created
48
+ # if it doesn't exist.
49
+ # @option options [Symbol] :each_key_server Some adapters are unable to do
50
+ # multiple operations with a single connection. For these, it is
51
+ # possible to specify a separate connection to use for `#each_key`. Use
52
+ # in conjunction with Sequel's `:servers` option
22
53
  # @option options All other options passed to `Sequel#connect`
23
- # @option options [Sequel connection] :backend Use existing backend instance
24
54
  def initialize(options = {})
25
- table = (options.delete(:table) || :moneta).to_sym
26
- extensions = options.delete(:extensions) || []
27
- raise ArgumentError, 'Option :extensions must be an Array' unless extensions.is_a?(Array)
28
- connection_validation_timeout = options.delete(:connection_validation_timeout)
29
- @backend = options[:backend] ||
30
- begin
31
- raise ArgumentError, 'Option :db is required' unless db = options.delete(:db)
32
- ::Sequel.connect(db, options)
33
- end
34
- extensions.each do |extension|
35
- @backend.extension(extension.to_sym)
55
+ super
56
+
57
+ if config.hstore
58
+ extend Sequel::PostgresHStore
59
+ elsif config.optimize
60
+ add_optimizations
36
61
  end
37
- @backend.pool.connection_validation_timeout = connection_validation_timeout if connection_validation_timeout
38
- @backend.create_table?(table) do
39
- String :k, null: false, primary_key: true
40
- File :v
62
+
63
+ if config.create_table.respond_to?(:call)
64
+ config.create_table.call(backend)
65
+ elsif config.create_table
66
+ create_table
41
67
  end
42
- @table = @backend[table]
68
+
69
+ @table = backend[config.table]
70
+ prepare_statements
43
71
  end
44
72
 
45
73
  # (see Proxy#key?)
46
74
  def key?(key, options = {})
47
- @table[k: key] != nil
75
+ @key.call(key: key) != nil
48
76
  end
49
77
 
50
78
  # (see Proxy#load)
51
79
  def load(key, options = {})
52
- record = @table[k: key]
53
- record && record[:v]
80
+ if row = @load.call(key: key)
81
+ row[config.value_column]
82
+ end
54
83
  end
55
84
 
56
85
  # (see Proxy#store)
57
86
  def store(key, value, options = {})
58
- begin
59
- @table.insert(k: key, v: blob(value))
60
- rescue UniqueConstraintViolation
61
- @table.where(k: key).update(v: blob(value))
87
+ blob_value = blob(value)
88
+ unless @store_update.call(key: key, value: blob_value) == 1
89
+ @create.call(key: key, value: blob_value)
62
90
  end
63
91
  value
64
92
  rescue ::Sequel::DatabaseError
@@ -66,26 +94,29 @@ module Moneta
66
94
  (tries += 1) < 10 ? retry : raise
67
95
  end
68
96
 
69
- # (see Proxy#store)
97
+ # (see Proxy#create)
70
98
  def create(key, value, options = {})
71
- @table.insert(k: key, v: blob(value))
99
+ @create.call(key: key, value: blob(value))
72
100
  true
73
- rescue UniqueConstraintViolation
101
+ rescue ::Sequel::ConstraintViolation
74
102
  false
75
103
  end
76
104
 
77
105
  # (see Proxy#increment)
78
106
  def increment(key, amount = 1, options = {})
79
- @backend.transaction do
80
- locked_table = @table.for_update
81
- if record = locked_table[k: key]
82
- value = Utils.to_int(record[:v]) + amount
83
- locked_table.where(k: key).update(v: blob(value.to_s))
84
- value
107
+ backend.transaction do
108
+ if existing = @load_for_update.call(key: key)
109
+ existing_value = existing[config.value_column]
110
+ amount += Integer(existing_value)
111
+ raise IncrementError, "no update" unless @increment_update.call(
112
+ key: key,
113
+ value: existing_value,
114
+ new_value: blob(amount.to_s)
115
+ ) == 1
85
116
  else
86
- locked_table.insert(k: key, v: blob(amount.to_s))
87
- amount
117
+ @create.call(key: key, value: blob(amount.to_s))
88
118
  end
119
+ amount
89
120
  end
90
121
  rescue ::Sequel::DatabaseError
91
122
  # Concurrent modification might throw a bunch of different errors
@@ -95,10 +126,9 @@ module Moneta
95
126
 
96
127
  # (see Proxy#delete)
97
128
  def delete(key, options = {})
98
- if value = load(key, options)
99
- @table.filter(k: key).delete
100
- value
101
- end
129
+ value = load(key, options)
130
+ @delete.call(key: key)
131
+ value
102
132
  end
103
133
 
104
134
  # (see Proxy#clear)
@@ -109,16 +139,187 @@ module Moneta
109
139
 
110
140
  # (see Proxy#close)
111
141
  def close
112
- @backend.disconnect
142
+ backend.disconnect
113
143
  nil
114
144
  end
115
145
 
116
- private
146
+ # (see Proxy#slice)
147
+ def slice(*keys, **options)
148
+ @slice.all(keys).map! { |row| [row[config.key_column], row[config.value_column]] }
149
+ end
150
+
151
+ # (see Proxy#values_at)
152
+ def values_at(*keys, **options)
153
+ pairs = Hash[slice(*keys, **options)]
154
+ keys.map { |key| pairs[key] }
155
+ end
156
+
157
+ # (see Proxy#fetch_values)
158
+ def fetch_values(*keys, **options)
159
+ return values_at(*keys, **options) unless block_given?
160
+ existing = Hash[slice(*keys, **options)]
161
+ keys.map do |key|
162
+ if existing.key? key
163
+ existing[key]
164
+ else
165
+ yield key
166
+ end
167
+ end
168
+ end
169
+
170
+ # (see Proxy#merge!)
171
+ def merge!(pairs, options = {})
172
+ backend.transaction do
173
+ existing = Hash[slice_for_update(pairs)]
174
+ update_pairs, insert_pairs = pairs.partition { |k, _| existing.key?(k) }
175
+ @table.import([config.key_column, config.value_column], blob_pairs(insert_pairs))
176
+
177
+ if block_given?
178
+ update_pairs.map! do |key, new_value|
179
+ [key, yield(key, existing[key], new_value)]
180
+ end
181
+ end
182
+
183
+ update_pairs.each do |key, value|
184
+ @store_update.call(key: key, value: blob(value))
185
+ end
186
+ end
187
+
188
+ self
189
+ end
190
+
191
+ # (see Proxy#each_key)
192
+ def each_key
193
+ return enum_for(:each_key) { @table.count } unless block_given?
194
+
195
+ key_column = config.key_column
196
+ if config.each_key_server
197
+ @table.server(config.each_key_server).order(key_column).select(key_column).paged_each do |row|
198
+ yield row[key_column]
199
+ end
200
+ else
201
+ @table.select(key_column).order(key_column).paged_each(stream: false) do |row|
202
+ yield row[key_column]
203
+ end
204
+ end
205
+ self
206
+ end
207
+
208
+ protected
117
209
 
118
- # See https://github.com/jeremyevans/sequel/issues/715
119
- def blob(s)
120
- s.empty? ? '' : ::Sequel.blob(s)
210
+ # @api private
211
+ def add_optimizations
212
+ case backend.database_type
213
+ when :mysql
214
+ extend Sequel::MySQL
215
+ when :postgres
216
+ if matches = backend.get(::Sequel[:version].function).match(/PostgreSQL (\d+)\.(\d+)/)
217
+ # Our optimisations only work on Postgres 9.5+
218
+ major, minor = matches[1..2].map(&:to_i)
219
+ extend Sequel::Postgres if major > 9 || (major == 9 && minor >= 5)
220
+ end
221
+ when :sqlite
222
+ extend Sequel::SQLite
223
+ end
224
+ end
225
+
226
+ def blob(str)
227
+ ::Sequel.blob(str) unless str == nil
228
+ end
229
+
230
+ def blob_pairs(pairs)
231
+ pairs.map do |key, value|
232
+ [key, blob(value)]
233
+ end
234
+ end
235
+
236
+ def create_table
237
+ key_column = config.key_column
238
+ value_column = config.value_column
239
+ backend.create_table?(config.table) do
240
+ String key_column, null: false, primary_key: true
241
+ File value_column
242
+ end
243
+ end
244
+
245
+ def slice_for_update(pairs)
246
+ @slice_for_update.all(pairs.map { |k, _| k }.to_a).map! do |row|
247
+ [row[config.key_column], row[config.value_column]]
248
+ end
249
+ end
250
+
251
+ def yield_merge_pairs(pairs)
252
+ existing = Hash[slice_for_update(pairs)]
253
+ pairs.map do |key, new_value|
254
+ new_value = yield(key, existing[key], new_value) if existing.key?(key)
255
+ [key, new_value]
256
+ end
257
+ end
258
+
259
+ def statement_id(id)
260
+ "moneta_#{config.table}_#{id}".to_sym
261
+ end
262
+
263
+ def prepare_statements
264
+ prepare_key
265
+ prepare_load
266
+ prepare_store
267
+ prepare_create
268
+ prepare_increment
269
+ prepare_delete
270
+ prepare_slice
271
+ end
272
+
273
+ def prepare_key
274
+ @key = @table
275
+ .where(config.key_column => :$key).select(1)
276
+ .prepare(:first, statement_id(:key))
277
+ end
278
+
279
+ def prepare_load
280
+ @load = @table
281
+ .where(config.key_column => :$key).select(config.value_column)
282
+ .prepare(:first, statement_id(:load))
121
283
  end
284
+
285
+ def prepare_store
286
+ @store_update = @table
287
+ .where(config.key_column => :$key)
288
+ .prepare(:update, statement_id(:store_update), config.value_column => :$value)
289
+ end
290
+
291
+ def prepare_create
292
+ @create = @table
293
+ .prepare(:insert, statement_id(:create), config.key_column => :$key, config.value_column => :$value)
294
+ end
295
+
296
+ def prepare_increment
297
+ @load_for_update = @table
298
+ .where(config.key_column => :$key).for_update
299
+ .select(config.value_column)
300
+ .prepare(:first, statement_id(:load_for_update))
301
+ @increment_update ||= @table
302
+ .where(config.key_column => :$key, config.value_column => :$value)
303
+ .prepare(:update, statement_id(:increment_update), config.value_column => :$new_value)
304
+ end
305
+
306
+ def prepare_delete
307
+ @delete = @table.where(config.key_column => :$key)
308
+ .prepare(:delete, statement_id(:delete))
309
+ end
310
+
311
+ def prepare_slice
312
+ @slice_for_update = ::Sequel::Dataset::PlaceholderLiteralizer.loader(@table) do |pl, ds|
313
+ ds.filter(config.key_column => pl.arg).select(config.key_column, config.value_column).for_update
314
+ end
315
+
316
+ @slice = ::Sequel::Dataset::PlaceholderLiteralizer.loader(@table) do |pl, ds|
317
+ ds.filter(config.key_column => pl.arg).select(config.key_column, config.value_column)
318
+ end
319
+ end
320
+
321
+ # @api private
322
+ class IncrementError < ::Sequel::DatabaseError; end
122
323
  end
123
324
  end
124
325
  end
@@ -4,34 +4,52 @@ module Moneta
4
4
  module Adapters
5
5
  # Sqlite3 backend
6
6
  # @api public
7
- class Sqlite
8
- include Defaults
7
+ class Sqlite < Adapter
9
8
  include IncrementSupport
10
9
 
11
- supports :create
12
- attr_reader :backend
10
+ supports :create, :each_key
11
+
12
+ config :table, default: 'moneta'
13
+ config :busy_timeout, default: 1000
14
+ config :journal_mode
15
+
16
+ backend { |file:| ::SQLite3::Database.new(file) }
13
17
 
14
18
  # @param [Hash] options
15
19
  # @option options [String] :file Database file
16
20
  # @option options [String] :table ('moneta') Table name
17
- # @option options [Fixnum] :busy_timeout (1000) Sqlite timeout if database is busy
21
+ # @option options [Integer] :busy_timeout (1000) Sqlite timeout if database is busy
18
22
  # @option options [::Sqlite3::Database] :backend Use existing backend instance
23
+ # @option options [String, Symbol] :journal_mode Set the journal mode for the connection
19
24
  def initialize(options = {})
20
- table = options[:table] || 'moneta'
21
- @backend = options[:backend] ||
22
- begin
23
- raise ArgumentError, 'Option :file is required' unless options[:file]
24
- ::SQLite3::Database.new(options[:file])
25
- end
26
- @backend.busy_timeout(options[:busy_timeout] || 1000)
27
- @backend.execute("create table if not exists #{table} (k blob not null primary key, v blob)")
25
+ super
26
+
27
+ backend.busy_timeout(config.busy_timeout)
28
+ backend.execute("create table if not exists #{config.table} (k blob not null primary key, v blob)")
29
+
30
+ if journal_mode = config.journal_mode
31
+ backend.journal_mode = journal_mode.to_s
32
+ end
33
+
28
34
  @stmts =
29
- [@exists = @backend.prepare("select exists(select 1 from #{table} where k = ?)"),
30
- @select = @backend.prepare("select v from #{table} where k = ?"),
31
- @replace = @backend.prepare("replace into #{table} values (?, ?)"),
32
- @delete = @backend.prepare("delete from #{table} where k = ?"),
33
- @clear = @backend.prepare("delete from #{table}"),
34
- @create = @backend.prepare("insert into #{table} values (?, ?)")]
35
+ [@exists = backend.prepare("select exists(select 1 from #{config.table} where k = ?)"),
36
+ @select = backend.prepare("select v from #{config.table} where k = ?"),
37
+ @replace = backend.prepare("replace into #{config.table} values (?, ?)"),
38
+ @delete = backend.prepare("delete from #{config.table} where k = ?"),
39
+ @clear = backend.prepare("delete from #{config.table}"),
40
+ @create = backend.prepare("insert into #{config.table} values (?, ?)"),
41
+ @keys = backend.prepare("select k from #{config.table}"),
42
+ @count = backend.prepare("select count(*) from #{config.table}")]
43
+
44
+ version = backend.execute("select sqlite_version()").first.first
45
+ if @can_upsert = ::Gem::Version.new(version) >= ::Gem::Version.new('3.24.0')
46
+ @stmts << (@increment = backend.prepare <<-SQL)
47
+ insert into #{config.table} values (?, ?)
48
+ on conflict (k)
49
+ do update set v = cast(cast(v as integer) + ? as blob)
50
+ where v = '0' or v = X'30' or cast(v as integer) != 0
51
+ SQL
52
+ end
35
53
  end
36
54
 
37
55
  # (see Proxy#key?)
@@ -60,7 +78,11 @@ module Moneta
60
78
 
61
79
  # (see Proxy#increment)
62
80
  def increment(key, amount = 1, options = {})
63
- @backend.transaction(:exclusive) { return super }
81
+ backend.transaction(:exclusive) { return super } unless @can_upsert
82
+ backend.transaction do
83
+ @increment.execute!(key, amount.to_s, amount)
84
+ return Integer(load(key))
85
+ end
64
86
  end
65
87
 
66
88
  # (see Proxy#clear)
@@ -71,7 +93,7 @@ module Moneta
71
93
 
72
94
  # (see Default#create)
73
95
  def create(key, value, options = {})
74
- @create.execute!(key,value)
96
+ @create.execute!(key, value)
75
97
  true
76
98
  rescue SQLite3::ConstraintException
77
99
  # If you know a better way to detect whether an insert-ignore
@@ -82,10 +104,68 @@ module Moneta
82
104
 
83
105
  # (see Proxy#close)
84
106
  def close
85
- @stmts.each {|s| s.close }
86
- @backend.close
107
+ @stmts.each { |s| s.close }
108
+ backend.close
87
109
  nil
88
110
  end
111
+
112
+ # (see Proxy#slice)
113
+ def slice(*keys, **options)
114
+ query = "select k, v from #{config.table} where k in (#{(['?'] * keys.length).join(',')})"
115
+ backend.execute(query, keys)
116
+ end
117
+
118
+ # (see Proxy#values_at)
119
+ def values_at(*keys, **options)
120
+ hash = Hash[slice(*keys, **options)]
121
+ keys.map { |key| hash[key] }
122
+ end
123
+
124
+ # (see Proxy#fetch_values)
125
+ def fetch_values(*keys, **options)
126
+ return values_at(*keys, **options) unless block_given?
127
+ hash = Hash[slice(*keys, **options)]
128
+ keys.map do |key|
129
+ if hash.key?(key)
130
+ hash[key]
131
+ else
132
+ yield key
133
+ end
134
+ end
135
+ end
136
+
137
+ # (see Proxy#merge!)
138
+ def merge!(pairs, options = {})
139
+ transaction = backend.transaction if block_given?
140
+
141
+ if block_given?
142
+ existing = Hash[slice(*pairs.map { |k, _| k }.to_a)]
143
+ pairs = pairs.map do |key, new_value|
144
+ new_value = yield(key, existing[key], new_value) if existing.key?(key)
145
+ [key, new_value]
146
+ end.to_a
147
+ else
148
+ pairs = pairs.to_a
149
+ end
150
+
151
+ query = "replace into #{config.table} (k, v) values" + (['(?, ?)'] * pairs.length).join(',')
152
+ backend.query(query, pairs.flatten).close
153
+ rescue
154
+ backend.rollback if transaction
155
+ raise
156
+ else
157
+ backend.commit if transaction
158
+ self
159
+ end
160
+
161
+ # (see Proxy#each_key)
162
+ def each_key
163
+ return enum_for(:each_key) { @count.execute!.first.first } unless block_given?
164
+ @keys.execute!.each do |row|
165
+ yield row.first
166
+ end
167
+ self
168
+ end
89
169
  end
90
170
  end
91
171
  end
@@ -4,23 +4,32 @@ module Moneta
4
4
  module Adapters
5
5
  # TDB backend
6
6
  # @api public
7
- class TDB < Memory
8
- # @param [Hash] options
9
- # @option options [String] :file Database file
10
- # @option options [::TDB] :backend Use existing backend instance
11
- def initialize(options)
12
- @backend = options[:backend] ||
13
- begin
14
- raise ArgumentError, 'Option :file is required' unless file = options.delete(:file)
15
- ::TDB.new(file, options)
16
- end
17
- end
7
+ class TDB < Adapter
8
+ include HashAdapter
9
+ include IncrementSupport
10
+ include EachKeySupport
11
+
12
+ supports :create
13
+
14
+ # @!method initialize(options = {})
15
+ # @param [Hash] options
16
+ # @option options [String] :file Database file
17
+ # @option options [::TDB] :backend Use existing backend instance
18
+ backend { |file:, **options| ::TDB.new(file, options) }
18
19
 
19
20
  # (see Proxy#close)
20
21
  def close
21
- @backend.close
22
+ backend.close
22
23
  nil
23
24
  end
25
+
26
+ # (see Proxy#create)
27
+ def create(key, value, options = {})
28
+ backend.insert!(key, value)
29
+ true
30
+ rescue ::TDB::ERR::EXISTS
31
+ false
32
+ end
24
33
  end
25
34
  end
26
35
  end
@@ -4,23 +4,31 @@ module Moneta
4
4
  module Adapters
5
5
  # TokyoCabinet backend
6
6
  # @api public
7
- class TokyoCabinet < Memory
8
- # @param [Hash] options
9
- # @option options [String] :file Database file
10
- # @option options [Symbol] :type (:hdb) Database type (:bdb and :hdb possible)
11
- # @option options [::TokyoCabinet::*DB] :backend Use existing backend instance
12
- def initialize(options = {})
13
- if options[:backend]
14
- @backend = options[:backend]
7
+ class TokyoCabinet < Adapter
8
+ include HashAdapter
9
+ include IncrementSupport
10
+ include CreateSupport
11
+ include EachKeySupport
12
+
13
+ # @!method initialize(options = {})
14
+ # @param [Hash] options
15
+ # @option options [String] :file Database file
16
+ # @option options [Symbol] :type (:hdb) Database type (:bdb and :hdb possible)
17
+ # @option options [::TokyoCabinet::*DB] :backend Use existing backend instance
18
+ backend do |file:, type: :hdb|
19
+ case type
20
+ when :bdb
21
+ ::TokyoCabinet::BDB.new.tap do |backend|
22
+ backend.open(file, ::TokyoCabinet::BDB::OWRITER | ::TokyoCabinet::BDB::OCREAT) or
23
+ raise backend.errmsg(backend.ecode)
24
+ end
25
+ when :hdb
26
+ ::TokyoCabinet::HDB.new.tap do |backend|
27
+ backend.open(file, ::TokyoCabinet::HDB::OWRITER | ::TokyoCabinet::HDB::OCREAT) or
28
+ raise backend.errmsg(backend.ecode)
29
+ end
15
30
  else
16
- raise ArgumentError, 'Option :file is required' unless options[:file]
17
- if options[:type] == :bdb
18
- @backend = ::TokyoCabinet::BDB.new
19
- @backend.open(options[:file], ::TokyoCabinet::BDB::OWRITER | ::TokyoCabinet::BDB::OCREAT)
20
- else
21
- @backend = ::TokyoCabinet::HDB.new
22
- @backend.open(options[:file], ::TokyoCabinet::HDB::OWRITER | ::TokyoCabinet::HDB::OCREAT)
23
- end or raise @backend.errmsg(@backend.ecode)
31
+ raise ArgumentError, ":type must be :bdb or :hdb"
24
32
  end
25
33
  end
26
34