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
@@ -8,127 +8,431 @@ module Moneta
8
8
  # You can store hashes directly using this adapter.
9
9
  #
10
10
  # @example Store hashes
11
- # db = Moneta::Adapters::Mongo.new
11
+ # db = Moneta::Adapters::Couch.new
12
12
  # db['key'] = {a: 1, b: 2}
13
13
  #
14
14
  # @api public
15
- class Couch
16
- include Defaults
15
+ class Couch < Adapter
16
+ # @api private
17
+ class HTTPError < StandardError
18
+ attr_reader :status, :request_method, :key
17
19
 
18
- attr_reader :backend
20
+ def initialize(status, request_method, key)
21
+ @status = status
22
+ @request_method = request_method.to_sym
23
+ @key = key
19
24
 
20
- supports :create
25
+ super "HTTP Error: #{@status} (#{@request_method.to_s.upcase} #{@key})"
26
+ end
27
+ end
28
+
29
+ supports :create, :each_key
30
+
31
+ config :value_field, default: 'value'
32
+ config :type_field, default: 'type'
33
+ config :login
34
+ config :password
35
+ config :adapter
36
+ config :skip_create_db
37
+
38
+ backend do |scheme: 'http', host: '127.0.0.1', port: 5984, db: 'moneta', adapter: nil, **options|
39
+ ::Faraday.new "#{scheme}://#{host}:#{port}/#{db}", options do |faraday|
40
+ faraday.adapter adapter if adapter
41
+ end
42
+ end
21
43
 
22
44
  # @param [Hash] options
23
45
  # @option options [String] :host ('127.0.0.1') Couch host
24
46
  # @option options [String] :port (5984) Couch port
25
47
  # @option options [String] :db ('moneta') Couch database
48
+ # @option options [String] :scheme ('http') HTTP scheme to use
26
49
  # @option options [String] :value_field ('value') Document field to store value
27
50
  # @option options [String] :type_field ('type') Document field to store value type
28
- # @option options [Faraday connection] :backend Use existing backend instance
51
+ # @option options [String] :login Login name to use for HTTP basic authentication
52
+ # @option options [String] :password Password to use for HTTP basic authentication
53
+ # @option options [Symbol] :adapter Adapter to use with Faraday
54
+ # @option options [Faraday::Connecton] :backend Use existing backend instance
55
+ # @option options Other options passed to {Faraday::new} (unless
56
+ # :backend option is provided).
29
57
  def initialize(options = {})
30
- @value_field = options[:value_field] || 'value'
31
- @type_field = options[:type_field] || 'type'
32
- url = "http://#{options[:host] || '127.0.0.1'}:#{options[:port] || 5984}/#{options[:db] || 'moneta'}"
33
- @backend = options[:backend] || ::Faraday.new(url: url)
34
- create_db
58
+ super
59
+
60
+ if config.login && config.password
61
+ # Faraday 1.x had a `basic_auth` function
62
+ if backend.respond_to? :basic_auth
63
+ backend.basic_auth(config.login, config.password)
64
+ else
65
+ backend.request :authorization, :basic, config.login, config.password
66
+ end
67
+ end
68
+
69
+ @rev_cache = Moneta.build do
70
+ use :Lock
71
+ adapter :LRUHash
72
+ end
73
+ create_db unless config.skip_create_db
35
74
  end
36
75
 
37
76
  # (see Proxy#key?)
77
+ # @option options [Boolean] :cache_rev (true) Whether to cache the rev of
78
+ # the document for faster updating
38
79
  def key?(key, options = {})
39
- @backend.head(key).status == 200
80
+ cache_rev = options[:cache_rev] != false
81
+ head(key, cache_rev: cache_rev)
40
82
  end
41
83
 
42
84
  # (see Proxy#load)
85
+ # @option (see #key?)
43
86
  def load(key, options = {})
44
- response = @backend.get(key)
45
- response.status == 200 ? body_to_value(response.body) : nil
87
+ cache_rev = options[:cache_rev] != false
88
+ doc = get(key, cache_rev: cache_rev)
89
+ doc ? doc_to_value(doc) : nil
46
90
  end
47
91
 
48
92
  # (see Proxy#store)
93
+ # @option (see #key?)
94
+ # @option options [Boolean] :batch (false) Whether to do a
95
+ # {https://docs.couchdb.org/en/stable/api/database/common.html#api-doc-batch-writes
96
+ # batch mode write}
97
+ # @option options [Boolean] :full_commit (nil) Set to `true` or `false`
98
+ # to override the server's
99
+ # {https://docs.couchdb.org/en/stable/config/couchdb.html#couchdb/delayed_commits
100
+ # commit policy}
49
101
  def store(key, value, options = {})
50
- response = @backend.head(key)
51
- body = value_to_body(value, response.status == 200 && response['etag'][1..-2])
52
- response = @backend.put(key, body, 'Content-Type' => 'application/json')
53
- raise "HTTP error #{response.status}" unless response.status == 201
102
+ put(key, value_to_doc(value, rev(key)),
103
+ headers: full_commit_header(options[:full_commit]),
104
+ query: options[:batch] ? { batch: 'ok' } : {},
105
+ cache_rev: options[:cache_rev] != false,
106
+ expect: options[:batch] ? 202 : 201)
54
107
  value
55
- rescue
108
+ rescue HTTPError
56
109
  tries ||= 0
57
110
  (tries += 1) < 10 ? retry : raise
58
111
  end
59
112
 
60
113
  # (see Proxy#delete)
114
+ # @option options [Boolean] :batch (false) Whether to do a
115
+ # {https://docs.couchdb.org/en/stable/api/database/common.html#api-doc-batch-writes
116
+ # batch mode write}
117
+ # @option options [Boolean] :full_commit (nil) Set to `true` or `false`
118
+ # to override the server's
119
+ # {https://docs.couchdb.org/en/stable/config/couchdb.html#couchdb/delayed_commits
120
+ # commit policy}
61
121
  def delete(key, options = {})
62
- response = @backend.get(key)
63
- if response.status == 200
64
- value = body_to_value(response.body)
65
- response = @backend.delete("#{key}?rev=#{response['etag'][1..-2]}")
66
- raise "HTTP error #{response.status}" unless response.status == 200
122
+ get_response = get(key, returns: :response)
123
+ if get_response.success?
124
+ value = body_to_value(get_response.body)
125
+ existing_rev = parse_rev(get_response)
126
+ query = { rev: existing_rev }
127
+ query[:batch] = 'ok' if options[:batch]
128
+ request(:delete, key,
129
+ headers: full_commit_header(options[:full_commit]),
130
+ query: query,
131
+ expect: options[:batch] ? 202 : 200)
132
+ delete_cached_rev(key)
67
133
  value
68
134
  end
69
- rescue
135
+ rescue HTTPError
70
136
  tries ||= 0
71
137
  (tries += 1) < 10 ? retry : raise
72
138
  end
73
139
 
74
140
  # (see Proxy#clear)
141
+ # @option options [Boolean] :compact (false) Whether to compact the database after clearing
142
+ # @option options [Boolean] :await_compact (false) Whether to wait for compaction to complete
143
+ # before returning.
144
+ # @option options [Boolean] :full_commit (nil) Set to `true` or `false`
145
+ # to override the server's
146
+ # {https://docs.couchdb.org/en/stable/config/couchdb.html#couchdb/delayed_commits
147
+ # commit policy}
75
148
  def clear(options = {})
76
- @backend.delete ''
77
- create_db
149
+ loop do
150
+ docs = all_docs(limit: 10_000)
151
+ break if docs['rows'].empty?
152
+ deletions = docs['rows'].map do |row|
153
+ { _id: row['id'], _rev: row['value']['rev'], _deleted: true }
154
+ end
155
+ bulk_docs(deletions, full_commit: options[:full_commit])
156
+ end
157
+
158
+ # Compact the database unless told not to
159
+ if options[:compact]
160
+ post('_compact', expect: 202)
161
+
162
+ # Performance won't be great while compaction is happening, so by default we wait for it
163
+ if options[:await_compact]
164
+ loop do
165
+ db_info = get('', expect: 200)
166
+ break unless db_info['compact_running']
167
+
168
+ # wait before checking again
169
+ sleep 1
170
+ end
171
+ end
172
+ end
173
+
78
174
  self
79
175
  end
80
176
 
81
177
  # (see Proxy#create)
178
+ # @option (see #key?)
82
179
  def create(key, value, options = {})
83
- body = value_to_body(value, nil)
84
- response = @backend.put(key, body, 'Content-Type' => 'application/json')
180
+ cache_rev = options[:cache_rev] != false
181
+ doc = value_to_doc(value, nil)
182
+ response = put(key, doc, cache_rev: cache_rev, returns: :response)
85
183
  case response.status
86
184
  when 201
87
185
  true
88
186
  when 409
89
187
  false
90
188
  else
91
- raise "HTTP error #{response.status}"
189
+ raise HTTPError.new(response.status, :put, @backend.create_url(key))
92
190
  end
93
- rescue
191
+ rescue HTTPError
94
192
  tries ||= 0
95
193
  (tries += 1) < 10 ? retry : raise
96
194
  end
97
195
 
196
+ # (see Proxy#each_key)
197
+ def each_key
198
+ return enum_for(:each_key) unless block_given?
199
+
200
+ skip = 0
201
+ limit = 1000
202
+ loop do
203
+ docs = all_docs(limit: limit, skip: skip)
204
+ break if docs['rows'].empty?
205
+ skip += docs['rows'].length
206
+ docs['rows'].each do |row|
207
+ key = row['id']
208
+ @rev_cache[key] = row['value']['rev']
209
+ yield key
210
+ end
211
+ end
212
+ self
213
+ end
214
+
215
+ # (see Proxy#values_at)
216
+ def values_at(*keys, **options)
217
+ hash = Hash[slice(*keys, **options)]
218
+ keys.map { |key| hash[key] }
219
+ end
220
+
221
+ # (see Proxy#slice)
222
+ def slice(*keys, **options)
223
+ docs = all_docs(keys: keys, include_docs: true)
224
+ docs["rows"].map do |row|
225
+ next unless doc = row['doc']
226
+ [row['id'], doc_to_value(doc)]
227
+ end.compact
228
+ end
229
+
230
+ # (see Proxy#merge!)
231
+ # @option options [Boolean] :full_commit (nil) Set to `true` or `false`
232
+ # to override the server's
233
+ # {https://docs.couchdb.org/en/stable/config/couchdb.html#couchdb/delayed_commits
234
+ # commit policy}
235
+ def merge!(pairs, options = {})
236
+ keys = pairs.map { |key, _| key }.to_a
237
+ cache_revs(*keys.reject { |key| @rev_cache[key] })
238
+
239
+ if block_given?
240
+ existing = Hash[slice(*keys, **options)]
241
+ pairs = pairs.map do |key, new_value|
242
+ [
243
+ key,
244
+ if existing.key?(key)
245
+ yield(key, existing[key], new_value)
246
+ else
247
+ new_value
248
+ end
249
+ ]
250
+ end
251
+ end
252
+
253
+ docs = pairs.map { |key, value| value_to_doc(value, @rev_cache[key], key) }.to_a
254
+ results = bulk_docs(docs, full_commit: options[:full_commit], returns: :doc)
255
+ retries = results.each_with_object([]) do |row, retries|
256
+ ok, id = row.values_at('ok', 'id')
257
+ if ok
258
+ @rev_cache[id] = row['rev']
259
+ elsif row['error'] == 'conflict'
260
+ delete_cached_rev(id)
261
+ retries << pairs.find { |key,| key == id }
262
+ else
263
+ raise "Unrecognised response: #{row}"
264
+ end
265
+ end
266
+
267
+ # Recursive call with all conflicts
268
+ if retries.empty?
269
+ self
270
+ else
271
+ merge!(retries, options)
272
+ end
273
+ end
274
+
98
275
  private
99
276
 
277
+ def full_commit_header(full_commit)
278
+ full_commit == nil ? {} : { 'X-Couch-Full-Commit' => (!!full_commit).to_s }
279
+ end
280
+
100
281
  def body_to_value(body)
101
- doc = MultiJson.load(body)
102
- case doc[@type_field]
282
+ doc_to_value(MultiJson.load(body))
283
+ end
284
+
285
+ def doc_to_value(doc)
286
+ case doc[config.type_field]
103
287
  when 'Hash'
104
288
  doc = doc.dup
105
289
  doc.delete('_id')
106
290
  doc.delete('_rev')
107
- doc.delete(@type_field)
291
+ doc.delete(config.type_field)
108
292
  doc
109
293
  else
110
- doc[@value_field]
294
+ doc[config.value_field]
111
295
  end
112
296
  end
113
297
 
114
- def value_to_body(value, rev)
115
- case value
116
- when Hash
117
- doc = value.merge(@type_field => 'Hash')
118
- when String
119
- doc = { @value_field => value, @type_field => 'String' }
120
- when Float, Fixnum
121
- doc = { @value_field => value, @type_field => 'Number' }
122
- else
123
- raise ArgumentError, "Invalid value type: #{value.class}"
124
- end
298
+ def value_to_doc(value, rev, id = nil)
299
+ doc =
300
+ case value
301
+ when Hash
302
+ value.merge(config.type_field => 'Hash')
303
+ when String
304
+ { config.value_field => value, config.type_field => 'String' }
305
+ when Float, Integer
306
+ { config.value_field => value, config.type_field => 'Number' }
307
+ else
308
+ raise ArgumentError, "Invalid value type: #{value.class}"
309
+ end
125
310
  doc['_rev'] = rev if rev
126
- MultiJson.dump(doc)
311
+ doc['_id'] = id if id
312
+ doc
127
313
  end
128
314
 
129
315
  def create_db
130
- response = @backend.put '', ''
131
- raise "HTTP error #{response.status}" unless response.status == 201 || response.status == 412
316
+ loop do
317
+ response = put('', returns: :response)
318
+ case response.status
319
+ when 201
320
+ break
321
+ when 412
322
+ # Make sure the database really does exist
323
+ # See https://github.com/apache/couchdb/issues/2073
324
+ break if head('')
325
+ else
326
+ raise HTTPError.new(response.status, :put, '')
327
+ end
328
+
329
+ # Wait before trying again
330
+ sleep 1
331
+ end
332
+
333
+ self
334
+ end
335
+
336
+ def cache_revs(*keys)
337
+ docs = all_docs(keys: keys)
338
+ docs['rows'].each do |row|
339
+ next if !row['value'] || row['value']['deleted']
340
+ @rev_cache[row['id']] = row['value']['rev']
341
+ end
342
+ end
343
+
344
+ def parse_rev(response)
345
+ response['etag'][1..-2]
346
+ end
347
+
348
+ def cache_response_rev(key, response)
349
+ case response.status
350
+ when 200, 201
351
+ @rev_cache[key] = parse_rev(response)
352
+ else
353
+ delete_cached_rev(key)
354
+ nil
355
+ end
356
+ end
357
+
358
+ def delete_cached_rev(key)
359
+ @rev_cache.delete(key)
360
+ end
361
+
362
+ def rev(key)
363
+ @rev_cache[key] || begin
364
+ response = @backend.head(key)
365
+ cache_response_rev(key, response)
366
+ end
367
+ end
368
+
369
+ def encode_query(query)
370
+ query.map { |key, value| [key, MultiJson.dump(value)] }
371
+ end
372
+
373
+ def request(method, key, body = nil, returns: :nil, cache_rev: false, expect: nil, query: nil, headers: {})
374
+ url = @backend.build_url(key, query)
375
+ headers['Content-Type'] = 'application/json' if %i{put post}.include?(method)
376
+ response = @backend.run_request(method, url, body || '', headers)
377
+
378
+ if cache_rev
379
+ cache_response_rev(key, response)
380
+ end
381
+
382
+ if expect
383
+ raise HTTPError.new(response.status, method, url) unless response.status == expect
384
+ end
385
+
386
+ case returns
387
+ when :response
388
+ response
389
+ when :success
390
+ response.success?
391
+ when :doc
392
+ response.success? ? MultiJson.load(response.body) : nil
393
+ when :nil
394
+ nil
395
+ else
396
+ raise "Unknown returns param: #{returns.inspect}"
397
+ end
398
+ end
399
+
400
+ def get(key, returns: :doc, **options)
401
+ request(:get, key, returns: returns, **options)
402
+ end
403
+
404
+ def head(key, returns: :success, **options)
405
+ request(:head, key, returns: returns, **options)
406
+ end
407
+
408
+ def put(key, doc = nil, returns: :success, **options)
409
+ body = doc == nil ? '' : MultiJson.dump(doc)
410
+ request(:put, key, body, returns: returns, **options)
411
+ end
412
+
413
+ def post(key, doc = nil, returns: :success, **options)
414
+ body = doc == nil ? '' : MultiJson.dump(doc)
415
+ request(:post, key, body, returns: returns, **options)
416
+ end
417
+
418
+ def all_docs(sorted: false, **params)
419
+ keys = params.delete(:keys)
420
+ query = encode_query(params.merge(sorted: sorted))
421
+ if keys
422
+ post('_all_docs', { keys: keys },
423
+ query: query,
424
+ expect: 200,
425
+ returns: :doc)
426
+ else
427
+ get('_all_docs', query: query, expect: 200)
428
+ end
429
+ end
430
+
431
+ def bulk_docs(docs, returns: :success, full_commit: nil)
432
+ post('_bulk_docs', { docs: docs },
433
+ headers: full_commit_header(full_commit),
434
+ returns: returns,
435
+ expect: 201)
132
436
  end
133
437
  end
134
438
  end
@@ -7,9 +7,12 @@ module Moneta
7
7
  # @api public
8
8
  class DataMapper
9
9
  include Defaults
10
+ include Config
11
+ include NilValues
10
12
 
11
13
  supports :create
12
14
 
15
+ # @api private
13
16
  class Store
14
17
  include ::DataMapper::Resource
15
18
  property :k, String, key: true, length: 255
@@ -17,15 +20,18 @@ module Moneta
17
20
  self.raise_on_save_failure = true
18
21
  end
19
22
 
23
+ config :setup, required: true
24
+ config :repository, default: :moneta, coerce: :to_sym
25
+ config :table, default: :moneta, coerce: :to_sym
26
+
20
27
  # @param [Hash] options
21
28
  # @option options [String] :setup Datamapper setup string
22
29
  # @option options [String/Symbol] :repository (:moneta) Repository name
23
30
  # @option options [String/Symbol] :table (:moneta) Table name
24
31
  def initialize(options = {})
25
- raise ArgumentError, 'Option :setup is required' unless options[:setup]
26
- @repository = (options.delete(:repository) || :moneta).to_sym
27
- Store.storage_names[@repository] = (options.delete(:table) || :moneta).to_s
28
- ::DataMapper.setup(@repository, options[:setup])
32
+ configure(options)
33
+ Store.storage_names[config.repository] = config.table.to_s
34
+ ::DataMapper.setup(config.repository, config.setup)
29
35
  context { Store.auto_upgrade! }
30
36
  end
31
37
 
@@ -89,7 +95,7 @@ module Moneta
89
95
  private
90
96
 
91
97
  def context
92
- ::DataMapper.repository(@repository) { yield }
98
+ ::DataMapper.repository(config.repository) { yield }
93
99
  end
94
100
  end
95
101
  end
@@ -4,44 +4,55 @@ module Moneta
4
4
  module Adapters
5
5
  # Daybreak backend
6
6
  # @api public
7
- class Daybreak < Memory
8
- # @param [Hash] options
9
- # @option options [String] :file Database file
10
- # @option options [::Daybreak] :backend Use existing backend instance
11
- def initialize(options = {})
12
- @backend = options[:backend] ||
13
- begin
14
- raise ArgumentError, 'Option :file is required' unless options[:file]
15
- ::Daybreak::DB.new(options[:file], serializer: ::Daybreak::Serializer::None)
16
- end
17
- end
7
+ class Daybreak < Adapter
8
+ include DBMAdapter
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 [::Daybreak] :backend Use existing backend instance
17
+ backend { |file:| ::Daybreak::DB.new(file, serializer: ::Daybreak::Serializer::None) }
18
18
 
19
19
  # (see Proxy#load)
20
20
  def load(key, options = {})
21
- @backend.load if options[:sync]
22
- @backend[key]
21
+ backend.load if options[:sync]
22
+ backend[key]
23
23
  end
24
24
 
25
25
  # (see Proxy#store)
26
26
  def store(key, value, options = {})
27
- @backend[key] = value
28
- @backend.flush if options[:sync]
27
+ backend[key] = value
28
+ backend.flush if options[:sync]
29
29
  value
30
30
  end
31
31
 
32
32
  # (see Proxy#increment)
33
33
  def increment(key, amount = 1, options = {})
34
- @backend.lock { super }
34
+ backend.lock { super }
35
35
  end
36
36
 
37
37
  # (see Proxy#create)
38
38
  def create(key, value, options = {})
39
- @backend.lock { super }
39
+ backend.lock { super }
40
40
  end
41
41
 
42
- # (see Proxy#close)
43
- def close
44
- @backend.close
42
+ # (see Proxy#merge!)
43
+ def merge!(pairs, options = {})
44
+ if block_given?
45
+ backend.lock do
46
+ backend.update(pairs.map do |key, new_value|
47
+ new_value = yield(key, load(key), new_value) if key?(key)
48
+ [key, new_value]
49
+ end)
50
+ end
51
+ else
52
+ backend.update(pairs)
53
+ end
54
+
55
+ self
45
56
  end
46
57
  end
47
58
  end
@@ -4,23 +4,17 @@ module Moneta
4
4
  module Adapters
5
5
  # DBM backend (Berkeley DB)
6
6
  # @api public
7
- class DBM < Memory
8
- # @param [Hash] options
9
- # @option options [String] :file Database file
10
- # @option options [::DBM] :backend Use existing backend instance
11
- def initialize(options = {})
12
- @backend = options[:backend] ||
13
- begin
14
- raise ArgumentError, 'Option :file is required' unless options[:file]
15
- ::DBM.new(options[:file])
16
- end
17
- end
7
+ class DBM < Adapter
8
+ include DBMAdapter
9
+ include IncrementSupport
10
+ include CreateSupport
11
+ include EachKeySupport
18
12
 
19
- # (see Proxy#close)
20
- def close
21
- @backend.close
22
- nil
23
- end
13
+ # @!method initialize(options = {})
14
+ # @param [Hash] options
15
+ # @option options [String] :file Database file
16
+ # @option options [::DBM] :backend Use existing backend instance
17
+ backend { |file:| ::DBM.new(file) }
24
18
  end
25
19
  end
26
20
  end