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,94 @@ module Moneta
4
4
  module Adapters
5
5
  # Cassandra backend
6
6
  # @api public
7
- # @author Potapov Sergey (aka Blake)
8
- class Cassandra
9
- include Defaults
7
+ class Cassandra < Adapter
10
8
  include ExpiresSupport
11
9
 
12
- attr_reader :backend
10
+ supports :each_key
11
+
12
+ config :table, default: 'moneta'
13
+ config :key_column, default: 'key'
14
+ config :value_column, default: 'value'
15
+ config :updated_column, default: 'updated_at'
16
+ config :expired_column, default: 'expired'
17
+ config :read_consistency, default: :all
18
+ config :write_consistency, default: :all
19
+
20
+ backend do |keyspace: 'moneta', cluster: nil, create_keyspace: nil, **options|
21
+ cluster ||= ::Cassandra.cluster(options).tap do |own_cluster|
22
+ @own_cluster = own_cluster
23
+ end
24
+
25
+ begin
26
+ cluster.connect(keyspace)
27
+ rescue ::Cassandra::Errors::InvalidError
28
+ backend = cluster.connect
29
+ create_keyspace(backend, keyspace, create_keyspace)
30
+ backend.execute("USE " + keyspace)
31
+ backend
32
+ end
33
+ end
13
34
 
14
35
  # @param [Hash] options
15
36
  # @option options [String] :keyspace ('moneta') Cassandra keyspace
16
- # @option options [String] :column_family ('moneta') Cassandra column family
37
+ # @option options [String] :table ('moneta') Cassandra table
17
38
  # @option options [String] :host ('127.0.0.1') Server host name
18
39
  # @option options [Integer] :port (9160) Server port
19
40
  # @option options [Integer] :expires Default expiration time
20
- # @option options [::Cassandra] :backend Use existing backend instance
21
- # @option options Other options passed to `Cassandra#new`
41
+ # @option options [String] :key_column ('key') Name of the key column
42
+ # @option options [String] :value_column ('value') Name of the value
43
+ # column
44
+ # @option options [String] :updated_column ('updated_at') Name of the
45
+ # column used to track last update
46
+ # @option options [String] :expired_column ('expired') Name of the column
47
+ # used to track expiry
48
+ # @option options [Symbol] :read_consistency (:all) Default read
49
+ # consistency
50
+ # @option options [Symbol] :write_consistency (:all) Default write
51
+ # consistency
52
+ # @option options [Proc, Boolean, Hash] :create_keyspace Provide a proc
53
+ # for creating the keyspace, or a Hash of options to use when creating
54
+ # it, or set to false to disable. The Proc will only be called if the
55
+ # keyspace does not already exist.
56
+ # @option options [::Cassandra::Cluster] :cluster Existing cluster to use
57
+ # @option options [::Cassandra::Session] :backend Existing session to use
58
+ # @option options Other options passed to `Cassandra#cluster`
22
59
  def initialize(options = {})
23
- self.default_expires = options.delete(:expires)
24
- @cf = (options.delete(:column_family) || 'moneta').to_sym
25
- if options[:backend]
26
- @backend = options[:backend]
27
- else
28
- keyspace = options.delete(:keyspace) || 'moneta'
29
- options[:retries] ||= 3
30
- options[:connect_timeout] ||= 10
31
- options[:timeout] ||= 10
32
- @backend = ::Cassandra.new('system',
33
- "#{options[:host] || '127.0.0.1'}:#{options[:port] || 9160}",
34
- options)
35
- unless @backend.keyspaces.include?(keyspace)
36
- cf_def = ::Cassandra::ColumnFamily.new(keyspace: keyspace, name: @cf.to_s)
37
- ks_def = ::Cassandra::Keyspace.new(name: keyspace,
38
- strategy_class: 'SimpleStrategy',
39
- strategy_options: { 'replication_factor' => '1' },
40
- replication_factor: 1,
41
- cf_defs: [cf_def])
42
- # Wait for keyspace to be created (issue #24)
43
- 10.times do
44
- begin
45
- @backend.add_keyspace(ks_def)
46
- rescue Exception => ex
47
- warn "Moneta::Adapters::Cassandra - #{ex.message}"
48
- end
49
- break if @backend.keyspaces.include?(keyspace)
50
- sleep 0.1
51
- end
52
- end
53
- @backend.keyspace = keyspace
54
- end
60
+ super
61
+
62
+ backend.execute <<-CQL
63
+ CREATE TABLE IF NOT EXISTS #{config.table} (
64
+ #{config.key_column} blob,
65
+ #{config.value_column} blob,
66
+ #{config.updated_column} timeuuid,
67
+ #{config.expired_column} boolean,
68
+ PRIMARY KEY (#{config.key_column}, #{config.updated_column})
69
+ )
70
+ CQL
71
+
72
+ prepare_statements
55
73
  end
56
74
 
57
75
  # (see Proxy#key?)
58
76
  def key?(key, options = {})
59
- if @backend.exists?(@cf, key)
60
- load(key, options) if options.include?(:expires)
77
+ rc, wc = consistency(options)
78
+ if (expires = expires_value(options, nil)) != nil
79
+ # Because Cassandra expires each value in a column, rather than the
80
+ # whole column, when we want to update the expiry we load the value
81
+ # and then re-set it in order to update the TTL.
82
+ return false unless
83
+ row = backend.execute(@load, options.merge(consistency: rc, arguments: [key])).first and
84
+ row[config.expired_column] != nil
85
+ backend.execute(@update_expires,
86
+ options.merge(consistency: wc,
87
+ arguments: [(expires || 0).to_i,
88
+ timestamp,
89
+ row[config.value_column],
90
+ key,
91
+ row[config.updated_column]]))
61
92
  true
93
+ elsif row = backend.execute(@key, options.merge(consistency: rc, arguments: [key])).first
94
+ row[config.expired_column] != nil
62
95
  else
63
96
  false
64
97
  end
@@ -66,38 +99,257 @@ module Moneta
66
99
 
67
100
  # (see Proxy#load)
68
101
  def load(key, options = {})
69
- if value = @backend.get(@cf, key)
70
- expires = expires_value(options, nil)
71
- @backend.insert(@cf, key, {'value' => value['value'] }, ttl: expires || nil) if expires != nil
72
- value['value']
102
+ rc, wc = consistency(options)
103
+ if row = backend.execute(@load, options.merge(consistency: rc, arguments: [key])).first and row[config.expired_column] != nil
104
+ if (expires = expires_value(options, nil)) != nil
105
+ backend.execute(@update_expires,
106
+ options.merge(consistency: wc,
107
+ arguments: [(expires || 0).to_i,
108
+ timestamp,
109
+ row[config.value_column],
110
+ key,
111
+ row[config.updated_column]]))
112
+ end
113
+ row[config.value_column]
73
114
  end
74
115
  end
75
116
 
76
117
  # (see Proxy#store)
77
118
  def store(key, value, options = {})
78
- @backend.insert(@cf, key, {'value' => value}, ttl: expires_value(options) || nil)
119
+ _, wc = consistency(options)
120
+ expires = expires_value(options)
121
+ t = timestamp
122
+ batch = backend.batch do |batch|
123
+ batch.add(@store_delete, arguments: [t, key])
124
+ batch.add(@store, arguments: [key, value, (expires || 0).to_i, t + 1])
125
+ end
126
+ backend.execute(batch, options.merge(consistency: wc))
79
127
  value
80
128
  end
81
129
 
82
130
  # (see Proxy#delete)
83
131
  def delete(key, options = {})
84
- if value = load(key, options)
85
- @backend.remove(@cf, key)
86
- value
132
+ rc, wc = consistency(options)
133
+ result = backend.execute(@delete_value, options.merge(consistency: rc, arguments: [key]))
134
+ if row = result.first and row[config.expired_column] != nil
135
+ backend.execute(@delete, options.merge(consistency: wc, arguments: [timestamp, key, row[config.updated_column]]))
136
+ row[config.value_column]
87
137
  end
88
138
  end
89
139
 
90
140
  # (see Proxy#clear)
91
141
  def clear(options = {})
92
- @backend.clear_column_family!(@cf)
142
+ backend.execute(@clear)
93
143
  self
94
144
  end
95
145
 
96
146
  # (see Proxy#close)
97
147
  def close
98
- @backend.disconnect!
148
+ backend.close_async
149
+ @backend = nil
150
+ if @own_cluster
151
+ @own_cluster.close_async
152
+ @own_cluster = nil
153
+ end
99
154
  nil
100
155
  end
156
+
157
+ # (see Proxy#each_key)
158
+ def each_key
159
+ rc, = consistency
160
+ return enum_for(:each_key) unless block_given?
161
+ result = backend.execute(@each_key, consistency: rc, page_size: 100)
162
+ loop do
163
+ result.each do |row|
164
+ next if row[config.expired_column] == nil
165
+ yield row[config.key_column]
166
+ end
167
+
168
+ break if result.last_page?
169
+ result = result.next_page
170
+ end
171
+ self
172
+ end
173
+
174
+ # (see Proxy#slice)
175
+ def slice(*keys, **options)
176
+ rc, wc = consistency(options)
177
+ result = backend.execute(@slice, options.merge(consistency: rc, arguments: [keys]))
178
+ expires = expires_value(options, nil)
179
+ updated = [] if expires != nil
180
+ pairs = result.map do |row|
181
+ next if row[config.expired_column] == nil
182
+ if expires != nil
183
+ updated << [row[config.key_column], row[config.value_column], row[config.updated_column]]
184
+ end
185
+ [row[config.key_column], row[config.value_column]]
186
+ end.compact
187
+
188
+ if expires != nil && !updated.empty?
189
+ ttl = (expires || 0).to_i
190
+ t = timestamp
191
+ batch = backend.batch do |batch|
192
+ updated.each do |key, value, updated|
193
+ batch.add(@update_expires, arguments: [ttl, t, value, key, updated])
194
+ end
195
+ end
196
+
197
+ backend.execute(batch, options.merge(consistency: wc))
198
+ end
199
+
200
+ pairs
201
+ end
202
+
203
+ # (see Proxy#values_at)
204
+ def values_at(*keys, **options)
205
+ hash = Hash[slice(*keys, **options)]
206
+ keys.map { |key| hash[key] }
207
+ end
208
+
209
+ # (see Proxy#fetch_values)
210
+ def fetch_values(*keys, **options)
211
+ return values_at(*keys, **options) unless block_given?
212
+ hash = Hash[slice(*keys, **options)]
213
+ keys.map do |key|
214
+ if hash.key?(key)
215
+ hash[key]
216
+ else
217
+ yield key
218
+ end
219
+ end
220
+ end
221
+
222
+ # (see Proxy#merge!)
223
+ def merge!(pairs, options = {})
224
+ keys = pairs.map { |k, _| k }.to_a
225
+ return self if keys.empty?
226
+
227
+ if block_given?
228
+ existing = Hash[slice(*keys, **options)]
229
+ pairs = pairs.map do |key, new_value|
230
+ if existing.key?(key)
231
+ [key, yield(key, existing[key], new_value)]
232
+ else
233
+ [key, new_value]
234
+ end
235
+ end
236
+ end
237
+
238
+ _rc, wc = consistency(options)
239
+ expires = expires_value(options)
240
+ t = timestamp
241
+ batch = backend.batch do |batch|
242
+ batch.add(@merge_delete, arguments: [t, keys])
243
+ pairs.each do |key, value|
244
+ batch.add(@store, arguments: [key, value, (expires || 0).to_i, t + 1])
245
+ end
246
+ end
247
+ backend.execute(batch, options.merge(consistency: wc))
248
+
249
+ self
250
+ end
251
+
252
+ private
253
+
254
+ def timestamp
255
+ (Time.now.to_r * 1_000_000).to_i
256
+ end
257
+
258
+ def create_keyspace(backend, keyspace, create_keyspace)
259
+ options = {
260
+ replication: {
261
+ class: 'SimpleStrategy',
262
+ replication_factor: 1
263
+ }
264
+ }
265
+
266
+ case create_keyspace
267
+ when Proc
268
+ return create_keyspace.call(keyspace)
269
+ when false
270
+ return
271
+ when Hash
272
+ options.merge!(create_keyspace)
273
+ end
274
+
275
+ # This is a bit hacky, but works. Options in Cassandra look like JSON,
276
+ # but use single quotes instead of double-quotes.
277
+ require 'multi_json'
278
+ option_str = options.map do |key, value|
279
+ key.to_s + ' = ' + MultiJson.dump(value).tr(?", ?')
280
+ end.join(' AND ')
281
+
282
+ backend.execute "CREATE KEYSPACE IF NOT EXISTS %<keyspace>s WITH %<options>s" % {
283
+ keyspace: keyspace,
284
+ options: option_str
285
+ }
286
+ rescue ::Cassandra::Errors::TimeoutError
287
+ tries ||= 0
288
+ (tries += 1) <= 3 ? retry : raise
289
+ end
290
+
291
+ def prepare_statements
292
+ @key = backend.prepare(<<-CQL)
293
+ SELECT #{config.updated_column}, #{config.expired_column}
294
+ FROM #{config.table} WHERE #{config.key_column} = ?
295
+ LIMIT 1
296
+ CQL
297
+ @store_delete = backend.prepare(<<-CQL)
298
+ DELETE FROM #{config.table}
299
+ USING TIMESTAMP ?
300
+ WHERE #{config.key_column} = ?
301
+ CQL
302
+ @store = backend.prepare(<<-CQL)
303
+ INSERT INTO #{config.table} (#{config.key_column}, #{config.value_column}, #{config.updated_column}, #{config.expired_column})
304
+ VALUES (?, ?, now(), false)
305
+ USING TTL ? AND TIMESTAMP ?
306
+ CQL
307
+ @load = backend.prepare(<<-CQL)
308
+ SELECT #{config.value_column}, #{config.updated_column}, #{config.expired_column}
309
+ FROM #{config.table}
310
+ WHERE #{config.key_column} = ?
311
+ LIMIT 1
312
+ CQL
313
+ @update_expires = backend.prepare(<<-CQL)
314
+ UPDATE #{config.table}
315
+ USING TTL ? AND TIMESTAMP ?
316
+ SET #{config.value_column} = ?, #{config.expired_column} = false
317
+ WHERE #{config.key_column} = ? AND #{config.updated_column} = ?
318
+ CQL
319
+ @clear = backend.prepare("TRUNCATE #{config.table}")
320
+ @delete_value = backend.prepare(<<-CQL)
321
+ SELECT #{config.value_column}, #{config.updated_column}, #{config.expired_column}
322
+ FROM #{config.table}
323
+ WHERE #{config.key_column} = ?
324
+ LIMIT 1
325
+ CQL
326
+ @delete = backend.prepare(<<-CQL, idempotent: true)
327
+ DELETE FROM #{config.table}
328
+ USING TIMESTAMP ?
329
+ WHERE #{config.key_column} = ? AND #{config.updated_column} = ?
330
+ CQL
331
+ @each_key = backend.prepare(<<-CQL)
332
+ SELECT #{config.key_column}, #{config.expired_column}
333
+ FROM #{config.table}
334
+ CQL
335
+ @slice = backend.prepare(<<-CQL)
336
+ SELECT #{config.key_column}, #{config.value_column}, #{config.updated_column}, #{config.expired_column}
337
+ FROM #{config.table}
338
+ WHERE #{config.key_column} IN ?
339
+ CQL
340
+ @merge_delete = backend.prepare(<<-CQL)
341
+ DELETE FROM #{config.table}
342
+ USING TIMESTAMP ?
343
+ WHERE #{config.key_column} IN ?
344
+ CQL
345
+ end
346
+
347
+ def consistency(options = {})
348
+ [
349
+ options[:read_consistency] || config.read_consistency,
350
+ options[:write_consistency] || config.write_consistency
351
+ ]
352
+ end
101
353
  end
102
354
  end
103
355
  end
@@ -4,74 +4,108 @@ module Moneta
4
4
  module Adapters
5
5
  # Moneta client backend
6
6
  # @api public
7
- class Client
8
- include Defaults
9
-
10
- # @param [Hash] options
11
- # @option options [Integer] :port (9000) TCP port
12
- # @option options [String] :host ('127.0.0.1') Hostname
13
- # @option options [String] :socket Unix socket file name as alternative to `:port` and `:host`
14
- def initialize(options = {})
15
- @socket = options[:socket] ? UNIXSocket.open(options[:socket]) :
16
- TCPSocket.open(options[:host] || '127.0.0.1', options[:port] || 9000)
7
+ class Client < Adapter
8
+ # @!method initialize(options = {})
9
+ # @param [Hash] options
10
+ # @option options [TCPSocket | UNIXSocket] :backend an open socket to use
11
+ # @option options [Integer] :port (9000) TCP port
12
+ # @option options [String] :host ('127.0.0.1') Hostname
13
+ # @option options [String] :socket Unix socket file name as alternative to `:port` and `:host`
14
+ backend do |socket: nil, host: '127.0.0.1', port: 9000|
15
+ if socket
16
+ UNIXSocket.open(socket)
17
+ else
18
+ TCPSocket.open(host, port)
19
+ end
17
20
  end
18
21
 
19
22
  # (see Proxy#key?)
20
23
  def key?(key, options = {})
21
24
  write(:key?, key, options)
22
- read
25
+ read_msg
23
26
  end
24
27
 
25
28
  # (see Proxy#load)
26
29
  def load(key, options = {})
27
30
  write(:load, key, options)
28
- read
31
+ read_msg
29
32
  end
30
33
 
31
34
  # (see Proxy#store)
32
35
  def store(key, value, options = {})
33
36
  write(:store, key, value, options)
34
- read
37
+ read_msg
35
38
  value
36
39
  end
37
40
 
38
41
  # (see Proxy#delete)
39
42
  def delete(key, options = {})
40
43
  write(:delete, key, options)
41
- read
44
+ read_msg
42
45
  end
43
46
 
44
47
  # (see Proxy#increment)
45
48
  def increment(key, amount = 1, options = {})
46
49
  write(:increment, key, amount, options)
47
- read
50
+ read_msg
48
51
  end
49
52
 
50
53
  # (see Proxy#create)
51
54
  def create(key, value, options = {})
52
55
  write(:create, key, value, options)
53
- read
56
+ read_msg
54
57
  end
55
58
 
56
59
  # (see Proxy#clear)
57
60
  def clear(options = {})
58
61
  write(:clear, options)
59
- read
62
+ read_msg
60
63
  self
61
64
  end
62
65
 
63
66
  # (see Proxy#close)
64
67
  def close
65
- @socket.close
68
+ backend.close
66
69
  nil
67
70
  end
68
71
 
72
+ # (see Proxy#each_key)
73
+ def each_key
74
+ raise NotImplementedError, 'each_key is not supported' unless supports?(:each_key)
75
+ return enum_for(:each_key) unless block_given?
76
+
77
+ begin
78
+ write(:each_key)
79
+ yield_break = false
80
+
81
+ loop do
82
+ write('NEXT')
83
+
84
+ # A StopIteration error will be raised by this call if the server
85
+ # reached the end of the enumeration. This will stop the loop
86
+ # automatically.
87
+ result = read_msg
88
+
89
+ # yield_break will be true in the ensure block (below) if anything
90
+ # happened during the yield to stop further enumeration.
91
+ yield_break = true
92
+ yield result
93
+ yield_break = false
94
+ end
95
+ ensure
96
+ write('BREAK') if yield_break
97
+ read_msg # nil return from each_key
98
+ end
99
+
100
+ self
101
+ end
102
+
69
103
  # (see Default#features)
70
104
  def features
71
105
  @features ||=
72
106
  begin
73
107
  write(:features)
74
- read.freeze
108
+ read_msg.freeze
75
109
  end
76
110
  end
77
111
 
@@ -79,12 +113,27 @@ module Moneta
79
113
 
80
114
  def write(*args)
81
115
  s = Marshal.dump(args)
82
- @socket.write([s.bytesize].pack('N') << s)
116
+ backend.write([s.bytesize].pack('N') << s)
117
+ end
118
+
119
+ # JRuby doesn't support socket#recv with flags
120
+ if defined?(JRUBY_VERSION)
121
+ def read(bytes)
122
+ received = backend.read(bytes)
123
+ raise EOFError, "Server closed socket" unless received && received.bytesize == bytes
124
+ received
125
+ end
126
+ else
127
+ def read(bytes)
128
+ received = backend.recv(bytes, Socket::MSG_WAITALL)
129
+ raise EOFError, "Server closed socket" unless received && received.bytesize == bytes
130
+ received
131
+ end
83
132
  end
84
133
 
85
- def read
86
- size = @socket.read(4).unpack('N').first
87
- result = Marshal.load(@socket.read(size))
134
+ def read_msg
135
+ size = read(4).unpack('N').first
136
+ result = Marshal.load(read(size))
88
137
  raise result if Exception === result
89
138
  result
90
139
  end