├── .gems ├── .gems.dev ├── .github └── workflows │ ├── deploy.yml │ └── pull_request.yml ├── .gitignore ├── .ruby-version ├── README.md ├── Rakefile ├── app.rb ├── config.ru ├── deploy ├── downloads.json ├── lib ├── interactive │ ├── commands.rb │ ├── namespace.rb │ ├── redis.rb │ └── session.rb ├── reference.rb └── template.rb ├── makefile ├── public ├── app.js ├── images │ ├── companies │ │ └── bump.png │ ├── favicon.png │ ├── pivotal.png │ ├── redis-300dpi.png │ ├── redis-gray.png │ ├── redis-logo.svg │ ├── redis-small.png │ ├── redis-white.png │ ├── redis.png │ ├── redisdoc │ │ ├── 2idx_0.png │ │ ├── 2idx_1.png │ │ ├── 2idx_2.png │ │ ├── lru_comparison.png │ │ └── pipeline_iops.png │ ├── redislabs.png │ ├── shuttleworth.png │ └── vmware.png ├── opensearch.xml ├── presentation │ ├── Pnoordhuis_whats_new_in_2_2.pdf │ └── Redis_Cluster.pdf └── styles.css ├── scripts └── generate_interactive_commands.rb ├── test ├── clients.rb ├── command_reference.rb ├── formatting.rb ├── helper.rb ├── interactive │ └── namespace.rb ├── json.rb ├── sitemap.rb └── topics.rb └── views ├── 404.haml ├── avatar.haml ├── buzz.haml ├── clients.haml ├── commands.haml ├── commands └── name.haml ├── comments.haml ├── community.md ├── documentation.md ├── download.haml ├── grid.scss ├── home.haml ├── interactive.haml ├── layout.haml ├── modules.haml ├── normalize.scss ├── support.md └── topics └── name.haml /.gems: -------------------------------------------------------------------------------- 1 | redis -v 4.2.1 2 | cuba -v 3.8.0 3 | haml -v 4.0.7 4 | htmlentities -v 4.3.4 5 | oga -v 2.7 6 | ohm -v 0.1.5 7 | redcarpet -v 2.1.1 8 | tilt -v 2.0.5 9 | -------------------------------------------------------------------------------- /.gems.dev: -------------------------------------------------------------------------------- 1 | capybara -v 2.10.1 2 | cutest -v 1.2.3 3 | -------------------------------------------------------------------------------- /.github/workflows/deploy.yml: -------------------------------------------------------------------------------- 1 | name: Deploy website 2 | on: 3 | push: 4 | branches: 5 | - master 6 | 7 | jobs: 8 | deploy: 9 | runs-on: ubuntu-latest 10 | steps: 11 | - uses: actions/checkout@v2.1.1 12 | - shell: bash 13 | env: 14 | DEPLOY_TOKEN: ${{ secrets.DEPLOY_TOKEN }} 15 | run: | 16 | curl -G https://redis.io/deploy --data-urlencode token=$DEPLOY_TOKEN -------------------------------------------------------------------------------- /.github/workflows/pull_request.yml: -------------------------------------------------------------------------------- 1 | name: Test pull request 2 | on: 3 | pull_request: 4 | branches: 5 | - master 6 | 7 | jobs: 8 | check: 9 | runs-on: ubuntu-latest 10 | steps: 11 | - name: Install aspell 12 | run: sudo apt-get install aspell 13 | - run: docker run -d -p 6379:6379 redis 14 | - uses: ruby/setup-ruby@v1 15 | with: 16 | ruby-version: 2.6 17 | - uses: actions/checkout@v2.1.1 18 | - name: Install dep gem 19 | run: gem install dep 20 | - name: Install dependencies 21 | run: dep install 22 | - name: Install dev dependencies 23 | run: dep -f .gems.dev install 24 | - name: Run tests 25 | run: make test 26 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | /tmp 2 | /.sass-cache 3 | /redis-doc 4 | -------------------------------------------------------------------------------- /.ruby-version: -------------------------------------------------------------------------------- 1 | system 2 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # redis-io 2 | 3 | This repository holds the source code for the website that runs [redis.io](http://redis.io). 4 | 5 | ## Getting started 6 | 7 | The required gems are listed in the .gems file. To get up and running, 8 | preferably using a gemset, run: 9 | 10 | gem install dep 11 | dep install 12 | 13 | Now you need to clone the [redis-doc](https://github.com/redis/redis-doc) 14 | project, and set its path in the `REDIS_DOC` environment variable before starting 15 | the server or running the tests. 16 | 17 | Finally, you need to have a `redis-server` running on port 6379. 18 | 19 | To start the website: 20 | 21 | REDIS_DOC=/path/to/redis-doc rackup 22 | 23 | To run the tests, you also need to install `cutest` and `capybara`: 24 | 25 | gem install cutest capybara 26 | 27 | Now, just run: 28 | 29 | REDIS_DOC=/path/to/redis-doc rake 30 | 31 | Or to run the tests in a particular file: 32 | 33 | REDIS_DOC=/path/to/redis-doc cutest test/some_file.rb 34 | -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | task :default => :test 2 | 3 | task :test do 4 | require "cutest" 5 | 6 | Cutest.run(Dir["test/**/*.rb"]) 7 | end 8 | 9 | task :formatting do 10 | require "cutest" 11 | 12 | Cutest.run(Dir["test/formatting.rb"]) 13 | end 14 | 15 | task :update do 16 | sh "rm -rf redis-doc" 17 | sh "git clone -q --depth 1 git://github.com/redis/redis-doc.git" 18 | sh "rm -rf redis-doc/.git" 19 | end 20 | 21 | desc "Deploy" 22 | task :deploy do 23 | script = <<-EOS 24 | cd ~/redis-doc 25 | git pull 26 | cd ~/redis-io 27 | git pull 28 | rvm 1.9.2 gem install dep --no-ri --no-rdoc 29 | (rvm 1.9.2 exec dep check || rvm 1.9.2 exec dep install) 30 | rvm 1.9.2 exec compass compile -c config/sass.rb views/styles.sass 31 | kill -s INT $(cat log/redis-io.pid) 32 | rvm 1.9.2 exec unicorn -D -c unicorn.rb -E production 33 | EOS 34 | 35 | sh "ssh redis-io '#{script.split("\n").map(&:strip).join(" && ")}'" 36 | end 37 | -------------------------------------------------------------------------------- /app.rb: -------------------------------------------------------------------------------- 1 | # encoding: utf-8 2 | 3 | ROOT_PATH = File.expand_path(File.dirname(__FILE__)) 4 | 5 | require "cuba" 6 | require "cuba/render" 7 | require "date" 8 | require "digest/md5" 9 | require "haml" 10 | require "htmlentities" 11 | require "json" 12 | require "oga" 13 | require "ohm" 14 | require "open-uri" 15 | require "rack/static" 16 | require "redcarpet" 17 | require "redis" 18 | require "fileutils" 19 | 20 | require File.expand_path("lib/reference", ROOT_PATH) 21 | require File.expand_path("lib/template", ROOT_PATH) 22 | 23 | require File.expand_path("lib/interactive/namespace", ROOT_PATH) 24 | require File.expand_path("lib/interactive/session", ROOT_PATH) 25 | 26 | Encoding.default_external = Encoding::UTF_8 27 | 28 | DOWNLOADS = JSON.parse(File.read("downloads.json"), symbolize_names: true) 29 | 30 | STABLE_VERSION = DOWNLOADS.fetch(:channels).fetch(:stable).fetch(:version) 31 | 32 | module Kernel 33 | private 34 | 35 | def documentation_path 36 | $documentation_path ||= File.expand_path(ENV["REDIS_DOC"] || "redis-doc") 37 | end 38 | 39 | def commands 40 | $commands ||= Reference.new(JSON.parse(File.read(documentation_path + "/commands.json"))) 41 | end 42 | 43 | def new_redis_connection 44 | Redis.new(url: ENV["REDISCLOUD_URL"]) 45 | end 46 | 47 | def redis 48 | $redis ||= new_redis_connection 49 | end 50 | 51 | def redis_versions 52 | $redis_versions ||= redis.hgetall("versions") 53 | end 54 | 55 | def related_topics_for(command) 56 | # For now this is a quick and dirty way of figuring out related topics. 57 | 58 | path = "#{documentation_path}/topics/#{command.group}.md" 59 | 60 | return [] unless File.exist?(path) 61 | 62 | _, title = topic(path) 63 | 64 | [[title, "/topics/#{command.group}"]] 65 | end 66 | 67 | def related_commands_for(group) 68 | commands.select do |command| 69 | command.group == group && command.is_listed? 70 | end.sort_by(&:name) 71 | end 72 | 73 | def update_redis_versions 74 | tags = `git ls-remote -t https://github.com/redis/redis.git` 75 | 76 | versions = tags.scan(%r{refs/tags/(v?(?:\d\.?)*\-(?:stable|rc\w+|alpha\w+))}).flatten.uniq 77 | 78 | stable, development = versions.partition { |v| v =~ /^v/ } 79 | 80 | redis.hmset( 81 | "versions", 82 | "stable", stable.sort.last, 83 | "development", development.sort.last 84 | ) 85 | end 86 | 87 | def clean_version(version) 88 | version[/((?:\d\.?)+)/, 1] 89 | end 90 | 91 | def version_name(tag) 92 | tag[/v?(.*)/, 1].sub(/\-stable$/, "") 93 | end 94 | end 95 | 96 | Ohm.redis = redis 97 | 98 | class App < Cuba 99 | plugin Cuba::Render 100 | 101 | settings[:render][:template_engine] = "haml" 102 | 103 | use Rack::Static, 104 | urls: ["/images", "/presentation", "/opensearch.xml", "/styles.css", "/app.js"], 105 | root: File.join(ROOT_PATH, "public") 106 | 107 | def custom_render(path, locals = {}, options = {}) 108 | res.headers["Content-Type"] ||= "text/html; charset=utf-8" 109 | res.write(custom_view(path, locals, options)) 110 | end 111 | 112 | def custom_view(path, locals = {}, options = {}) 113 | options = { 114 | fenced_code_blocks: true, 115 | superscript: true, 116 | layout: true 117 | }.merge(options) 118 | 119 | path_with_extension = File.extname(path).empty? ? 120 | "#{path}.#{settings[:render][:template_engine]}" : 121 | path 122 | 123 | if path_with_extension.start_with?("/") 124 | expanded_path = path_with_extension 125 | else 126 | expanded_path = File.expand_path(path_with_extension, File.join(settings[:render][:views])) 127 | end 128 | 129 | layout_path = File.expand_path("#{settings[:render][:layout]}.#{settings[:render][:template_engine]}", File.join(settings[:render][:views])) 130 | 131 | data = _render(expanded_path, locals, options) 132 | 133 | unless options[:layout] == false 134 | data = _render(layout_path, locals.merge(content: data), options) 135 | end 136 | 137 | if expanded_path.start_with?(documentation_path) 138 | filter_interactive_examples(data) 139 | elsif expanded_path.start_with?(ROOT_PATH) && options[:anchors] != false 140 | add_header_ids(data) 141 | else 142 | data 143 | end 144 | end 145 | 146 | # Setup a new interactive session for every
with @cli
147 | def filter_interactive_examples(data)
148 | namespace = Digest::MD5.hexdigest([rand(2**32), Time.now.usec, Process.pid].join("-"))
149 | session = ::Interactive::Session.create(namespace)
150 |
151 | data.gsub %r{\s*\s*(.*?)\s*
\s*
}m do |match|
152 | lines = $1.split(/\n+/m).map(&:strip)
153 | _render("views/interactive.haml", session: session, lines: lines)
154 | end
155 | end
156 |
157 | def add_header_ids(data)
158 | data.gsub %r{(<(?h.)>(?.*?))} do |match|
159 | found = $~
160 | hdr = found[:hdr]
161 | section = found[:section]
162 | # convert spaces to underscores
163 | id = anchorize(section)
164 | %Q[<#{hdr} >*#{section}#{hdr}>]
165 | end
166 | end
167 |
168 | def anchorize(str)
169 | str.downcase.gsub(/[\s+]/, '-').gsub(/[^[:alnum:]-]/, "")
170 | end
171 |
172 | def anchorize_language(str)
173 | # Addresses languages such as C++ and C#
174 | anchorize(str.gsub(/\+/, '-plus').gsub(/#/, '-sharp'))
175 | end
176 |
177 | def topic(template)
178 | body = custom_view(template, {}, layout: false)
179 | title = body[%r{(.+?)
}, 1] # Nokogiri may be overkill
180 |
181 | return body, title
182 | end
183 |
184 | def gravatar_hash(email)
185 | Digest::MD5.hexdigest(email)
186 | end
187 |
188 | def not_found(locals = {path: nil})
189 | res.status = 404
190 | res.write(custom_view("404", locals))
191 | end
192 |
193 | define do
194 | on get, "" do
195 | custom_render("home", {}, anchors: false)
196 | end
197 |
198 | on get, "buzz" do
199 | custom_render("buzz", {}, anchors: false)
200 | end
201 |
202 | on get, "download" do
203 | custom_render("download")
204 | end
205 |
206 | on get, /(download|community|documentation|support)/ do |topic|
207 | @body, @title = topic("#{topic}.md")
208 | custom_render("topics/name")
209 | end
210 |
211 | on get, "commands" do
212 | on :name do |name|
213 | @name = name
214 | @title = @name.upcase.sub("-", " ")
215 | @command = commands[@title]
216 |
217 | if @command.nil?
218 | res.redirect "https://www.google.com/search?q=#{CGI.escape(name)}+site%3Aredis.io", 307
219 | halt res.finish
220 | end
221 |
222 | @related_commands = related_commands_for(@command.group)
223 | @related_topics = related_topics_for(@command)
224 |
225 | custom_render("commands/name")
226 | end
227 |
228 | on default do
229 | @commands = commands
230 | @title = "Command reference"
231 |
232 | custom_render("commands")
233 | end
234 | end
235 |
236 | on post, "session", /([0-9a-f]{32})/i do |id|
237 | if session = ::Interactive::Session.find(id)
238 | res.write session.run(req.params["command"].to_s)
239 | else
240 | res.status = 404
241 | res.write "ERR Session does not exist or has timed out."
242 | end
243 | end
244 |
245 | on get, "clients" do
246 | @clients = JSON.parse(File.read(documentation_path + "/clients.json"))
247 | @redis_tools = JSON.parse(File.read(documentation_path + "/tools.json"))
248 |
249 | @clients_by_language = @clients.group_by { |info| info["language"] }.sort_by { |name, _| name.downcase }
250 |
251 | custom_render("clients")
252 | end
253 |
254 | on get, "modules" do
255 | @modules = JSON.parse(File.read(documentation_path + "/modules.json"))
256 | @modules = @modules.sort_by {|m| -m["stars"]}
257 | custom_render("modules")
258 | end
259 |
260 | on get, "topics/:name" do |name|
261 | path = "/topics/#{name}.md"
262 |
263 | if File.exist?(File.join(documentation_path, path))
264 | @css = [:topics, name]
265 | @body, @title = topic(File.join(documentation_path, path))
266 | @related_commands = related_commands_for(name)
267 |
268 | custom_render("topics/name")
269 | else
270 | not_found(path: path)
271 | end
272 | end
273 |
274 | on get, "deploy" do
275 | if ENV["DEPLOY_TOKEN"] && req.GET["token"] == ENV["DEPLOY_TOKEN"]
276 | FileUtils.touch("deploy.txt")
277 | else
278 | res.status = 401
279 | end
280 | end
281 |
282 | on get, extension("json") do |file|
283 | res.headers["Cache-Control"] = "public, max-age=29030400" if req.query_string =~ /[0-9]{10}/
284 | res.headers["Content-Type"] = "application/json;charset=UTF-8"
285 | res.write File.read(documentation_path + "/#{file}.json")
286 | end
287 |
288 | on get, extension("js") do |file|
289 | res.headers["Cache-Control"] = "public, max-age=29030400" if req.query_string =~ /[0-9]{10}/
290 | res.headers["Content-Type"] = "text/javascript; charset=utf-8"
291 | res.write File.read("views/#{file}.js")
292 | end
293 |
294 | on post, "commits/payload" do
295 | update_redis_versions
296 | end
297 | end
298 | end
299 |
300 | Cuba.define {
301 | begin
302 | run App
303 | rescue Exception => e
304 | res.status = 500
305 | res.write "I'm sorry, Dave. I'm afraid I can't do that."
306 | end
307 | }
308 |
--------------------------------------------------------------------------------
/config.ru:
--------------------------------------------------------------------------------
1 | require File.expand_path("app", File.dirname(__FILE__))
2 |
3 | run Cuba
4 |
--------------------------------------------------------------------------------
/deploy:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | while true; do
4 | inotifywait deploy.txt
5 | make deploy
6 | done
7 |
--------------------------------------------------------------------------------
/downloads.json:
--------------------------------------------------------------------------------
1 | {
2 | "channels": {
3 | "unstable": {
4 | "branch": "unstable",
5 | "notes": "This is where all the development happens. Only for hard-core hackers. Use only if you need to test the latest features or performance improvements. This is going to be the next Redis release in a few months."
6 | },
7 |
8 | "release-candidate": {
9 | "branch": "7.0",
10 | "version": "7.0-rc2",
11 | "notes": "Redis 7.0 includes several new user-facing features, significant performance optimizations, and many other improvements. It also includes changes that potentially break backwards compatibility with older versions."
12 | },
13 |
14 | "stable": {
15 | "branch": "6.2",
16 | "version": "6.2.6",
17 | "notes": "Redis 6.2 includes many new commands and improvements, but no big features. It mainly makes Redis more complete and addresses issues that have been requested by many users frequently or for a long time."
18 | },
19 |
20 | "docker": {
21 | "name": "Docker Hub",
22 | "url": "https://hub.docker.com/_/redis/",
23 | "notes": "It is possible to get Docker images of Redis from the Docker Hub. Multiple versions are available, usually updated in a short time after a new release is available."
24 | },
25 |
26 | "cloud": {
27 | "name": "In the Cloud",
28 | "url": "https://redis.com/try-free",
29 | "notes": "Get a free-for-life Redis instance with Redis Cloud Essentials."
30 | }
31 | },
32 |
33 | "other": {
34 | "old": {
35 | "branch": "6.0",
36 | "version": "6.0.16",
37 | "notes": "Redis 6.0 introduces SSL, the new RESP3 protocol, ACLs, client side caching, diskless replicas, I/O threads, faster RDB loading, new modules APIs and many more improvements."
38 | },
39 |
40 | "older": {
41 | "branch": "5.0",
42 | "version": "5.0.14",
43 | "notes": "Redis 5.0 is the first version of Redis to introduce the new stream data type with consumer groups, sorted sets blocking pop operations, LFU/LRU info in RDB, Cluster manager inside redis-cli, active defragmentation V2, HyperLogLogs improvements and many other improvements. Redis 5 was release as GA in October 2018."
44 | }
45 | }
46 | }
47 |
--------------------------------------------------------------------------------
/lib/interactive/commands.rb:
--------------------------------------------------------------------------------
1 | # This file is automatically generated by:
2 | # scripts/generate_interactive_commands.rb.
3 | #
4 | # Do not edit.
5 | #
6 |
7 | module Interactive
8 | COMMANDS = {
9 | # connection
10 | "client info" => [],
11 | "echo" => [:zip, [nil]],
12 | "ping" => [:custom],
13 |
14 | # generic
15 | "del" => [:all],
16 | "dump" => [:first],
17 | "exists" => [:all],
18 | "expire" => [:first],
19 | "expireat" => [:first],
20 | "keys" => [:zip, [nil]],
21 | "persist" => [:first],
22 | "pexpire" => [:first],
23 | "pexpireat" => [:first],
24 | "pttl" => [:first],
25 | "rename" => [:all],
26 | "renamenx" => [:all],
27 | "scan" => [:custom],
28 | "sort" => [:custom],
29 | "touch" => [:all],
30 | "ttl" => [:first],
31 | "type" => [:first],
32 | "unlink" => [:all],
33 | "wait" => [:zip, [nil, nil]],
34 |
35 | # geo
36 | "geoadd" => [:first],
37 | "geodist" => [:first],
38 | "geohash" => [:first],
39 | "geopos" => [:first],
40 | "georadius" => [:custom],
41 | "georadiusbymember" => [:custom],
42 | "geosearch" => [:first],
43 | "geosearchstore" => [:custom],
44 |
45 | # hash
46 | "hdel" => [:first],
47 | "hexists" => [:first],
48 | "hget" => [:first],
49 | "hgetall" => [:first],
50 | "hincrby" => [:first],
51 | "hincrbyfloat" => [:first],
52 | "hkeys" => [:first],
53 | "hlen" => [:first],
54 | "hmget" => [:first],
55 | "hmset" => [:first],
56 | "hscan" => [:first],
57 | "hset" => [:first],
58 | "hsetnx" => [:first],
59 | "hstrlen" => [:first],
60 | "hvals" => [:first],
61 |
62 | # hyperloglog
63 | "pfadd" => [:first],
64 | "pfcount" => [:all],
65 | "pfmerge" => [:all],
66 |
67 | # list
68 | "blmove" => [:zip, [:key, :key, nil, nil, nil]],
69 | "lindex" => [:first],
70 | "linsert" => [:first],
71 | "llen" => [:first],
72 | "lmove" => [:zip, [:key, :key, nil, nil]],
73 | "lpop" => [:first],
74 | "lpos" => [:first],
75 | "lpush" => [:first],
76 | "lpushx" => [:first],
77 | "lrange" => [:first],
78 | "lrem" => [:first],
79 | "lset" => [:first],
80 | "ltrim" => [:first],
81 | "rpop" => [:first],
82 | "rpoplpush" => [:all],
83 | "rpush" => [:first],
84 | "rpushx" => [:first],
85 |
86 | # server
87 | "command" => [],
88 | "command count" => [],
89 | "command getkeys" => [],
90 | "command info" => [:custom],
91 | "info" => [:custom],
92 | "lastsave" => [],
93 | "role" => [],
94 | "time" => [],
95 |
96 | # set
97 | "sadd" => [:first],
98 | "scard" => [:first],
99 | "sdiff" => [:all],
100 | "sdiffstore" => [:all],
101 | "sinter" => [:all],
102 | "sinterstore" => [:all],
103 | "sismember" => [:first],
104 | "smembers" => [:first],
105 | "smismember" => [:first],
106 | "smove" => [:zip, [:key, :key, nil]],
107 | "spop" => [:first],
108 | "srandmember" => [:first],
109 | "srem" => [:first],
110 | "sscan" => [:first],
111 | "sunion" => [:all],
112 | "sunionstore" => [:all],
113 |
114 | # sorted-set
115 | "zadd" => [:first],
116 | "zcard" => [:first],
117 | "zcount" => [:first],
118 | "zdiff" => [:custom],
119 | "zdiffstore" => [:custom],
120 | "zincrby" => [:first],
121 | "zinter" => [:custom],
122 | "zinterstore" => [:custom],
123 | "zlexcount" => [:first],
124 | "zmscore" => [:first],
125 | "zpopmax" => [:first],
126 | "zpopmin" => [:first],
127 | "zrange" => [:first],
128 | "zrangebylex" => [:first],
129 | "zrangebyscore" => [:first],
130 | "zrank" => [:first],
131 | "zrem" => [:first],
132 | "zremrangebylex" => [:first],
133 | "zremrangebyrank" => [:first],
134 | "zremrangebyscore" => [:first],
135 | "zrevrange" => [:first],
136 | "zrevrangebylex" => [:first],
137 | "zrevrangebyscore" => [:first],
138 | "zrevrank" => [:first],
139 | "zscan" => [:first],
140 | "zscore" => [:first],
141 | "zunion" => [:custom],
142 | "zunionstore" => [:custom],
143 |
144 | # stream
145 | "xadd" => [:first],
146 | "xdel" => [:first],
147 | "xlen" => [:first],
148 | "xrange" => [:first],
149 | "xrevrange" => [:first],
150 | "xtrim" => [:first],
151 |
152 | # string
153 | "append" => [:first],
154 | "bitcount" => [:first],
155 | "bitfield" => [:first],
156 | "bitop" => [:custom],
157 | "bitpos" => [:first],
158 | "decr" => [:first],
159 | "decrby" => [:first],
160 | "get" => [:first],
161 | "getbit" => [:first],
162 | "getrange" => [:first],
163 | "getset" => [:first],
164 | "incr" => [:first],
165 | "incrby" => [:first],
166 | "incrbyfloat" => [:first],
167 | "mget" => [:all],
168 | "mset" => [:zip, [:key, nil]],
169 | "msetnx" => [:zip, [:key, nil]],
170 | "psetex" => [:first],
171 | "set" => [:first],
172 | "setbit" => [:first],
173 | "setex" => [:first],
174 | "setnx" => [:first],
175 | "setrange" => [:first],
176 | "stralgo" => [:custom],
177 | "strlen" => [:first],
178 |
179 | }.freeze
180 |
181 | SUBCOMMANDS = {
182 | "client" => 1,
183 |
184 | }.freeze
185 | end
186 |
187 |
--------------------------------------------------------------------------------
/lib/interactive/namespace.rb:
--------------------------------------------------------------------------------
1 | require File.expand_path(File.dirname(__FILE__) + "/commands")
2 |
3 | module Interactive
4 |
5 | def self.namespace(ns, args)
6 | pattern(args).map do |arg,type|
7 | if type == :key || type == :ns
8 | [ns, arg].join(":")
9 | else
10 | arg
11 | end
12 | end
13 | end
14 |
15 | def self.keys(args)
16 | pattern(args).map do |arg,type|
17 | arg if type == :key
18 | end.compact
19 | end
20 |
21 | def self.command_name(args)
22 | name = args.shift
23 | subcommands = SUBCOMMANDS[name.downcase]
24 | if not subcommands.nil?
25 | name = name + " " + args.shift(subcommands).join(" ")
26 | end
27 | return name
28 | end
29 |
30 | def self.pattern(args)
31 | args = args.dup
32 | name = command_name(args)
33 | return [] if COMMANDS[name.downcase].nil?
34 | type, pattern = COMMANDS[name.downcase]
35 | out = []
36 |
37 | case type
38 | when :first
39 | out[0] = :key
40 | when :all
41 | out = args.size.times.map { :key }
42 | when :zip
43 | out = args.zip(pattern.cycle).map do |arg, type|
44 | :key if type == :key
45 | end
46 | when :custom
47 | case name.downcase
48 | when "info", "geoencode", "ping"
49 | # Commands without keys
50 | nil
51 | when "bitop"
52 | # BITOP arg positions 1,2,3 are always keys
53 | out[1,3] = 3.times.map{:key}
54 | when "zunionstore", "zinterstore"
55 | # Destination key
56 | if args.size >= 1
57 | out[0] = :key
58 |
59 | # Number of input keys
60 | if args.size >= 3
61 | numkeys = args[1].to_i
62 | out[2,numkeys] = numkeys.times.map { :key }
63 | end
64 | end
65 | when "sort"
66 | tmpargs = args.dup
67 |
68 | # Key to sort
69 | if !tmpargs.empty?
70 | tmpargs.shift
71 | out << :key
72 |
73 | while keyword = tmpargs.shift
74 | out << nil
75 | if !tmpargs.empty?
76 | case keyword.downcase
77 | when "get"
78 | if tmpargs.shift == "#"
79 | out << nil
80 | else
81 | out << :key
82 | end
83 | when "by", "store"
84 | tmpargs.shift
85 | out << :key
86 | when "limit"
87 | break if tmpargs.size < 2
88 | 2.times { tmpargs.shift; out << nil }
89 | end
90 | end
91 | end
92 | end
93 | when "geosearchstore"
94 | out = [:key, :key]
95 | when "zdiff", "zinter", "zunion"
96 | numkeys = args[0].to_i
97 | out[1,numkeys] = numkeys.times.map { :key }
98 | when "zdiffstore"
99 | numkeys = args[1].to_i
100 | out[0] = :key
101 | out[2,numkeys] = numkeys.times.map { :key }
102 | when "zrangestore"
103 | out = [:key, :key]
104 | when "georadius","georadiusbymember"
105 | tmpargs = args.dup
106 |
107 | # First key with the sorted set
108 | if !tmpargs.empty?
109 | tmpargs.shift
110 | out << :key
111 |
112 | while keyword = tmpargs.shift
113 | out << nil
114 | if !tmpargs.empty?
115 | case keyword.downcase
116 | when "store", "storedist"
117 | tmpargs.shift
118 | out << :key
119 | end
120 | end
121 | end
122 | end
123 | else
124 | raise "Don't know what to do for \"#{name.downcase}\""
125 | end
126 | end
127 |
128 | # Hack KEYS command
129 | if name.downcase == "keys"
130 | out[0] = :ns
131 | end
132 |
133 | cmd = name.split(" ")
134 | out = args.zip(out).to_a
135 | [*cmd, *out]
136 | end
137 | end
138 |
139 |
--------------------------------------------------------------------------------
/lib/interactive/redis.rb:
--------------------------------------------------------------------------------
1 | module Interactive
2 |
3 | class LineReply < String; end
4 | class StatusReply < LineReply; end
5 | class ErrorReply < LineReply; end
6 |
7 | module RedisHacks
8 |
9 | def format_status_reply(line)
10 | StatusReply.new(line.strip)
11 | end
12 |
13 | def format_error_reply(line)
14 | ErrorReply.new(line.strip)
15 | end
16 | end
17 |
18 | def self.redis
19 | @redis ||=
20 | begin
21 | redis = new_redis_connection
22 | class << redis._client.connection
23 | include RedisHacks
24 | end
25 | redis
26 | end
27 | end
28 | end
29 |
30 |
--------------------------------------------------------------------------------
/lib/interactive/session.rb:
--------------------------------------------------------------------------------
1 | require File.expand_path(File.dirname(__FILE__) + "/redis")
2 |
3 | module Interactive
4 |
5 | UNESCAPES = {
6 | 'a' => "\x07", 'b' => "\x08", 't' => "\x09",
7 | 'n' => "\x0a", 'v' => "\x0b", 'f' => "\x0c",
8 | 'r' => "\x0d", 'e' => "\x1b", "\\\\" => "\x5c",
9 | "\"" => "\x22", "'" => "\x27"
10 | }
11 |
12 | # Create and find actions have race conditions, but I don't
13 | # really care about them right now...
14 | class Session
15 |
16 | TIMEOUT = 3600
17 |
18 | # Create new instance.
19 | def self.create(namespace)
20 | raise "Already exists" if redis.zscore("sessions", namespace)
21 | touch(namespace)
22 | new(namespace)
23 | end
24 |
25 | # Return instance if namespace exists in sorted set.
26 | def self.find(namespace)
27 | if timestamp = redis.zscore("sessions", namespace)
28 | if Time.now.to_i - timestamp.to_i < TIMEOUT
29 | touch(namespace)
30 | new(namespace)
31 | end
32 | end
33 | end
34 |
35 | # Try to clean up old sessions
36 | def self.clean!
37 | now = Time.now.to_i
38 | threshold = now - TIMEOUT
39 | namespace, timestamp = redis.zrangebyscore("sessions", "-inf", threshold,
40 | :with_scores => true, :limit => [0, 1])
41 | return if namespace.nil?
42 |
43 | if redis.zrem("sessions", namespace)
44 | keys = redis.smembers("session:#{namespace}:keys")
45 | redis.del(*keys.map { |key| "#{namespace}:#{key}" }) if !keys.empty?
46 | redis.del("session:#{namespace}:keys")
47 | redis.del("session:#{namespace}:commands")
48 | end
49 | end
50 |
51 | # This should only be created through #new or #create
52 | private_class_method :new
53 |
54 | attr :namespace
55 |
56 | def initialize(namespace)
57 | @namespace = namespace
58 | self.class.clean!
59 | end
60 |
61 | def run(line)
62 | _run(line)
63 | rescue => error
64 | format_reply(ErrorReply.new("ERR " + error.message))
65 | end
66 |
67 | private
68 |
69 | def self.touch(namespace)
70 | redis.zadd("sessions", Time.now.to_i, namespace)
71 | end
72 |
73 | def register(arguments)
74 | # TODO: Only store keys that receive writes.
75 | keys = ::Interactive.keys(arguments)
76 | redis.pipelined do
77 | keys.each do |key|
78 | redis.sadd("session:#{namespace}:keys", key)
79 | end
80 | end
81 |
82 | # Command counter, not yet used
83 | redis.incr("session:#{namespace}:commands")
84 | end
85 |
86 | def unescape_literal(str)
87 | # Escape all the things
88 | str.gsub(/\\(?:([#{UNESCAPES.keys.join}])|u([\da-fA-F]{4}))|\\0?x([\da-fA-F]{2})/) {
89 | if $1
90 | if $1 == '\\' then '\\' else UNESCAPES[$1] end
91 | elsif $2 # escape \u0000 unicode
92 | ["#$2".hex].pack('U*')
93 | elsif $3 # escape \0xff or \xff
94 | [$3].pack('H2')
95 | end
96 | }
97 | end
98 |
99 | # Parse a command line style list of arguments that can be optionally
100 | # delimited by '' or "" quotes, and return it as an array of arguments.
101 | #
102 | # Strings delimited by "" are unescaped by converting escape characters
103 | # such as \n \x.. to their value according to the unescape_literal()
104 | # function.
105 | #
106 | # Example of line that this function can parse:
107 | #
108 | # "Hello World\n" other arguments 'this is a single argument'
109 | #
110 | # The above example will return an array of four strings.
111 | def cli_split(line)
112 | argv = []
113 | arg = ""
114 | inquotes = false
115 | pos = 0
116 | while pos < line.length
117 | char = line[pos..pos] # Current character
118 | isspace = char =~ /\s/
119 |
120 | # Skip empty spaces if we are between strings
121 | if !inquotes && isspace
122 | if arg.length != 0
123 | argv << arg
124 | arg = ""
125 | end
126 | pos += 1
127 | next
128 | end
129 |
130 | # Append current char to string
131 | arg << char
132 | pos += 1
133 |
134 | if arg.length == 1 && (char == '"' || char == '\'')
135 | inquotes = char
136 | elsif arg.length > 1 && inquotes && char == inquotes
137 | inquotes = false
138 | end
139 | end
140 | # Put the last argument into the array
141 | argv << arg if arg.length != 0
142 |
143 | # We need to make some post-processing.
144 | # For strings delimited by '' we just strip initial and final '.
145 | # For strings delimited by "" we call unescape_literal().
146 | # This is not perfect but should be enough for redis.io interactive
147 | # editing.
148 | argv.map {|x|
149 | if x[0..0] == '"'
150 | unescape_literal(x[1..-2])
151 | elsif x[0..0] == '\''
152 | x[1..-2]
153 | else
154 | x
155 | end
156 | }
157 | end
158 |
159 | def _run(line)
160 | begin
161 | arguments = cli_split(line)
162 | rescue => error
163 | raise error.message.split(":").first
164 | end
165 |
166 | if arguments.empty?
167 | raise "No command"
168 | end
169 |
170 | if arguments.size > 100 || arguments.any? { |arg| arg.size > 100 }
171 | raise "Web-based interface is limited"
172 | end
173 |
174 | case arguments[0].downcase
175 | when "setbit"
176 | if arguments[2].to_i >= 2048
177 | raise "Web-based interface is limited"
178 | end
179 | when "setrange"
180 | if arguments[2].to_i + arguments[3].to_s.size >= 256
181 | raise "Web-based interface is limited"
182 | end
183 | end
184 |
185 | namespaced = ::Interactive.namespace(namespace, arguments)
186 | if namespaced.empty?
187 | raise "Unknown or disabled command '%s'" % arguments.first
188 | end
189 |
190 | # Register the call
191 | register(arguments)
192 |
193 | # Make the call
194 | reply = ::Interactive.redis.call(*namespaced)
195 |
196 | case arguments.first.downcase
197 | when "keys"
198 | # Strip namespace for KEYS
199 | if reply.respond_to?(:map)
200 | format_reply(reply.map { |key| key[/^\w+:(.*)$/,1] })
201 | else
202 | format_reply(reply)
203 | end
204 | when "info"
205 | # Don't #inspect the string reply for INFO
206 | reply.to_s
207 | else
208 | format_reply(reply)
209 | end
210 | end
211 |
212 | def format_reply(reply, prefix = "")
213 | case reply
214 | when LineReply
215 | reply.to_s + "\n"
216 | when Integer
217 | "(integer) " + reply.to_s + "\n"
218 | when String
219 | reply.inspect + "\n"
220 | when NilClass
221 | "(nil)\n"
222 | when Array
223 | if reply.empty?
224 | "(empty list or set)\n"
225 | else
226 | out = ""
227 | index_size = reply.size.to_s.size
228 | reply.each_with_index do |element, index|
229 | out << prefix if index > 0
230 | out << "%#{index_size}d) " % (index + 1)
231 | out << format_reply(element, prefix + (" " * (index_size + 2)))
232 | end
233 | out
234 | end
235 | else
236 | raise "Don't know how to format #{reply.inspect}"
237 | end
238 | end
239 | end
240 | end
241 |
242 |
--------------------------------------------------------------------------------
/lib/reference.rb:
--------------------------------------------------------------------------------
1 | class Reference
2 | GROUPS = {
3 | "generic" => "Keys",
4 | "string" => "Strings",
5 | "hash" => "Hashes",
6 | "list" => "Lists",
7 | "set" => "Sets",
8 | "sorted-set" => "Sorted Sets",
9 | "hyperloglog" => "HyperLogLog",
10 | "pubsub" => "Pub/Sub",
11 | "transactions" => "Transactions",
12 | "scripting" => "Scripting",
13 | "connection" => "Connection",
14 | "server" => "Server",
15 | "cluster" => "Cluster",
16 | "geo" => "Geo",
17 | "stream" => "Streams",
18 | "bitmap" => "Bitmaps",
19 | "cluster" => "Cluster",
20 | "sentinel" => "Sentinel"
21 | }
22 |
23 | class Command
24 | class Argument
25 | attr :argument
26 |
27 | def initialize(argument)
28 | @argument = argument
29 | end
30 |
31 | def type
32 | argument["type"]
33 | end
34 |
35 | def optional?
36 | argument["optional"] || false
37 | end
38 |
39 | def multiple?
40 | argument["multiple"] || false
41 | end
42 |
43 | def multiple_token?
44 | argument["multiple_token"] || false
45 | end
46 |
47 | def to_s
48 | if type == "block"
49 | res = block(argument)
50 | elsif type == "oneof"
51 | res = oneof(argument)
52 | elsif type != "pure-token"
53 | res = argument["name"]
54 | else
55 | res = ""
56 | end
57 |
58 | token = argument["token"]
59 | if token == ""
60 | token = "\"\""
61 | end
62 | if multiple_token?
63 | res = "#{res} [#{token} #{res} ...]"
64 | elsif multiple?
65 | res = "#{res} [#{res} ...]"
66 | end
67 |
68 | if token
69 | res = "#{token} #{res}"
70 | res = res.strip! || res
71 | end
72 |
73 | optional? ? "[#{res}]" : res
74 | end
75 |
76 | private
77 |
78 | def block(argument)
79 | argument["arguments"].map do |entry|
80 | Argument.new(entry)
81 | end.join(" ")
82 | end
83 |
84 | def oneof(argument)
85 | argument["arguments"].map do |entry|
86 | Argument.new(entry)
87 | end.join("|")
88 | end
89 | end
90 |
91 | attr :name
92 | attr :command
93 | attr :group
94 |
95 | def initialize(name, command)
96 | @name = name
97 | @command = command
98 | end
99 |
100 | def to_s
101 | @to_s ||= [name, *arguments].join(" ")
102 | end
103 |
104 | def since
105 | command["since"]
106 | end
107 |
108 | def group
109 | command["group"]
110 | end
111 |
112 | def complexity
113 | command["complexity"]
114 | end
115 |
116 | def deprecated_since
117 | command["deprecated_since"]
118 | end
119 |
120 | def replaced_by
121 | command["replaced_by"]
122 | end
123 |
124 | def history
125 | command["history"]
126 | end
127 |
128 | def is_helpsubcommand?
129 | name.downcase.end_with?(" help")
130 | end
131 |
132 | def is_purecontainer?
133 | command["arity"] == -2 && !command["arguments"]
134 | end
135 |
136 | def is_listed?
137 | !is_purecontainer? && !is_helpsubcommand?
138 | end
139 |
140 | def to_param
141 | name.downcase.gsub(" ", "-")
142 | end
143 |
144 | def arguments
145 | (command["arguments"] || []).map do |argument|
146 | Argument.new(argument)
147 | end
148 | end
149 |
150 | include Comparable
151 |
152 | def ==(other)
153 | name == other.name
154 | end
155 | alias eql? ==
156 |
157 | def hash
158 | name.hash
159 | end
160 | end
161 |
162 | include Enumerable
163 |
164 | def initialize(commands)
165 | @commands = commands
166 | end
167 |
168 | def [](name)
169 | Command.new(name, @commands[name]) if @commands[name]
170 | end
171 |
172 | def each
173 | @commands.each do |name, attrs|
174 | yield Command.new(name, attrs)
175 | end
176 | end
177 |
178 | def sample
179 | key = @commands.keys.sample
180 |
181 | Command.new(key, @commands[key])
182 | end
183 | end
184 |
--------------------------------------------------------------------------------
/lib/template.rb:
--------------------------------------------------------------------------------
1 | require "tilt/redcarpet"
2 |
3 | class RedisTemplate < Tilt::Redcarpet2Template
4 | SECTIONS = {
5 | "description" => "Description",
6 | "examples" => "Examples",
7 | "return" => "Return value",
8 | "history" => "History"
9 | }
10 |
11 | REPLY_TYPES = {
12 | "nil" => "Null reply",
13 | "simple-string" => "Simple string reply",
14 | "integer" => "Integer reply",
15 | "bulk-string" => "Bulk string reply",
16 | "array" => "Array reply"
17 | }
18 |
19 | def sections(source)
20 | source.gsub(/^\@(\w+)$/) do
21 | title = SECTIONS[$1]
22 | "## #{title}\n"
23 | end
24 | end
25 |
26 | # Prefix commands that should *not* be autolinked with "!".
27 | def autolink_commands(source)
28 | source.gsub(/\B`(!?[A-Z\- ]+)`\B/) do
29 | name = $1
30 | command = commands[name]
31 |
32 | if command
33 | "[#{name}](/commands/#{name.downcase.gsub(' ', '-')})"
34 | else
35 | name.gsub!(/^!/, "")
36 | "`#{name}`"
37 | end
38 | end
39 | end
40 |
41 | def reply_types(source)
42 | source.gsub(/@(#{REPLY_TYPES.keys.join("|")})\-reply/) do
43 | type = $1
44 | "[#{REPLY_TYPES[type]}](/topics/protocol##{type}-reply)"
45 | end
46 | end
47 |
48 | def formulas(source)
49 | source.gsub(/(O\(.+?\)[\+\s\.,])/) do
50 | %Q[#{$1}]
51 | end
52 | end
53 |
54 | def erb(data)
55 | ERB.new(data).result(binding)
56 | end
57 |
58 | def preprocess(data)
59 | data = erb(data)
60 | data = sections(data)
61 | data = autolink_commands(data)
62 | data = reply_types(data)
63 | data = formulas(data)
64 | data
65 | end
66 |
67 | def prepare
68 | @data = preprocess(@data)
69 | super
70 | end
71 | end
72 |
73 | Tilt.register "md", RedisTemplate
74 |
--------------------------------------------------------------------------------
/makefile:
--------------------------------------------------------------------------------
1 | TEST_FILES=$(shell find test -name '*.rb')
2 |
3 | test:
4 | test -x redis-doc || git clone https://github.com/redis/redis-doc
5 | cutest $(TEST_FILES)
6 |
7 | deploy:
8 | cd /srv/redis-doc && git pull
9 | cd /srv/redis-io && git stash && git pull
10 | # bash --login -c "cd /srv/redis-io && rvm use 2.7.0 && REDIS_DOC=/srv/redis-doc /srv/redis-io/scripts/generate_interactive_commands.rb > /srv/redis-io/lib/interactive/commands.rb"
11 | service redis-io-app restart
12 |
13 | .PHONY: deploy test
14 |
--------------------------------------------------------------------------------
/public/app.js:
--------------------------------------------------------------------------------
1 | !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.Slideout=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o self._tolerance) ? self.open() : self.close();
182 | }
183 | self._moved = false;
184 | });
185 |
186 | /**
187 | * Translates panel on touchmove
188 | */
189 | this.panel.addEventListener(touch.move, function(eve) {
190 |
191 | if (scrolling || self._preventOpen) { return; }
192 |
193 | var dif_x = eve.touches[0].clientX - self._startOffsetX;
194 | var translateX = self._currentOffsetX = dif_x;
195 |
196 | if (Math.abs(translateX) > self._padding) { return; }
197 |
198 | if (Math.abs(dif_x) > 20) {
199 | self._opening = true;
200 |
201 | if (self._opened && dif_x > 0 || !self._opened && dif_x < 0) { return; }
202 |
203 | if (!self._moved && html.className.search('slideout-open') === -1) {
204 | html.className += ' slideout-open';
205 | }
206 |
207 | if (dif_x <= 0) {
208 | translateX = dif_x + self._padding;
209 | self._opening = false;
210 | }
211 |
212 | self.panel.style[prefix + 'transform'] = self.panel.style.transform = 'translate3d(' + translateX + 'px, 0, 0)';
213 |
214 | self._moved = true;
215 | }
216 |
217 | });
218 |
219 | };
220 |
221 | /**
222 | * Expose Slideout
223 | */
224 | module.exports = Slideout;
225 |
226 | },{"decouple":2}],2:[function(require,module,exports){
227 | 'use strict';
228 |
229 | var requestAnimFrame = (function() {
230 | return window.requestAnimationFrame ||
231 | window.webkitRequestAnimationFrame ||
232 | function (callback) {
233 | window.setTimeout(callback, 1000 / 60);
234 | };
235 | }());
236 |
237 | function decouple(node, event, fn) {
238 | var eve,
239 | tracking = false;
240 |
241 | function captureEvent(e) {
242 | eve = e;
243 | track();
244 | }
245 |
246 | function track() {
247 | if (!tracking) {
248 | requestAnimFrame(update);
249 | tracking = true;
250 | }
251 | }
252 |
253 | function update() {
254 | fn.call(node, eve);
255 | tracking = false;
256 | }
257 |
258 | node.addEventListener(event, captureEvent, false);
259 | }
260 |
261 | /**
262 | * Expose decouple
263 | */
264 | module.exports = decouple;
265 |
266 | },{}]},{},[1])(1)
267 | });
268 |
269 |
270 |
271 | ;(function($) {
272 |
273 | var cssPrefix = null;
274 |
275 | if ($.browser.mozilla) cssPrefix = "moz";
276 | else if ($.browser.webkit) cssPrefix = "webkit";
277 | else if ($.browser.opera) cssPrefix = "o";
278 |
279 | $.cssHooks["columnCount"] = {
280 | get: function(element, computed) {
281 | var browserSpecificName = "-" + cssPrefix + "-column-count";
282 |
283 | if (computed) {
284 | return $.css(element, browserSpecificName);
285 | }
286 | else {
287 | return element.style[browserSpecificName];
288 | }
289 | }
290 | }
291 |
292 | function commandReference() {
293 | var $groups = $("#commands nav a")
294 |
295 | $groups.click(function() {
296 | window.location.hash = this.getAttribute("href").substring(1)
297 |
298 | filterCommandReference()
299 |
300 | return false
301 | })
302 |
303 | var filter = document.querySelector('.command-reference-filter');
304 |
305 | filter.addEventListener('change', function(e) {
306 | window.location.hash = e.target.value;
307 | });
308 |
309 | window.onhashchange = function() {
310 | filterCommandReference();
311 | }
312 | }
313 |
314 | function filterCommandReference() {
315 | var $commands = $("#commands ul")
316 |
317 | var group = window.location.hash.substring(1)
318 |
319 | if (group.length == 0) {
320 | $commands.children().show()
321 | // $commands.css("height", "auto")
322 | }
323 | else {
324 | $commands.find("li[data-group='" + group + "']").show()
325 | $commands.find("li[data-group!='" + group + "']").hide()
326 | }
327 |
328 | var $groups = $("#commands nav a")
329 |
330 | $groups.removeClass("current")
331 |
332 | $groups.filter("[href='#" + group + "']").addClass("current")
333 |
334 | document.querySelector('.command-reference-filter').value = group;
335 | }
336 |
337 | function autolink(text) {
338 | return text.replace(/(https?:\/\/[-\w\.]+:?\/[\w\/_\-\.]*(\?\S+)?)/, "$1");
339 | }
340 |
341 | function massageTweet(text) {
342 | text = text.replace(/^.* @\w+: /, "");
343 |
344 | return autolink(text);
345 | }
346 |
347 | function searchCommandReference() {
348 | var $commands = $('li')
349 |
350 | $('.js-command-reference-search').bind('input', function(ev) {
351 | window.location.hash = '';
352 |
353 | if (ev.keyCode === 13) {
354 | var name = $commands.filter(':visible')[0].getAttribute('data-name');
355 |
356 | window.location = '/commands/' + name.replace(/ /g, '-');
357 |
358 | return;
359 | }
360 |
361 | var val = $(this).val().toLowerCase().replace(/[^a-z0-9 ]/g, '');
362 |
363 | if (val === '') {
364 | $commands.show()
365 | } else {
366 | $commands.hide()
367 | $('li[data-name*="' + val + '"]').show()
368 | }
369 | })
370 | }
371 |
372 | function buzz() {
373 | var $buzz = $("#buzz");
374 |
375 | if ($buzz.length == 0) return;
376 |
377 | var $ul = $buzz.find("ul");
378 | var count = 0;
379 | var limit = parseInt($buzz.attr("data-limit"));
380 | var users = {};
381 |
382 | $.getJSON("https://redis-buzz.herokuapp.com/?callback=?", function(response) {
383 | $.each(response, function() {
384 |
385 | if (count++ == limit) { return false; }
386 |
387 | if (this.retweeted_status) {
388 | var status = this.retweeted_status;
389 | } else {
390 | var status = this;
391 | }
392 |
393 | $ul.append(
394 | "" +
395 | "" +
396 | "
" +
397 | " " +
398 | massageTweet(status.text) +
399 | " "
400 | );
401 | });
402 | });
403 | }
404 |
405 | // Easily set caret position in input field
406 | $.fn.setSelection = function(start, end) {
407 | var i, size = this.size();
408 |
409 | // Only set caret by default
410 | if (end === undefined) end = start;
411 |
412 | for (i = 0; i < size; i++) {
413 | var element = this.get(i);
414 | if (element.createTextRange) { // IE
415 | var range = element.createTextRange();
416 | range.collapse(true);
417 | range.moveEnd('character', end);
418 | range.moveStart('character', start);
419 | range.select();
420 | } else if (element.setSelectionRange) { // Other browsers
421 | element.setSelectionRange(start, end);
422 | }
423 | }
424 | }
425 |
426 | function examples() {
427 | $('div.example').each(function() {
428 | var $example = $(this);
429 | var $form = $example.find("form");
430 | var $input = $form.find("input");
431 |
432 | $input.keydown(function(event) {
433 | var count = $example.find(".command").size();
434 | var index = $input.data("index");
435 | if (index == undefined) index = count;
436 |
437 | if (event.keyCode == 38) {
438 | index--; // up
439 | } else if (event.keyCode == 40) {
440 | index++; // down
441 | } else {
442 | return;
443 | }
444 |
445 | // Out of range at the positive side of the range makes sure
446 | // we can get back to an empty value.
447 | if (index >= 0 && index <= count) {
448 | $input.data("index", index);
449 | $input.val($example.find(".command").eq(index).text());
450 | $input.setSelection($input.val().length);
451 | }
452 |
453 | return false;
454 | });
455 |
456 | $form.submit(function(event) {
457 | if ($input.val().length == 0)
458 | return false;
459 |
460 | // Append command to execute
461 | var ps1 = $("")
462 | .addClass("monospace")
463 | .addClass("prompt")
464 | .html("redis> ");
465 | var cmd = $("")
466 | .addClass("monospace")
467 | .addClass("command")
468 | .text($input.val());
469 | $form.before(ps1);
470 | $form.before(cmd);
471 |
472 | // Hide form
473 | $form.hide();
474 |
475 | // POST command to app
476 | $.ajax({
477 | type: "post",
478 | url: "/session/" + $example.attr("data-session"),
479 | data: $form.serialize(),
480 | complete: function(xhr, textStatus) {
481 | var data = xhr.responseText;
482 | var pre = $("").text(data);
483 | $form.before(pre);
484 |
485 | // Reset input field and show form
486 | $input.val("");
487 | $input.removeData("index");
488 | $form.show();
489 | }
490 | });
491 |
492 | return false;
493 | });
494 | });
495 |
496 | // Only focus field when it is visible
497 | var $first = $('div.example:first :text');
498 | if ($first.size() > 0) {
499 | var windowTop = $(window).scrollTop();
500 | var windowBottom = windowTop + $(window).height();
501 | var elemTop = $first.offset().top;
502 | var elemBottom = elemTop + $first.height();
503 | if (elemTop >= windowTop && elemBottom < windowBottom) {
504 | $first.focus();
505 | }
506 | }
507 | }
508 |
509 | $(document).ready(function() {
510 | var slideout = new Slideout({
511 | 'panel': document.querySelector('.site-wrapper'),
512 | 'menu': document.querySelector('.mobile-menu'),
513 | 'padding': 256,
514 | 'tolerance': 70
515 | });
516 |
517 | document.querySelector('.js-slideout-toggle').addEventListener('click', function() {
518 | slideout.toggle();
519 | });
520 |
521 | document.querySelector('.mobile-menu').addEventListener('click', function(eve) {
522 | if (eve.target.nodeName === 'A') { slideout.close(); }
523 | });
524 |
525 | if (document.getElementById('commands')) {
526 | commandReference();
527 | filterCommandReference();
528 | searchCommandReference()
529 | }
530 |
531 | buzz()
532 |
533 | examples()
534 | })
535 |
536 | })(jQuery);
537 |
--------------------------------------------------------------------------------
/public/images/companies/bump.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/companies/bump.png
--------------------------------------------------------------------------------
/public/images/favicon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/favicon.png
--------------------------------------------------------------------------------
/public/images/pivotal.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/pivotal.png
--------------------------------------------------------------------------------
/public/images/redis-300dpi.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redis-300dpi.png
--------------------------------------------------------------------------------
/public/images/redis-gray.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redis-gray.png
--------------------------------------------------------------------------------
/public/images/redis-logo.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
74 |
--------------------------------------------------------------------------------
/public/images/redis-small.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redis-small.png
--------------------------------------------------------------------------------
/public/images/redis-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redis-white.png
--------------------------------------------------------------------------------
/public/images/redis.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redis.png
--------------------------------------------------------------------------------
/public/images/redisdoc/2idx_0.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redisdoc/2idx_0.png
--------------------------------------------------------------------------------
/public/images/redisdoc/2idx_1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redisdoc/2idx_1.png
--------------------------------------------------------------------------------
/public/images/redisdoc/2idx_2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redisdoc/2idx_2.png
--------------------------------------------------------------------------------
/public/images/redisdoc/lru_comparison.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redisdoc/lru_comparison.png
--------------------------------------------------------------------------------
/public/images/redisdoc/pipeline_iops.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redisdoc/pipeline_iops.png
--------------------------------------------------------------------------------
/public/images/redislabs.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/redislabs.png
--------------------------------------------------------------------------------
/public/images/shuttleworth.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/shuttleworth.png
--------------------------------------------------------------------------------
/public/images/vmware.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/images/vmware.png
--------------------------------------------------------------------------------
/public/opensearch.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | redis.io
4 | Look up a Redis command
5 | redis-db@googlegroups.com
6 |
7 | 
8 |
9 | false
10 | en
11 | UTF-8
12 | UTF-8
13 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/public/presentation/Pnoordhuis_whats_new_in_2_2.pdf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/presentation/Pnoordhuis_whats_new_in_2_2.pdf
--------------------------------------------------------------------------------
/public/presentation/Redis_Cluster.pdf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/redis/redis-io/8407239da448d843968d53a412e9c84138a60f1c/public/presentation/Redis_Cluster.pdf
--------------------------------------------------------------------------------
/public/styles.css:
--------------------------------------------------------------------------------
1 | @import url("//fonts.googleapis.com/css?family=Open+Sans:300,400,700");
2 | @import url("//maxcdn.bootstrapcdn.com/font-awesome/4.3.0/css/font-awesome.min.css");
3 | /* normalize.css v3.0.2 | MIT License | git.io/normalize */
4 | /** 1. Set default font family to sans-serif. 2. Prevent iOS text size adjust after orientation change, without disabling user zoom. */
5 | html { font-family: sans-serif; /* 1 */ -ms-text-size-adjust: 100%; /* 2 */ -webkit-text-size-adjust: 100%; /* 2 */ }
6 |
7 | /** Remove default margin. */
8 | body { margin: 0; }
9 |
10 | /* HTML5 display definitions ========================================================================== */
11 | /** Correct `block` display not defined for any HTML5 element in IE 8/9. Correct `block` display not defined for `details` or `summary` in IE 10/11 and Firefox. Correct `block` display not defined for `main` in IE 11. */
12 | article, aside, details, figcaption, figure, footer, header, hgroup, main, menu, nav, section, summary { display: block; }
13 |
14 | /** 1. Correct `inline-block` display not defined in IE 8/9. 2. Normalize vertical alignment of `progress` in Chrome, Firefox, and Opera. */
15 | audio, canvas, progress, video { display: inline-block; /* 1 */ vertical-align: baseline; /* 2 */ }
16 |
17 | /** Prevent modern browsers from displaying `audio` without controls. Remove excess height in iOS 5 devices. */
18 | audio:not([controls]) { display: none; height: 0; }
19 |
20 | /** Address `[hidden]` styling not present in IE 8/9/10. Hide the `template` element in IE 8/9/11, Safari, and Firefox < 22. */
21 | [hidden], template { display: none; }
22 |
23 | /* Links ========================================================================== */
24 | /** Remove the gray background color from active links in IE 10. */
25 | a { background-color: transparent; }
26 |
27 | /** Improve readability when focused and also mouse hovered in all browsers. */
28 | a:active, a:hover { outline: 0; }
29 |
30 | /* Text-level semantics ========================================================================== */
31 | /** Address styling not present in IE 8/9/10/11, Safari, and Chrome. */
32 | abbr[title] { border-bottom: 1px dotted; }
33 |
34 | /** Address style set to `bolder` in Firefox 4+, Safari, and Chrome. */
35 | b, strong { font-weight: bold; }
36 |
37 | /** Address styling not present in Safari and Chrome. */
38 | dfn { font-style: italic; }
39 |
40 | /** Address variable `h1` font-size and margin within `section` and `article` contexts in Firefox 4+, Safari, and Chrome. */
41 | h1 { font-size: 2em; margin: 0.67em 0; }
42 |
43 | /** Address styling not present in IE 8/9. */
44 | mark { background: #ff0; color: #000; }
45 |
46 | /** Address inconsistent and variable font size in all browsers. */
47 | small { font-size: 80%; }
48 |
49 | /** Prevent `sub` and `sup` affecting `line-height` in all browsers. */
50 | sub, sup { font-size: 75%; line-height: 0; position: relative; vertical-align: baseline; }
51 |
52 | sup { top: -0.5em; }
53 |
54 | sub { bottom: -0.25em; }
55 |
56 | /* Embedded content ========================================================================== */
57 | /** Remove border when inside `a` element in IE 8/9/10. */
58 | img { border: 0; }
59 |
60 | /** Correct overflow not hidden in IE 9/10/11. */
61 | svg:not(:root) { overflow: hidden; }
62 |
63 | /* Grouping content ========================================================================== */
64 | /** Address margin not present in IE 8/9 and Safari. */
65 | figure { margin: 1em 40px; }
66 |
67 | /** Address differences between Firefox and other browsers. */
68 | hr { -moz-box-sizing: content-box; box-sizing: content-box; height: 0; }
69 |
70 | /** Contain overflow in all browsers. */
71 | pre { overflow: auto; }
72 |
73 | /** Address odd `em`-unit font size rendering in all browsers. */
74 | code, kbd, pre, samp { font-family: monospace, monospace; font-size: 1em; }
75 |
76 | /* Forms ========================================================================== */
77 | /** Known limitation: by default, Chrome and Safari on OS X allow very limited styling of `select`, unless a `border` property is set. */
78 | /** 1. Correct color not being inherited. Known issue: affects color of disabled elements. 2. Correct font properties not being inherited. 3. Address margins set differently in Firefox 4+, Safari, and Chrome. */
79 | button, input, optgroup, select, textarea { color: inherit; /* 1 */ font: inherit; /* 2 */ margin: 0; /* 3 */ }
80 |
81 | /** Address `overflow` set to `hidden` in IE 8/9/10/11. */
82 | button { overflow: visible; }
83 |
84 | /** Address inconsistent `text-transform` inheritance for `button` and `select`. All other form control elements do not inherit `text-transform` values. Correct `button` style inheritance in Firefox, IE 8/9/10/11, and Opera. Correct `select` style inheritance in Firefox. */
85 | button, select { text-transform: none; }
86 |
87 | /** 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio` and `video` controls. 2. Correct inability to style clickable `input` types in iOS. 3. Improve usability and consistency of cursor style between image-type `input` and others. */
88 | button, html input[type="button"], input[type="reset"], input[type="submit"] { -webkit-appearance: button; /* 2 */ cursor: pointer; /* 3 */ }
89 |
90 | /** Re-set default cursor for disabled elements. */
91 | button[disabled], html input[disabled] { cursor: default; }
92 |
93 | /** Remove inner padding and border in Firefox 4+. */
94 | button::-moz-focus-inner, input::-moz-focus-inner { border: 0; padding: 0; }
95 |
96 | /** Address Firefox 4+ setting `line-height` on `input` using `!important` in the UA stylesheet. */
97 | input { line-height: normal; }
98 |
99 | /** It's recommended that you don't attempt to style these elements. Firefox's implementation doesn't respect box-sizing, padding, or width. 1. Address box sizing set to `content-box` in IE 8/9/10. 2. Remove excess padding in IE 8/9/10. */
100 | input[type="checkbox"], input[type="radio"] { box-sizing: border-box; /* 1 */ padding: 0; /* 2 */ }
101 |
102 | /** Fix the cursor style for Chrome's increment/decrement buttons. For certain `font-size` values of the `input`, it causes the cursor style of the decrement button to change from `default` to `text`. */
103 | input[type="number"]::-webkit-inner-spin-button, input[type="number"]::-webkit-outer-spin-button { height: auto; }
104 |
105 | /** 1. Address `appearance` set to `searchfield` in Safari and Chrome. 2. Address `box-sizing` set to `border-box` in Safari and Chrome (include `-moz` to future-proof). */
106 | input[type="search"] { -webkit-appearance: textfield; /* 1 */ -moz-box-sizing: content-box; -webkit-box-sizing: content-box; /* 2 */ box-sizing: content-box; }
107 |
108 | /** Remove inner padding and search cancel button in Safari and Chrome on OS X. Safari (but not Chrome) clips the cancel button when the search input has padding (and `textfield` appearance). */
109 | input[type="search"]::-webkit-search-cancel-button, input[type="search"]::-webkit-search-decoration { -webkit-appearance: none; }
110 |
111 | /** Define consistent border, margin, and padding. */
112 | fieldset { border: 1px solid #c0c0c0; margin: 0 2px; padding: 0.35em 0.625em 0.75em; }
113 |
114 | /** 1. Correct `color` not being inherited in IE 8/9/10/11. 2. Remove padding so people aren't caught out if they zero out fieldsets. */
115 | legend { border: 0; /* 1 */ padding: 0; /* 2 */ }
116 |
117 | /** Remove default vertical scrollbar in IE 8/9/10/11. */
118 | textarea { overflow: auto; }
119 |
120 | /** Don't inherit the `font-weight` (applied by a rule above). NOTE: the default cannot safely be changed in Chrome and Safari on OS X. */
121 | optgroup { font-weight: bold; }
122 |
123 | /* Tables ========================================================================== */
124 | /** Remove most spacing between table cells. */
125 | table { border-collapse: collapse; border-spacing: 0; }
126 |
127 | td, th { padding: 0; }
128 |
129 | body { width: 100%; *zoom: 1; }
130 | body:before, body:after { content: ""; display: table; }
131 | body:after { clear: both; }
132 |
133 | body, textarea { font: 16px/1.6 "Open Sans", Helvetica, sans-serif; }
134 |
135 | .anchor { display: block; position: relative; z-index: -1; top: -45px; }
136 |
137 | h1, h2, h3 { position: relative; z-index: 1; }
138 | h1 a.anchor-link, h2 a.anchor-link, h3 a.anchor-link { display: none; font-family: monospace; font-size: 25px; height: 1em; left: 0; margin-left: 30px; margin-top: 4px; padding-left: 30px; padding-right: 8px; position: absolute; text-align: center; text-decoration: none; width: 1em; }
139 | h1:hover a.anchor-link, h2:hover a.anchor-link, h3:hover a.anchor-link { color: #888888; display: block; margin-left: -30px; padding-left: 8px; }
140 | h1:hover a.anchor-link:hover, h2:hover a.anchor-link:hover, h3:hover a.anchor-link:hover { color: black; }
141 |
142 | h1 { font-size: 24px; margin: 18px 0 2px 0; }
143 |
144 | h2 { font-size: 18px; margin: 14px 0 8px 0; }
145 |
146 | h3 { font-size: 15px; margin: 13px 0 6px 0; }
147 |
148 | @media only screen and (min-width: 992px) { h1 { font-size: 32px; }
149 | h2 { font-size: 22px; }
150 | h3 { font-size: 18px; } }
151 | code { font-family: "Menlo", "Monaco", monospace; }
152 |
153 | a { color: #0066aa; text-decoration: none; }
154 | a:hover { text-decoration: underline; color: #0099ff; }
155 |
156 | .container, .site-content { margin: 0 auto; max-width: 900px; }
157 |
158 | body { background-color: white; color: #333333; padding: 0; margin: 0; }
159 |
160 | .site-footer { color: #777777; font-size: 11px; margin: 20px; border-top: 1px solid #eeeeee; padding: 10px 0 0 0; }
161 | .site-footer a { color: #444444; }
162 |
163 | @media only screen and (min-width: 992px) { .site-footer { background: url(/images/redis-small.png) no-repeat left top; padding: 0 0 0 50px; max-width: 840px; margin: 5em auto 3em auto; position: relative; border-top-width: 0; }
164 | .site-footer p { line-height: 34px; margin: 0; } }
165 | .text { *zoom: 1; color: #333333; margin: 0 auto; padding: 20px; }
166 | .text:before, .text:after { content: ""; display: table; }
167 | .text:after { clear: both; }
168 | .text aside { position: relative; z-index: 2; }
169 | .text p { margin: 5px 0; }
170 | .text > p:first-of-type, .text > * > p:first-of-type { margin-top: 15px; }
171 | .text em { font-style: italic; }
172 | .text sup { line-height: 100%; vertical-align: super; font-size: 70%; }
173 | .text pre code, .text .example { display: inline-block; padding: 15px; border: 1px solid #eeeeee; margin: 10px 0; background-color: #fefefe; }
174 | .text .metadata { border-left: 3px solid #dfdfdf; color: #666666; font-size: 0.9em; margin: 0 0 1.5em 0; padding: 5px 15px; background: #fcfcfc; }
175 | .text .deprecation { border-left: 3px solid #d42e15; color: #333333; font-size: 0.9em; margin: 0 0 1.5em 0; padding: 5px 15px; background: #fcfcfc; }
176 | .text .example { max-height: 400px; max-width: 390px; overflow: auto; }
177 | .text .example .monospace, .text .example pre, .text .example input { margin: 0; padding: 0; line-height: 20px; font-size: 12px; font-family: Menlo, monospace; }
178 | .text .example pre { clear: both; }
179 | .text .example .prompt, .text .example .command { float: left; }
180 | .text .example .prompt { width: 50px; color: #888888; }
181 | .text .example .command, .text .example input { font-weight: bold; }
182 | .text .example input { width: 325px; margin-left: -1px; outline: none; border: 0; color: #333333; float: left; }
183 | .text h1.command { padding-left: 20px; }
184 | .text h1.command span { display: inline-block; }
185 | .text h1.command span.name { margin-left: -20px; }
186 | .text h1.command span.arg { display: inline; }
187 | .text h1:first-child { margin-top: 0; }
188 |
189 | #comments { margin-top: 15px; }
190 |
191 | .wide-callout { background: #fffcc2; border-color: #f0ec8e; border-style: solid; border-width: 0 0 3px 0; display: block; padding: 0.5em 0; text-align: center; margin: 0; }
192 |
193 | .home-section { padding: 20px; border-bottom: 1px solid #eeeeee; }
194 | .home-section:last-of-type { border-bottom-width: 0; }
195 |
196 | .home-callout { max-width: 100%; }
197 | .home-callout .title { font-size: 20px; }
198 |
199 | @media only screen and (min-width: 992px) { .home-intro { display: block; width: 102.08333%; margin: 0 -1.04167%; *zoom: 1; }
200 | .home-intro:before, .home-intro:after { content: ""; display: table; }
201 | .home-intro:after { clear: both; }
202 | .home-intro > section { display: inline; float: left; width: 31.25%; margin: 0 1.04167%; } }
203 | #clients h2 { margin: 2em 0 1em 0; text-align: left; }
204 | #clients .icon { color: #333333; font-size: 20px; }
205 | #clients .icon-star { color: #ffe00d; }
206 | #clients table { table-layout: fixed; }
207 | #clients table td, #clients table th { padding: 0.8em; border-bottom: 1px solid #efefef; }
208 | #clients table td.description { color: #666666; font-size: 0.9em; }
209 | #clients table td.authors { text-align: right; }
210 | #clients col.homepage, #clients col.recommended, #clients col.active, #clients col.repository { width: 1em; }
211 | #clients col.authors { width: 120px; }
212 | #clients .twitter-avatar { width: 36px; height: 36px; }
213 | #clients .languages { list-style-type: none; margin: 1.5em 0 0 0; padding: 1.5em 0 0 0; border-top: 1px solid #efefef; }
214 | #clients .languages li { display: inline-block; width: 7em; margin: 0.3em; }
215 |
216 | article { overflow: hidden; *zoom: 1; }
217 | article p, article ul { margin: 7px 0; }
218 | article ul { list-style-type: disc; padding-left: 25px; }
219 | article ol { list-style-type: decimal; }
220 | article ol li { margin: 4px 0; }
221 | article ol ol { list-style-type: lower-alpha; }
222 | article h2, article h3 { font-weight: 500; }
223 | article h2 { margin-top: 22px; }
224 | article h3 { margin-top: 18px; }
225 | article aside > *:first-child { margin-top: 0; }
226 | article aside h2 { font-size: 16px; font-weight: 300; }
227 | article table { margin: 10px 0; }
228 | article table tr td { vertical-align: top; padding: 6px 10px; border-bottom: 1px solid #dddddd; }
229 | article table tr:last-child td { border-bottom-width: 0; }
230 | article table tr.current { background-color: #feffe8; }
231 | article table.versions td:first-child { font-size: 22px; line-height: 22px; }
232 |
233 | @media only screen and (min-width: 992px) { article { display: block; width: 102.08333%; margin: 0 -1.04167%; overflow: hidden; *zoom: 1; }
234 | .article-main { display: inline; float: left; width: 64.58333%; margin: 0 1.04167%; }
235 | .article-aside { display: inline; float: left; width: 31.25%; margin: 0 1.04167%; }
236 | aside { border-left: 1px solid #dfdfdf; margin: 0 0 15px 20px; padding: 0 0 0 20px; } }
237 | body.topics.whos-using-redis ul:first-of-type { text-align: center; list-style-type: none; margin: 15px; }
238 | body.topics.whos-using-redis ul:first-of-type li { margin: 10px 10px; padding: 0; display: inline-block; vertical-align: middle; }
239 | body.topics.whos-using-redis ul:first-of-type li img { vertical-align: middle; max-width: 200px; max-height: 76px; }
240 |
241 | #commands { text-align: center; }
242 | #commands ul { overflow: hidden; *zoom: 1; margin: 1em 0; text-align: left; padding: 0; }
243 | #commands li { -webkit-box-sizing: border-box; -moz-box-sizing: border-box; box-sizing: border-box; }
244 | #commands li a { display: block; padding: 1em; color: #333333; text-decoration: none; }
245 | #commands li a:hover, #commands li a:focus { color: #333333; background: #ecf7fa; }
246 | #commands li .command { white-space: nowrap; overflow: hidden; -ms-text-overflow: ellipsis; -o-text-overflow: ellipsis; text-overflow: ellipsis; font-family: "Monaco", "Menlo", monospace; display: block; margin: 0 0 2px 0; color: #0066aa; }
247 | #commands li .command .args { color: #555555; }
248 | #commands li .summary { display: block; font-size: 0.8em; line-height: 1.5em; }
249 | #commands nav { margin: 0 auto; background-color: #f0f0f0; width: 100%; font-size: 0.85em; display: block; padding: 10px 0; }
250 | #commands nav a { color: #555555; text-decoration: none; display: inline-block; padding: 0.3em 0.5em; }
251 | #commands nav a.current { background-color: #666666; color: white; }
252 | #commands nav label { margin: 0 10px; }
253 | #commands nav label span { display: none; }
254 | #commands nav select, #commands nav input { border: 1px solid #bbbbbb; font-size: 14px; margin: 0 5px; }
255 | #commands nav input { border-radius: 5px; padding: 5px 5px 6px 5px; width: 100px; }
256 | #commands nav input:focus { outline: none; }
257 | #commands nav select { height: 32px; }
258 |
259 | h1.command { margin-bottom: 0.5em; }
260 |
261 | .betacmd { background-color: #d42e15; border-radius: 0.5em; color: white; padding: 0.2em 0.3em; }
262 |
263 | @media only screen and (min-width: 992px) { #commands .command { font-size: 0.9em; }
264 | #commands .command .args { font-size: 0.8em; }
265 | #commands nav label { margin: 0 10px 0 0; }
266 | #commands nav label span { display: inline; }
267 | #commands nav input { width: 200px; }
268 | #commands ul li { float: left; margin: 1em 1.5%; overflow: hidden; width: 30%; }
269 | #commands ul li a { -webkit-border-radius: 5px; -moz-border-radius: 5px; -ms-border-radius: 5px; -o-border-radius: 5px; border-radius: 5px; background-color: #fafafa; height: 6em; } }
270 | #buzz { text-align: right; }
271 | #buzz h2 { margin-bottom: 20px; text-align: center; }
272 | #buzz ul { margin: 7px 0; text-align: left; }
273 | #buzz li { margin: 0 0 8px 0; padding-left: 34px; position: relative; min-height: 26px; word-wrap: break-word; }
274 | #buzz li a:first-child { display: block; height: 26px; left: 0; position: absolute; top: 3px; width: 26px; }
275 | #buzz li a:first-child img { height: 26px; width: 26px; }
276 |
277 | .download-versions { list-style-type: none; margin: 0; padding: 0; }
278 | .download-versions > li { display: grid; grid-template-rows: min-content 1fr min-content; margin-bottom: 20px; border-bottom: 1px solid #eeeeee; padding-bottom: 20px; }
279 |
280 | .download-version { color: #999999; }
281 |
282 | .download-links { margin: 15px 0 0 0; text-align: center; }
283 |
284 | .download-link { display: inline-block; color: #333333; text-align: center; width: 80px; border-radius: 5px; padding: 10px; margin: 10px; border: 1px solid #dddddd; line-height: 1.4; }
285 | .download-link .fa { font-size: 20px; margin-bottom: 8px; display: block; }
286 | .download-link:hover { text-decoration: none; background: #fafafa; }
287 |
288 | @media only screen and (min-width: 992px) { .download-versions { display: grid; grid-template-columns: repeat(3, 1fr); grid-gap: 1rem;} }
289 |
290 | body { width: 100%; height: 100%; }
291 |
292 | .slideout-menu { position: fixed; left: 0; top: 0; bottom: 0; right: 0; z-index: 0; width: 256px; overflow-y: auto; -webkit-overflow-scrolling: touch; display: none; }
293 |
294 | .slideout-panel { position: relative; z-index: 1; height: 100%; }
295 |
296 | .slideout-open, .slideout-open body, .slideout-open .slideout-panel { overflow: hidden; }
297 |
298 | .slideout-open .slideout-menu { display: block; }
299 |
300 | .site-header { overflow: hidden; *zoom: 1; background-color: #222222; padding: 0; width: 100%; }
301 |
302 | .desktop-header { padding: 0.7em 0; white-space: nowrap; overflow: hidden; }
303 | .desktop-header .home { margin-right: 0.8em; }
304 | .desktop-header a { line-height: 40px; display: inline-block; color: #f9f9f9; padding: 0 0.4em; text-decoration: none; }
305 | .desktop-header a:first-child { margin-left: 0; }
306 | .desktop-header a:hover { text-decoration: none; color: white; }
307 | .desktop-header .tryfree { border-radius: 6px!important; background-color: #a51f17; margin-left: 0.8em; }
308 |
309 | .site-header .desktop-header { display: none; }
310 |
311 | .site-header .mobile-header { display: grid; grid-template-columns: auto 1fr auto;}
312 | .site-header .mobile-header .btn-hamburger { border: 0; color: #eeeeee; padding: 0; background: transparent; outline: none; margin-left: 0.5em;}
313 | .site-header .mobile-header a.home { justify-self: center; align-self: center; }
314 | .site-header .mobile-header a.home img { width: 80px; height: 27px; padding: 0.2em; }
315 | .site-header .mobile-header a.tryfree { padding: 0 0.4em 0 0.4em; margin: 0.4em; border-radius: 6px!important; color: white; background-color: #a51f17; }
316 |
317 | @media only screen and (min-width: 992px) { .site-header .desktop-header { display: block; }
318 | .site-header .mobile-header { display: none; } }
319 | .site-header .home img { height: 40px; width: 120px; vertical-align: middle; }
320 |
321 |
322 | .site-wrapper { background: white; }
323 |
324 | .mobile-menu { background-color: #424242; }
325 | .mobile-menu a { color: white; }
326 | .mobile-menu a:hover { text-decoration: underline; }
327 | .mobile-menu .menu-header { border-bottom: 1px solid #58595a; padding: 15px; background: url("/images/redis-gray.png") no-repeat center center; background-size: 80px 71px; min-height: 71px; text-align: center; }
328 | .mobile-menu .menu-header-title { font-weight: 400; letter-spacing: 0.5px; margin: 0; }
329 | .mobile-menu .menu-section { margin: 25px 0; }
330 | .mobile-menu .menu-section-title { text-transform: uppercase; color: #85888d; font-weight: 200; font-size: 13px; letter-spacing: 1px; padding: 0 20px; margin: 0; }
331 | .mobile-menu .menu-section-list { padding: 0; margin: 10px 0; list-style: none; }
332 | .mobile-menu .menu-section-list a { display: block; padding: 10px 20px; }
333 | .mobile-menu .menu-section-list a:hover { background-color: rgba(255, 255, 255, 0.1); text-decoration: none; }
334 | .mobile-menu .logo { width: 120px; height: 106px; }
335 |
--------------------------------------------------------------------------------
/scripts/generate_interactive_commands.rb:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 |
3 | require "./app"
4 | require "erb"
5 |
6 | # Treat the subcommands of these commands as part of the command.
7 | SUBCOMMANDS = %w(
8 | client
9 | ).freeze
10 |
11 | # Explicitly allow certain groups (don't allow "server", "connection" and
12 | # "pubsub" commands by default).
13 | ALLOW_GROUPS = %w(
14 | generic
15 | hash
16 | list
17 | set
18 | sorted-set
19 | hyperloglog
20 | string
21 | scripting
22 | geo
23 | stream
24 | bitmap
25 | ).freeze
26 |
27 | # Override ALLOW_GROUPS for some commands.
28 | ALLOW_COMMANDS = %w(
29 | ping
30 | echo
31 | info
32 | lastsave
33 | time
34 | role
35 | command
36 | client\ info
37 | ).freeze
38 |
39 | # Explicitly deny some commands.
40 | DENY_COMMANDS = %w(
41 | blpop
42 | brpop
43 | brpoplpush
44 | bzpopmin
45 | bzpopmax
46 | copy
47 | select
48 | move
49 | randomkey
50 | script
51 | eval
52 | evalsha
53 | object
54 | migrate
55 | xread
56 | xclaim
57 | xreadgroup
58 | xack
59 | xgroup
60 | xinfo
61 | xpending
62 | restore
63 | ).freeze
64 |
65 | def allowed_commands
66 | @allowed_commands ||= commands.select do |cmd|
67 | name = cmd.name.split(/\s+/).first.downcase
68 | fullname = cmd.name.downcase
69 |
70 | !DENY_COMMANDS.include?(name) &&
71 | (ALLOW_COMMANDS.include?(name) ||
72 | (SUBCOMMANDS.include?(name) &&
73 | ALLOW_COMMANDS.include?(fullname)) ||
74 | ALLOW_GROUPS.include?(cmd.group))
75 | end
76 | end
77 |
78 | def patterns
79 | @patterns ||= Hash[*allowed_commands.map do |cmd|
80 | args = cmd.arguments
81 | if args.empty?
82 | pattern = []
83 | elsif args[0].type == ["key"] && !args[0].multiple? &&
84 | args[1..-1].none? { |arg| arg.type.include?("key") }
85 | # Only namespace the first argument
86 | pattern = [:first]
87 | elsif args.all? { |arg| arg.type == ["key"] }
88 | # Namespace all arguments
89 | pattern = [:all]
90 | elsif args.all? { |arg| !arg.multiple? && !arg.optional? && arg.type.size == 1 }
91 | # Constant number of arguments can be zipped
92 | pattern = [:zip, args.map { |arg| :key if arg.type == ["key"] } ]
93 | elsif args.size == 1 && args[0].multiple? && args[0].type.include?("key")
94 | # Single variadic argument with key can be zipped
95 | pattern = [:zip, args[0].type.map { |type| :key if type == "key" } ]
96 | else
97 | # Non-standard
98 | pattern = [:custom]
99 | end
100 |
101 | [cmd.name.downcase, pattern]
102 | end.compact.flatten(1)]
103 | end
104 |
105 | by_group = allowed_commands.group_by(&:group).sort.map do |group,commands|
106 | lines = commands.sort_by(&:name).map do |cmd|
107 | %{"%s" => %s} % [cmd.name.downcase, patterns[cmd.name.downcase].inspect]
108 | end
109 |
110 | [group, lines]
111 | end
112 |
113 | def file
114 | __FILE__
115 | end
116 |
117 | template = <<-TPL
118 | # This file is automatically generated by:
119 | # <%= file %>.
120 | #
121 | # Do not edit.
122 | #
123 |
124 | module Interactive
125 | COMMANDS = {
126 | <%- by_group.each do |group,commands| -%>
127 | # <%= group %>
128 | <%- commands.each do |command| -%>
129 | <%= command %>,
130 | <%- end %>
131 | <%- end -%>
132 | }.freeze
133 |
134 | SUBCOMMANDS = {
135 | <%- SUBCOMMANDS.each do |command| -%>
136 | "<%= command %>" => 1,
137 | <%- end %>
138 | }.freeze
139 | end
140 |
141 | TPL
142 |
143 | STDOUT.puts ERB.new(template, 0, "-").result(binding)
144 |
145 |
--------------------------------------------------------------------------------
/test/clients.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | scope do
4 | test "Clients page" do
5 | visit "/clients"
6 |
7 | assert has_css?("h2", text: "Ruby")
8 | assert has_content?("redis-rb")
9 | end
10 | end
11 |
--------------------------------------------------------------------------------
/test/command_reference.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | scope do
4 | test "Command reference" do
5 | visit "/commands"
6 |
7 | assert has_content?("ECHO")
8 | assert has_content?("Echo the given string")
9 | end
10 |
11 | test "Command page" do
12 | visit "/commands"
13 |
14 | click_link_or_button "ECHO"
15 |
16 | assert has_title?("ECHO")
17 |
18 | within "h1" do
19 | assert has_content?("ECHO")
20 | assert has_content?("message")
21 | end
22 |
23 | within "article" do
24 | assert has_css?("p", text: "Returns message.")
25 | assert has_content?("Available since 1.0.0")
26 | end
27 | end
28 |
29 | test "Command page with complex arguments" do
30 | visit "/commands"
31 |
32 | click_link(href: /\/sort$/)
33 |
34 | within "h1" do
35 | assert has_content?("[BY pattern]")
36 | assert has_content?("[LIMIT offset count]")
37 | assert has_content?("[ASC|DESC]")
38 | assert has_content?("[ALPHA]")
39 | assert has_content?("[STORE destination]")
40 | end
41 | end
42 |
43 | test "Commands with spaces" do
44 | visit "/commands"
45 |
46 | click_link_or_button "OBJECT ENCODING"
47 |
48 | assert has_title?("OBJECT ENCODING")
49 | assert has_css?("h1", text: "OBJECT ENCODING")
50 | end
51 |
52 | test "Missing command" do
53 | visit "/commands/foobar"
54 |
55 | assert_equal page.current_url, "https://www.google.com/search?q=foobar+site%3Aredis.io"
56 | end
57 | end
58 |
--------------------------------------------------------------------------------
/test/formatting.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | require File.expand_path("../lib/reference", File.dirname(__FILE__))
4 |
5 | reference = Reference.new(JSON.parse(File.read("#{documentation_path}/commands.json")))
6 |
7 | setup do
8 | reference
9 | end
10 |
11 | test "OBJECT ENCODING" do |reference|
12 | res = "OBJECT ENCODING key"
13 | assert_equal reference["OBJECT ENCODING"].to_s, res
14 | end
15 |
16 | test "DEL" do |reference|
17 | res = "DEL key [key ...]"
18 | assert_equal reference["DEL"].to_s, res
19 | end
20 |
21 | test "DISCARD" do |reference|
22 | res = "DISCARD"
23 | assert_equal reference["DISCARD"].to_s, res
24 | end
25 |
26 | test "SORT" do |reference|
27 | res = "SORT key [BY pattern] [LIMIT offset count] [GET pattern [GET pattern ...]]" +
28 | " [ASC|DESC] [ALPHA] [STORE destination]"
29 | assert_equal reference["SORT"].to_s, res
30 | end
31 |
32 | test "UNSUBSCRIBE" do |reference|
33 | res = "UNSUBSCRIBE [channel [channel ...]]"
34 | assert_equal reference["UNSUBSCRIBE"].to_s, res
35 | end
36 |
37 | test "ZINTERSTORE" do |reference|
38 | res = "ZINTERSTORE destination numkeys key [key ...] [WEIGHTS weight [weight ...]]" +
39 | " [AGGREGATE SUM|MIN|MAX]"
40 | assert_equal reference["ZINTERSTORE"].to_s, res
41 | end
42 |
--------------------------------------------------------------------------------
/test/helper.rb:
--------------------------------------------------------------------------------
1 | require "cuba/capybara"
2 | require File.expand_path("../app", File.dirname(__FILE__))
3 |
4 | begin
5 | require "ruby-debug"
6 | rescue LoadError
7 | end
8 |
9 | Capybara.default_selector = :css
10 |
--------------------------------------------------------------------------------
/test/interactive/namespace.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | require File.expand_path("../../lib/interactive/namespace", File.dirname(__FILE__))
4 |
5 | scope do
6 | test "Namespace for static arguments" do
7 | assert ["set", "ns:key", "foo"] == \
8 | Interactive.namespace("ns", ["set", "key", "foo"])
9 |
10 | assert ["ping"] == \
11 | Interactive.namespace("ns", ["ping"])
12 |
13 | assert ["echo", "foo"] == \
14 | Interactive.namespace("ns", ["echo", "foo"])
15 | end
16 |
17 | test "Namespace with optional arguments" do
18 | assert ["zrange", "ns:key", "0", "-1", "withscores"] == \
19 | Interactive.namespace("ns", ["zrange", "key", "0", "-1", "withscores"])
20 | end
21 |
22 | test "Namespace with variable number of arguments" do
23 | assert ["mset", "ns:key1", "foo", "ns:key2", "foo"] == \
24 | Interactive.namespace("ns", ["mset", "key1", "foo", "key2", "foo"])
25 |
26 | assert ["hmset", "ns:key1", "field1", "foo", "field2", "bar"] == \
27 | Interactive.namespace("ns", ["hmset", "key1", "field1", "foo", "field2", "bar"])
28 | end
29 |
30 | test "Namespace ZUNIONSTORE/ZINTERSTORE" do
31 | for cmd in %w(zunionstore zinterstore)
32 | assert [cmd, "ns:dst", "2", "ns:key1", "ns:key2", "weights", "1.0", "2.0"] == \
33 | Interactive.namespace("ns", [cmd, "dst", "2", "key1", "key2", "weights", "1.0", "2.0"])
34 | end
35 | end
36 |
37 | test "Namespace SORT" do
38 | command = %w(sort key by foo by bar get foo get bar store dst limit 0 10 asc)
39 | expected = %w(sort ns:key by ns:foo by ns:bar get ns:foo get ns:bar store ns:dst limit 0 10 asc)
40 | assert expected == Interactive.namespace("ns", command)
41 | end
42 |
43 | test "Namespace GEORADIUS" do
44 | command = %w(georadius key 10 20 40 km withcoord store foo storedist bar)
45 | expected = %w(georadius ns:key 10 20 40 km withcoord store ns:foo storedist ns:bar)
46 | assert expected == Interactive.namespace("ns", command)
47 | end
48 | end
49 |
50 |
--------------------------------------------------------------------------------
/test/json.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | scope do
4 | test do
5 | visit "/commands.json"
6 |
7 | assert has_content?("\"APPEND\": {")
8 |
9 | visit "/clients.json"
10 |
11 | assert has_content?("\"name\": \"redis-rb\"")
12 | end
13 | end
14 |
--------------------------------------------------------------------------------
/test/sitemap.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | prepare do
4 | redis.flushdb
5 | redis.hmset("versions", "stable", "2.0.4", "development", "2.1.9")
6 | end
7 |
8 | scope do
9 | test "Sitemap" do
10 | visit "/"
11 |
12 | find(".desktop-header").click_link_or_button "Commands"
13 |
14 | assert has_content?("PING")
15 |
16 | find(".desktop-header").click_link_or_button "Clients"
17 |
18 | assert has_content?("redis-rb")
19 |
20 | find(".desktop-header").click_link_or_button "Download"
21 |
22 | assert has_content?("Run Redis with:")
23 |
24 | find(".desktop-header").click_link_or_button "Community"
25 |
26 | assert has_css?("a[href='http://groups.google.com/group/redis-db']", text: "mailing list")
27 | assert has_css?("a[href='http://twitter.com/redisfeed']", text: "Redis news feed")
28 |
29 | find(".desktop-header").click_link_or_button "Documentation"
30 |
31 | click_link_or_button "full list of commands"
32 |
33 | assert has_content?("PING")
34 | end
35 | end
36 |
--------------------------------------------------------------------------------
/test/topics.rb:
--------------------------------------------------------------------------------
1 | require "./test/helper"
2 |
3 | scope do
4 | test "Topic" do
5 | visit "/topics/replication"
6 |
7 | assert has_title?("Replication")
8 | assert has_xpath?("//h1", text: "Replication")
9 | assert has_xpath?("//h2", text: "Configuration")
10 | end
11 |
12 | test "Missing topic" do
13 | visit "/topics/foobar"
14 |
15 | assert has_content?("Sorry")
16 | assert has_content?("topics/foobar.md")
17 | assert page.driver.response.status == 404
18 | end
19 | end
20 |
--------------------------------------------------------------------------------
/views/404.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 | %h1 Sorry, I can't find that page :-/
4 |
5 | - if path
6 | %p
7 | If you think this page should exist, add a
8 |
9 | %code
10 | != path
11 |
12 | file to the repository and send a pull request.
13 |
--------------------------------------------------------------------------------
/views/avatar.haml:
--------------------------------------------------------------------------------
1 | - size ||= 32
2 | - alt ||= ""
3 |
4 | %img(src="http://www.gravatar.com/avatar/#{gravatar_hash email}.jpg?s=#{size}"
5 | width="#{size}"
6 | height="#{size}"
7 | alt="#{alt}"
8 | title="#{alt}"
9 | )
10 |
--------------------------------------------------------------------------------
/views/buzz.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 | %h1 Buzz
4 |
5 | %section#buzz(data-limit="30")
6 | %ul
7 |
--------------------------------------------------------------------------------
/views/clients.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 | %section#clients
4 |
5 | %h1 Clients
6 |
7 | %p
8 | The recommended client(s) for a language are marked with a .
9 |
10 | %p
11 | Clients with some activity in the official repository within the latest six months are marked with a .
12 |
13 | %p
14 | Want
15 | %strong your client listed here?
16 | Please fork the
17 | %a(href="https://github.com/redis/redis-doc") redis-doc repository
18 | and edit the clients.json file.
19 | %strong Submit a pull request
20 | and you are done.
21 |
22 | .languages
23 | Browse by language:
24 |
25 | %ul
26 | - @clients_by_language.each do |language, _|
27 | %li
28 | %a(href="##{anchorize_language(language)}")= language
29 |
30 | %table
31 | %col(class="name")
32 | %col(class="active")
33 | %col(class="recommended")
34 | %col(class="homepage")
35 | %col(class="repository")
36 | %col(class="description")
37 | %col(class="authors")
38 |
39 | - @clients_by_language.each do |language, clients|
40 | %tr
41 | %th(colspan="7")
42 | %h2(id="#{anchorize_language(language)}")= language
43 |
44 | - clients.sort_by { |c| [c["active"] ? 0 : 1, c["recommended"] ? 0 : 1, c["name"].downcase] }.each do |client|
45 | %tr
46 | %td
47 | = client["name"]
48 | %td
49 | - if client["active"]
50 | %i(class="fa fa-smile-o icon")
51 |
52 | %td
53 | - if client["recommended"]
54 | %i(class="fa fa-star icon icon-star")
55 |
56 | %td
57 | - if client["url"]
58 | %a(href="#{client["url"]}" class="icon")
59 | %i(class="fa fa-home")
60 |
61 | %td
62 | - if client["repository"]
63 | %a(href="#{client["repository"]}" class="icon")
64 | %i(class="fa fa-code-fork")
65 |
66 | %td.description
67 | = client["description"]
68 |
69 | %td.authors
70 | - if client["authors"]
71 | - client["authors"].each do |author|
72 | %a(href="https://twitter.com/#{author}")
73 | = "@#{author} "
74 |
75 | %h1(style="margin-top:80px") Higher level libraries and tools
76 |
77 | %p
78 | This is an additional list of libraries that are not direct layers on top of the Redis API, but higher level libraries such as ORMs, messaging libraries, and other misc tools that are designed for Redis.
79 |
80 | %table
81 | - @redis_tools.each do |tool|
82 | - tool["description"] += " (#{tool['language']})"
83 | %tr
84 | %td
85 | = tool["name"]
86 |
87 | %td
88 | - if tool["repository"]
89 | %a(href="#{tool["repository"]}") Repository
90 |
91 | - if tool["url"]
92 | %a(href="#{tool["url"]}") Homepage
93 |
94 | %td
95 | - tool["authors"].each do |author|
96 | %a(href="http://twitter.com/#{author}")= author
97 |
98 | %td
99 | = tool["description"]
100 |
101 |
--------------------------------------------------------------------------------
/views/commands.haml:
--------------------------------------------------------------------------------
1 | %section#commands
2 | %nav
3 | .container
4 | %label
5 | %span Filter by group:
6 |
7 | %select.command-reference-filter
8 | %option(value="") All
9 |
10 | - Reference::GROUPS.sort_by(&:last).each do |name, description|
11 | %option(value="#{name}")= description
12 |
13 | or
14 |
15 | %label
16 | %span search for:
17 | %input.js-command-reference-search(placeholder="e.g. #{@commands.sample.name}" autofocus autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false")
18 |
19 | .container
20 | %ul
21 | - @commands.each do |command|
22 | - if command.is_listed?
23 | %li(data-group="#{command.command["group"]}" data-name="#{command.name.downcase}")
24 | %a(href="/commands/#{command.to_param}")
25 | %span.command
26 | = command.name
27 |
28 | %span.args
29 | - command.arguments.each do |argument|
30 | = argument
31 |
32 | %span.summary= command.command["summary"]
33 |
--------------------------------------------------------------------------------
/views/commands/name.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 |
4 | %h1.command
5 | %span.name= @command.name
6 | - @command.arguments.each do |arg|
7 | %span.arg= arg
8 |
9 | %article
10 | .article-main
11 |
12 | %div.metadata
13 | - if @command.since
14 | %p Available since #{@command.since}.
15 | - else
16 | %p
17 | %span.betacmd Beta
18 | Not yet available in a stable version of Redis.
19 | Download unstable if you want to test this command.
20 |
21 | - if @command.complexity
22 | %p Time complexity: #{@command.complexity}
23 |
24 | - if @command.deprecated_since
25 | %div.deprecation
26 | %p
27 | Deprecation notice:
28 | as of Redis version #{@command.deprecated_since} this command is
29 | considered as deprecated. While it is unlikely that it will be
30 | completely removed, prefer using #{@command.replaced_by} in its
31 | stead.
32 |
33 | ~ custom_view("#{documentation_path}/commands/#{@name.downcase}.md", {}, layout: false)
34 |
35 | - if @command.history
36 | %h2 History
37 | %ul
38 | - @command.history.each do |entry|
39 | %li
40 | Redis version >= #{entry[0]}: #{entry[1]}
41 |
42 | .article-aside
43 | %aside
44 |
45 | - unless @related_topics.empty?
46 | %h2 Related topics
47 |
48 | %ul
49 | - @related_topics.each do |name, url|
50 | %li
51 | %a(href="#{url}")= name
52 |
53 | %h2
54 | Related commands
55 |
56 | %ul
57 | - @related_commands.each do |command|
58 | %li
59 | %a(href="/commands/#{command.to_param}")
60 | - if command == @command
61 | %strong= command.name
62 | - else
63 | = command.name
64 |
--------------------------------------------------------------------------------
/views/comments.haml:
--------------------------------------------------------------------------------
1 | %aside#comments
2 | %ul
3 | - Comment.find(url: req.path).sort(order: "desc").each do |comment|
4 | %li
5 | = partial "avatar", email: comment.user.email, size: 32
6 |
7 | %article
8 | - time = Time.at(comment.last_modified.to_i).utc
9 |
10 | %time(datetime="#{time.iso8601}" pubdate="pubdate")
11 | = time.strftime("%b %e, %Y %I.%M%P")
12 |
13 | ~ RDiscount.new(comment.body, :smart, :filter_html).to_html
14 |
15 | %form.user.logged-in(method="post" action="/comments" style="display: none")
16 | %input(type="hidden" name="url" value="#{req.path}")
17 |
18 | %img.user.gravatar(src="" width="32" height="32")
19 |
20 | %textarea(name="body")
21 |
22 | %input(type="submit" value="Post")
23 |
24 | %p.user.anonymous
25 | %a(href="/login") Log in to comment
26 |
--------------------------------------------------------------------------------
/views/community.md:
--------------------------------------------------------------------------------
1 | Community
2 | ===
3 |
4 | To get help or provide any feedback, the main channel is the Redis mailing list:
5 |
6 | * Join the [mailing list](http://groups.google.com/group/redis-db) (Subscribe via [email](mailto:redis-db+subscribe@googlegroups.com)).
7 |
8 | For bug reports please just use [Github](https://github.com/redis/redis).
9 |
10 | Other places where you can find people interested in Redis:
11 |
12 | * Meet people interested in Redis at the [Redis Discord server](https://discord.gg/redis).
13 | * The [Redis tag on Stack Overflow](http://stackoverflow.com/questions/tagged/redis?sort=newest&pageSize=30).
14 | * Follow [Redis news feed](http://twitter.com/redisfeed) on Twitter.
15 | * The Redis community uses a Reddit sub for news and certain announcements (that also always go to the ML): [/r/redis sub on Reddit](https://www.reddit.com/r/redis/).
16 |
17 | Project governance
18 | ---
19 |
20 | Redis has adopted a light governance model that is intended to be a meritocracy, aiming to empower individuals who demonstrate a long-term commitment and make significant contributions.
21 |
22 | For more information refer to the [Governance page](/topics/governance).
23 |
24 | Conferences and meetups
25 | ---
26 |
27 | * [Redis Live](https://meetups.redis.com/redis-live/)
28 | * [RedisConf Annual Conference](https://redis.com/redisconf)
29 | * [Redis Day Seattle](https://connect.redis.com/redisdayseattle/mktg)
30 | * [Redis Day Bangalore](https://connect.redis.com/redisdaybangalore)
31 | * [London Redis Meetup Group](https://www.meetup.com/Redis-London)
32 | * [San Francisco Meetup Group](http://sfmeetup.redis.io)
33 | * [New York Meetup Group](https://www.meetup.com/New-York-REDIS-Meetup)
34 | * [#RedisTLV (Tel Aviv Redis) Meetup Group](https://www.meetup.com/Tel-Aviv-Redis-Meetup)
35 | * [Paris Redis Meetup](https://www.meetup.com/Paris-Redis-Meetup/)
36 |
37 | Contributing to Redis
38 | ---
39 |
40 | Would you like to contribute a feature to Redis?
41 |
42 | 1. Drop a message to the [mailing list](http://groups.google.com/group/redis-db) with your proposal. Make sure you explain what the use case is and how the API would look like.
43 |
44 | 2. If you get good feedbacks, do the following to submit a patch:
45 |
46 | 1. Fork [the official repository](http://github.com/redis/redis).
47 | 2. Clone your fork: `git clone git@github.com:/redis.git`
48 | 3. Make sure tests are passing for you: `make && make test`
49 | 4. Create a topic branch: `git checkout -b new-feature`
50 | 5. Add tests and code for your changes.
51 | 6. Once you're done, make sure all tests still pass: `make && make test`
52 | 7. Commit and push to your fork.
53 | 8. [Create an issue](https://github.com/redis/redis/issues) with a link to your patch.
54 | 9. Sit back and enjoy.
55 |
56 | There are other ways to help:
57 |
58 | * [Fix a bug or share your experience on issues](https://github.com/redis/redis/issues)
59 |
60 | * Improve the [documentation](http://github.com/redis/redis-doc)
61 |
62 | * Help maintain or create new [client libraries](/clients)
63 |
64 | * Improve [this very website](http://github.com/redis/redis-io)
65 |
--------------------------------------------------------------------------------
/views/documentation.md:
--------------------------------------------------------------------------------
1 | Documentation
2 | ===
3 |
4 | Note: The Redis Documentation is also available in raw (computer friendly) format in the [redis-doc github repository](http://github.com/redis/redis-doc). The Redis Documentation is released under the [Creative Commons Attribution-ShareAlike 4.0 International license](https://creativecommons.org/licenses/by-sa/4.0/).
5 |
6 | Programming with Redis
7 | ---
8 |
9 | * [The full list of commands](/commands) implemented by Redis, along with thorough documentation for each of them.
10 | * [Pipelining](/topics/pipelining): Learn how to send multiple commands
11 | at once, saving on round trip time.
12 | * [Redis Pub/Sub](topics/pubsub): Redis is a fast and stable Publish/Subscribe messaging system! Check it out.
13 | * [Memory optimization](/topics/memory-optimization): Understand how
14 | Redis uses RAM and learn some tricks to use less of it.
15 | * [Expires](/commands/expire): Redis allows to set a time to live different for every key so that the key will be automatically removed from the server when it expires.
16 | * [Redis as an LRU cache](/topics/lru-cache): How to configure and use Redis as a cache with a fixed amount of memory and auto eviction of keys.
17 | * [Redis transactions](/topics/transactions): It is possible to group commands together so that they are executed as a single transaction.
18 | * [Client side caching](/topics/client-side-caching): Starting with version 6 Redis supports server assisted client side caching. This document describes how to use it.
19 | * [Mass insertion of data](/topics/mass-insert): How to add a big amount of pre existing or generated data to a Redis instance in a short time.
20 | * [Partitioning](/topics/partitioning): How to distribute your data among multiple Redis instances.
21 | * [Distributed locks](/topics/distlock): Implementing a distributed lock manager with Redis.
22 | * [Redis keyspace notifications](/topics/notifications): Get notifications of keyspace events via Pub/Sub (Redis 2.8 or greater).
23 | * [Creating secondary indexes with Redis](/topics/indexes): Use Redis data structures to create secondary indexes, composed indexes and traverse graphs.
24 |
25 | Redis programmability
26 | ---
27 |
28 | * [Redis Programability](/topics/programmability): An overview of programmability in Redis.
29 | * [Redis Lua API](/topics/lua-api): Reference about the embedded [Lua 5.1](https://lua.org) interepreter runtime environment and APIs.
30 | * [Introduction to Eval Scripts](/topics/eval-intro): An introduction about using cached scripts.
31 | * [Introduction to Redis Functions](/topics/functions-intro): An introduction about using functions.
32 | * [Debugging Lua scripts](/topics/ldb): An overveiw of the native Redis Lua debugger for cached scripts.
33 |
34 | Redis modules API
35 | ---
36 |
37 | * [Introduction to Redis modules](/topics/modules-intro). A good place to start learing about Redis 4.0 modules programming.
38 | * [Implementing native data types](/topics/modules-native-types). Modules scan implement new data types (data structures and more) that look like built-in data types. This documentation covers the API to do so.
39 | * [Blocking operations](topics/modules-blocking-ops) with modules. This is still an experimental API, but a very powerful one to write commands that can block the client (without blocking Redis) and can execute tasks in other threads.
40 | * [Redis modules API reference](topics/modules-api-ref). Directly generated from the top comments in the source code inside `src/module.c`. Contains many low level details about API usage.
41 |
42 | Tutorials & FAQ
43 | ---
44 |
45 | * [Introduction to Redis data types](/topics/data-types-intro): This is a good starting point to learn the Redis API and data model.
46 | * [Introduction to Redis streams](/topics/streams-intro): A detailed description of the Redis 5 new data type, the Stream.
47 | * [Writing a simple Twitter clone with PHP and Redis](/topics/twitter-clone)
48 | * [Auto complete with Redis](http://autocomplete.redis.io)
49 | * [Data types short summary](/topics/data-types): A short summary of the different types of values that Redis supports, not as updated and info rich as the first tutorial listed in this section.
50 | * [FAQ](/topics/faq): Some common questions about Redis.
51 |
52 | Administration
53 | ---
54 | * [Quick Start](/topics/quickstart): How to quickly install and configure Redis. This targets people without prior experience with Redis.
55 | * [Redis-cli](/topics/rediscli): Learn how to master the Redis command line interface, something you'll be using a lot in order to administer, troubleshoot and experiment with Redis.
56 | * [Configuration](/topics/config): How to configure redis.
57 | * [Replication](/topics/replication): What you need to know in order to
58 | set up master-replicas replication.
59 | * [Persistence](/topics/persistence): Know your options when configuring
60 | Redis' durability.
61 | * [Redis Administration](/topics/admin): Selected administration topics.
62 | * [Security](/topics/security): An overview of Redis security.
63 | * [Redis Access Control Lists](/topics/acl): Starting with version 6 Redis supports ACLs. It is possible to configure users able to run only selected commands and able to access only specific key patterns.
64 | * [Encryption](/topics/encryption): How to encrypt Redis client-server communication.
65 | * [Signals Handling](/topics/signals): How Redis handles signals.
66 | * [Connections Handling](/topics/clients): How Redis handles clients connections.
67 | * [High Availability](/topics/sentinel): Redis Sentinel is the official high availability solution for Redis.
68 | * [Redis Releases](/topics/releases): Redis development cycle and version numbering.
69 |
70 | Performance
71 | ---
72 | * [Latency monitoring](/topics/latency-monitor): Redis integrated latency monitoring and reporting capabilities are helpful to tune Redis instances for low latency workloads.
73 | * [Benchmarks](/topics/benchmarks): See how fast Redis is in different platforms.
74 | * [Redis on-CPU profiling and tracing](/topics/performance-on-cpu): See how to perform on-CPU resource bottlenecks analysis in Redis.
75 |
76 | Embedded and IoT
77 | ---
78 |
79 | * [Redis on ARM and Raspberry Pi](/topics/ARM): Starting with Redis 4.0 ARM and the Raspberry Pi are officially supported platforms. This page contains general information and benchmarks.
80 | * [A reference implementation of Redis for IoT and Edge Computing can be found here](https://redis.com/redis-enterprise/redis-edge/).
81 |
82 | Troubleshooting
83 | ---
84 |
85 | * [Redis problems?](/topics/problems): Bugs? High latency? Other issues? Use [our problems troubleshooting page](/topics/problems) as a starting point to find more information.
86 |
87 | Redis Cluster
88 | ---
89 |
90 | * [Redis Cluster tutorial](/topics/cluster-tutorial): a gentle introduction and setup guide to Redis Cluster.
91 | * [Redis Cluster specification](/topics/cluster-spec): the more formal description of the behavior and algorithms used in Redis Cluster.
92 |
93 | Command runtime introspection
94 | ---
95 |
96 | * [Command key specifications](/topics/key-specs): as of Redis 7.0, the server reports how to extract the names of keys accessed by every command.
97 | * [Command tips](/topics/command-tips): tips communicate non-trivial execution modes and post-processing information about commands.
98 | * [Command arguments](/topics/command-arguments): an overview of command arguments as returned by the `COMMAND DOCS` command.
99 |
100 | Other distributed systems based on Redis
101 | ---
102 |
103 | * [Redis CRDTs](https://redis.com/redis-enterprise/technology/active-active-geo-distribution/) an active-active geo-distribution solutions for Redis.
104 | * [Roshi](https://github.com/soundcloud/roshi) is a large-scale CRDT set implementation for timestamped events based on Redis and implemented in Go. It was initially developed for [the SoundCloud stream](http://developers.soundcloud.com/blog/roshi-a-crdt-system-for-timestamped-events).
105 |
106 | Redis on SSD and persistent memory
107 | ---
108 |
109 | * [Redis on Flash](https://redis.com/redis-enterprise/technology/redis-on-flash/) by Redis Ltd. extends DRAM capacity with SSD and persistent memory.
110 |
111 | Specifications
112 | ---
113 |
114 | * [Redis Design Drafts](/topics/rdd): Design drafts of new proposals.
115 | * [Redis Protocol specification](/topics/protocol): if you're implementing a
116 | client, or out of curiosity, learn how to communicate with Redis at a
117 | low level.
118 | * [Redis RDB format](https://github.com/sripathikrishnan/redis-rdb-tools/wiki/Redis-RDB-Dump-File-Format) specification, and [RDB version history](https://github.com/sripathikrishnan/redis-rdb-tools/blob/master/docs/RDB_Version_History.textile).
119 | * [Internals](/topics/internals): Learn details about how Redis is implemented under the hood.
120 |
121 | Resources
122 | ---
123 |
124 | * [Redis Cheat Sheet](http://www.cheatography.com/tasjaevan/cheat-sheets/redis/): Online or printable function reference for Redis.
125 |
126 | Use cases
127 | ---
128 | * [Who is using Redis](/topics/whos-using-redis)
129 |
130 | Books
131 | ---
132 |
133 | The following is a list of books covering Redis that are already published. Books are ordered by release date (newer books first).
134 |
135 | * [Mastering Redis (Packt, 2016)](https://www.packtpub.com/big-data-and-business-intelligence/mastering-redis) by [Jeremy Nelson](https://www.packtpub.com/books/info/authors/jeremy-nelson).
136 | * [Redis Essentials (Packt, 2015)](http://www.amazon.com/Redis-Essentials-Maxwell-Dayvson-Silva-ebook/dp/B00ZXFCFLO) by [Maxwell Da Silva](http://twitter.com/dayvson) and [Hugo Tavares](https://twitter.com/hltbra)
137 | * [Redis in Action (Manning, 2013)](http://www.manning.com/carlson/) by [Josiah L. Carlson](http://twitter.com/dr_josiah) (early access edition).
138 | * [Instant Redis Optimization How-to (Packt, 2013)](http://www.packtpub.com/redis-optimization-how-to/book) by [Arun Chinnachamy](http://twitter.com/ArunChinnachamy).
139 | * [Instant Redis Persistence (Packt, 2013)](http://www.packtpub.com/redis-persistence/book) by Matt Palmer.
140 | * [The Little Redis Book (Free Book, 2012)](http://openmymind.net/2012/1/23/The-Little-Redis-Book/) by [Karl Seguin](http://twitter.com/karlseguin) is a great *free* and concise book that will get you started with Redis.
141 | * [Redis Cookbook (O'Reilly Media, 2011)](http://shop.oreilly.com/product/0636920020127.do) by [Tiago Macedo](http://twitter.com/tmacedo) and [Fred Oliveira](http://twitter.com/f).
142 |
143 | The following books have Redis related content but are not specifically about Redis:
144 |
145 | * [Seven databases in seven weeks (The Pragmatic Bookshelf, 2012)](http://pragprog.com/book/rwdata/seven-databases-in-seven-weeks).
146 | * [Mining the Social Web (O'Reilly Media, 2011)](http://shop.oreilly.com/product/0636920010203.do)
147 | * [Professional NoSQL (Wrox, 2011)](http://www.wrox.com/WileyCDA/WroxTitle/Professional-NoSQL.productCd-047094224X.html)
148 |
149 | Credits
150 | ---
151 |
152 | Redis is developed and maintained by the [Redis community](/community).
153 |
154 | The project was created, developed and maintained by [Salvatore Sanfilippo](http://twitter.com/antirez) until [June 30th, 2020](http://antirez.com/news/133). In the past [Pieter Noordhuis](http://twitter.com/pnoordhuis) and [Matt Stancliff](https://matt.sh) provided a very significant amount of code and ideas to both the Redis core and client libraries.
155 |
156 | The full list of Redis contributors can be found in the [Redis contributors page at Github](https://github.com/redis/redis/graphs/contributors). However there are other forms of contributions such as ideas, testing, and bug reporting. When it is possible, contributions are acknowledged in commit messages. The [mailing list archives](http://groups.google.com/group/redis-db) and the [Github issues page](https://github.com/redis/redis/issues) are good sources to find people active in the Redis community providing ideas and helping other users.
157 |
158 | Sponsors
159 | ---
160 |
161 | The work [Salvatore Sanfilippo](http://antirez.com) does in order to develop Redis is sponsored by [Redis Ltd.](http://redis.com) Other sponsors and past sponsors of the Redis project are listed in the [Sponsors page](/topics/sponsors).
162 |
163 | License, Trademark and Logo
164 | ---
165 |
166 | * Redis is released under the three clause BSD license. You can find [additional information in our license page](/topics/license).
167 | * The Redis trademark and logos are owned by Redis Ltd., please read the [Redis trademark guidelines](/topics/trademark) for our policy about the use of the Redis trademarks and logo.
168 |
--------------------------------------------------------------------------------
/views/download.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 | %h1 Download
4 |
5 | %p
6 | Stable releases liberally follow the usual
7 | %strong major.minor.patch
8 | semantic versioning schema.
9 |
10 | %ul.download-versions
11 | - DOWNLOADS[:channels].each do |channel, download|
12 | %li
13 | %h2
14 | = download[:name] || channel.capitalize
15 |
16 | - if download[:branch] =~ /^\d/
17 | %span.download-version= "(#{download[:branch]})"
18 |
19 | = download[:notes]
20 |
21 | .download-links
22 |
23 | - if channel != :unstable && channel != :docker && channel != :cloud
24 | %a.download-link(href="https://raw.githubusercontent.com/redis/redis/#{download[:branch]}/00-RELEASENOTES")
25 | %i.fa.fa-file-text-o
26 | Release notes
27 |
28 | - if channel == :stable
29 | - download_link = "https://download.redis.io/releases/redis-#{download[:version]}.tar.gz"
30 | - elsif channel == :docker || channel == :cloud
31 | - download_link = download[:url]
32 | - else
33 | - download_link = "https://github.com/redis/redis/archive/#{download[:version] || download[:branch]}.tar.gz"
34 |
35 | - if channel == :docker
36 | %a.download-link(href="#{download_link}")
37 | %i.fa.fa-external-link
38 | Docker Hub
39 | - elsif channel == :cloud
40 | %a.download-link(href="#{download_link}")
41 | %i.fa.fa-external-link
42 | Redis Cloud
43 | - else
44 | %a.download-link(href="#{download_link}")
45 | %i.fa.fa-arrow-circle-o-down
46 | Download
47 | = download[:version] || download[:branch]
48 |
49 | %h2 Other versions
50 |
51 | - DOWNLOADS[:other].each do |name, download|
52 | %h3
53 | = name.capitalize
54 |
55 | - if download[:branch] =~ /^\d/
56 | %span.download-version= "(#{download[:branch]})"
57 |
58 | = download[:notes]
59 |
60 | - if download[:version]
61 | %br
62 | See the
63 |
64 | %a(href="https://raw.githubusercontent.com/redis/redis/#{download[:branch]}/00-RELEASENOTES")
65 | release notes
66 |
67 | or
68 |
69 | - download_link = "https://download.redis.io/releases/redis-#{download[:version]}.tar.gz"
70 |
71 | %a(href="#{download_link}")
72 | download #{download[:version]}.
73 |
74 | - else
75 | %a(href="#{download[:url]}") Learn more
76 |
77 | %h3 Other
78 |
79 | Historical downloads are still available on
80 | = succeed "." do
81 | %a(href="https://download.redis.io/releases/") https://download.redis.io/releases/
82 |
83 | %p
84 | %strong Scripts and other automatic downloads
85 | can easily access the tarball of the latest Redis stable version at
86 | = succeed "," do
87 | %a{:href => "https://download.redis.io/redis-stable.tar.gz"} https://download.redis.io/redis-stable.tar.gz
88 | and its respective SHA256 sum at
89 | = succeed "." do
90 | %a{:href => "https://download.redis.io/redis-stable.tar.gz.SHA256SUM"} https://download.redis.io/redis-stable.tar.gz.SHA256SUM
91 | The source code of the latest stable release is
92 | = succeed "," do
93 | %a{:href => "https://download.redis.io/redis-stable"} always browsable here
94 | use the file
95 | %strong src/version.h
96 | in order to extract the version in an automatic way.
97 |
98 | %h2 How to verify files for integrity
99 |
100 | %p
101 | The Github repository
102 | %a(href="https://github.com/redis/redis-hashes/blob/master/README") redis-hashes
103 | contains a README file with SHA1 digests of released tarball archives.
104 | Note: the generic redis-stable.tar.gz tarball does not match any hash because it is modified to untar to the redis-stable directory.
105 |
106 | %h2 Installation
107 | %h3 From source code
108 | %p Download, extract and compile Redis with:
109 | %pre
110 | %code
111 | :preserve
112 | $ wget https://download.redis.io/releases/redis-#{STABLE_VERSION}.tar.gz
113 | $ tar xzf redis-#{STABLE_VERSION}.tar.gz
114 | $ cd redis-#{STABLE_VERSION}
115 | $ make
116 | %p
117 | The binaries that are now compiled are available in the
118 | %code src
119 | directory. Run Redis with:
120 | %pre
121 | %code
122 | :preserve
123 | $ src/redis-server
124 | %p You can interact with Redis using the built-in client:
125 | %pre
126 | %code
127 | :preserve
128 | $ src/redis-cli
129 | redis> set foo bar
130 | OK
131 | redis> get foo
132 | "bar"
133 | %h3 From the official Debian/Ubuntu APT Repository [Beta]
134 | %p
135 | You can install recent stable versions of Redis from the official
136 | packages.redis.io
APT repository. Add the repository to the
137 | apt
index, update it and install:
138 | %pre
139 | %code
140 | :preserve
141 | $ curl -fsSL https://packages.redis.io/gpg | sudo gpg --dearmor -o /usr/share/keyrings/redis-archive-keyring.gpg
142 | $ echo "deb [signed-by=/usr/share/keyrings/redis-archive-keyring.gpg] https://packages.redis.io/deb $(lsb_release -cs) main" | sudo tee /etc/apt/sources.list.d/redis.list
143 | $ sudo apt-get update
144 | $ sudo apt-get install redis
145 | %h3 From the official Ubuntu PPA
146 | %p
147 | You can install the latest stable version of Redis from the
148 | redislabs/redis
package repository. Add the repository to the
149 | apt
index, update it and install:
150 | %pre
151 | %code
152 | :preserve
153 | $ sudo add-apt-repository ppa:redislabs/redis
154 | $ sudo apt-get update
155 | $ sudo apt-get install redis
156 | %h3 From Snapcraft
157 | %p
158 | You can install the latest stable version of Redis from the Snapcraft
159 | marketplace:
160 | %pre
161 | %code
162 | :preserve
163 | $ sudo snap install redis
164 | %p
165 | Are you new to Redis? Try our
166 | = succeed "." do
167 | %a{:href => "http://try.redis.io"} online, interactive tutorial
168 |
169 |
--------------------------------------------------------------------------------
/views/grid.scss:
--------------------------------------------------------------------------------
1 | /////////////////
2 | // Semantic.gs // for SCSS: http://sass-lang.com/
3 | /////////////////
4 |
5 | // Defaults which you can freely override
6 | $column-width: 60px;
7 | $gutter-width: 20px;
8 | $columns: 12;
9 |
10 | // Utility function — you should never need to modify this
11 | @function gridsystem-width($columns:$columns) {
12 | @return ($column-width * $columns) + ($gutter-width * $columns);
13 | }
14 |
15 | // Set $total-width to 100% for a fluid layout
16 | $total-width: 100%;
17 |
18 | // Uncomment these two lines and the star-hack width/margin lines below to enable sub-pixel fix for IE6 & 7. See http://tylertate.com/blog/2012/01/05/subpixel-rounding.html
19 | // $min-width: 999999;
20 | // $correction: 0.5 / $min-width * 100;
21 |
22 | // The micro clearfix http://nicolasgallagher.com/micro-clearfix-hack/
23 | @mixin clearfix() {
24 | *zoom:1;
25 |
26 | &:before,
27 | &:after {
28 | content:"";
29 | display:table;
30 | }
31 | &:after {
32 | clear:both;
33 | }
34 | }
35 |
36 |
37 | //////////
38 | // GRID //
39 | //////////
40 |
41 | body {
42 | width: 100%;
43 | @include clearfix();
44 | }
45 |
46 | @mixin row($columns:$columns) {
47 | display: block;
48 | width: $total-width*(($gutter-width + gridsystem-width($columns))/gridsystem-width($columns));
49 | margin: 0 $total-width*((($gutter-width*.5)/gridsystem-width($columns))*-1);
50 | // *width: $total-width*(($gutter-width + gridsystem-width($columns))/gridsystem-width($columns))-$correction;
51 | // *margin: 0 $total-width*((($gutter-width*.5)/gridsystem-width($columns))*-1)-$correction;
52 | @include clearfix();
53 | }
54 | @mixin column($x,$columns:$columns) {
55 | display: inline;
56 | float: left;
57 | width: $total-width*(((($gutter-width+$column-width)*$x)-$gutter-width) / gridsystem-width($columns));
58 | margin: 0 $total-width*(($gutter-width*.5)/gridsystem-width($columns));
59 | // *width: $total-width*(((($gutter-width+$column-width)*$x)-$gutter-width) / gridsystem-width($columns))-$correction;
60 | // *margin: 0 $total-width*(($gutter-width*.5)/gridsystem-width($columns))-$correction;
61 | }
62 | @mixin push($offset:1) {
63 | margin-left: $total-width*((($gutter-width+$column-width)*$offset) / gridsystem-width($columns)) + $total-width*(($gutter-width*.5)/gridsystem-width($columns));
64 | }
65 | @mixin pull($offset:1) {
66 | margin-right: $total-width*((($gutter-width+$column-width)*$offset) / gridsystem-width($columns)) + $total-width*(($gutter-width*.5)/gridsystem-width($columns));
67 | }
68 |
--------------------------------------------------------------------------------
/views/home.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | %section.home-callout.home-section
3 | .title
4 | != Oga.parse_html(custom_view("#{documentation_path}/topics/introduction.md")).at_xpath("//p").text
5 |
6 | %a.learn-more(href="/topics/introduction") Learn more →
7 |
8 | .home-intro.home-section
9 |
10 | %section
11 | %h2 Try it
12 |
13 | %p
14 | Ready for a test drive? Check this
15 | %a(href="http://try.redis.io") interactive tutorial
16 | that will walk you through the most important features of
17 | Redis.
18 |
19 | %section
20 | %h2 Download it
21 |
22 | %p
23 | %a(href="https://download.redis.io/releases/redis-#{STABLE_VERSION}.tar.gz") Redis #{STABLE_VERSION} is the latest stable version.
24 | Interested in release candidates or unstable versions?
25 | %a(href="/download") Check the downloads page.
26 |
27 | %section
28 | %h2 Quick links
29 | %p
30 | Follow day-to-day Redis on Twitter and
31 | GitHub.
32 | Get help or help others by subscribing to our mailing list, we are 5,000 and counting!
33 |
34 | .home-buzz.home-section
35 | %section#buzz(data-limit="5")
36 | %h2 Redis News
37 |
38 | %ul
39 |
40 | %a(href="/buzz") More...
41 |
--------------------------------------------------------------------------------
/views/interactive.haml:
--------------------------------------------------------------------------------
1 | %div(class="example" data-session="#{session.namespace}")
2 | - lines.each do |line|
3 | %span(class="monospace prompt")= "redis> "
4 | %span(class="monospace command")= line
5 | %pre<= session.run(HTMLEntities.new.decode(line))
6 | %form>
7 | %span(class="monospace prompt")= "redis> "
8 | %input(type="text" name="command" autocomplete="off" spellcheck="false")
9 |
10 |
--------------------------------------------------------------------------------
/views/layout.haml:
--------------------------------------------------------------------------------
1 | !!! 5
2 |
3 | %html
4 | %head
5 | %meta(charset="utf-8")
6 | %title= [@title, "Redis"].compact.join(" – ")
7 | %link(rel="stylesheet" href="/styles.css")
8 | %link(rel="shortcut icon" href="/images/favicon.png")
9 | %link(rel="search" type="application/opensearchdescription+xml" title="Look up a Redis command" href="/opensearch.xml")
10 | %meta(name="viewport" content="width=device-width, minimum-scale=1.0, maximum-scale=1.0")
11 |
12 | :javascript
13 | (function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
14 | new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
15 | j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
16 | 'https://www.googletagmanager.com/gtm.js?id='+i+dl;f.parentNode.insertBefore(j,f);
17 | })(window,document,'script','dataLayer','GTM-T4MTBKP');
18 |
19 |
20 | %body(class="#{@css.join(" ") if @css}")
21 |
22 | %noscript
23 | %iframe(src="https://www.googletagmanager.com/ns.html?id=GTM-T4MTBKP"
24 | height="0" width="0"
25 | style="display:none;visibility:hidden")
26 |
27 | .mobile-menu.slideout-menu
28 | %header.menu-header
29 |
30 | %section.menu-section
31 | %ul.menu-section-list
32 | %li
33 | %a.home(href="/") Home
34 | %li
35 | %a(href="/commands") Commands
36 | %li
37 | %a(href="/clients") Clients
38 | %li
39 | %a(href="/documentation") Documentation
40 | %li
41 | %a(href="/community") Community
42 | %li
43 | %a(href="/download") Download
44 | %li
45 | %a(href="/modules") Modules
46 | %li
47 | %a(href="/support") Support
48 |
49 | .site-wrapper
50 |
51 | %header.site-header
52 | %nav.container
53 | .mobile-header
54 | %button.btn-hamburger.js-slideout-toggle
55 | %span.fa.fa-bars
56 |
57 | %a.home(href="/")
58 | %img(src="/images/redis-white.png" alt="Redis")
59 |
60 | %a.tryfree(href="https://redis.com/try-free") Try Free
61 |
62 | .desktop-header
63 | %a.home(href="/")
64 | %img(src="/images/redis-white.png" alt="Redis")
65 | %a(href="/commands") Commands
66 | %a(href="/clients") Clients
67 | %a(href="/documentation") Documentation
68 | %a(href="/community") Community
69 | %a(href="/download") Download
70 | %a(href="/modules") Modules
71 | %a(href="/support") Support
72 | %a.tryfree(href="https://redis.com/try-free") Try Free
73 |
74 | - if Date.today < Date.new(2022, 4, 1)
75 | %aside.wide-callout
76 | .container
77 | %a(href="https://redis.com/redisdays/") RedisDays registration
78 | is open! Don’t miss our big announcements.
79 |
80 | = content
81 |
82 | %footer.site-footer
83 | .container
84 | %p
85 | This website is
86 | open source software
87 | and is sponsored by
88 | Redis Ltd.
89 | See all credits.
90 |
91 | %script(src="https://ajax.googleapis.com/ajax/libs/jquery/1.4/jquery.min.js")
92 | %script(src="/app.js?#{File.mtime("public/app.js").to_i}")
93 |
--------------------------------------------------------------------------------
/views/modules.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 | %section#clients
4 |
5 | %h1 Redis Modules
6 |
7 | %p
8 | This is a list of Redis modules, for Redis v4.0 or greater, ordered by Github stars. This list contains two set of modules: modules under an OSI approved license, and modules that are under some proprietary license. Non OSI modules are clearly flagged as not open source. Also to have the source code hosted at Github is currently mandatory. To add your module here please send a pull request for the modules.json file in the Redis-doc repository. More information about Redis modules can be found here.
9 |
10 | %table
11 | %col(class="name")
12 | %col(class="license")
13 | %col(class="stars")
14 | %col(class="repository")
15 | %col(class="description")
16 | %col(class="authors")
17 |
18 | - @modules.each do |mod|
19 | %tr
20 | %td
21 | = mod["name"]
22 |
23 | %td
24 | - if mod["repository"]
25 | %a(href="#{mod["repository"]}" class="icon")
26 | %i(class="fa fa-home")
27 |
28 | %td.description
29 | = mod["description"]
30 |
31 | %td.authors
32 | - if mod["authors"]
33 | - mod["authors"].each do |author|
34 | %a(href="https://github.com/#{author}") #{author}
35 |
36 | %td
37 | = mod["license"]
38 |
39 | %td
40 | #{mod["stars"]}
41 | %i(class="fa fa-star icon-star")
42 |
--------------------------------------------------------------------------------
/views/normalize.scss:
--------------------------------------------------------------------------------
1 | /*! normalize.css v3.0.2 | MIT License | git.io/normalize */
2 |
3 | /**
4 | * 1. Set default font family to sans-serif.
5 | * 2. Prevent iOS text size adjust after orientation change, without disabling
6 | * user zoom.
7 | */
8 |
9 | html {
10 | font-family: sans-serif; /* 1 */
11 | -ms-text-size-adjust: 100%; /* 2 */
12 | -webkit-text-size-adjust: 100%; /* 2 */
13 | }
14 |
15 | /**
16 | * Remove default margin.
17 | */
18 |
19 | body {
20 | margin: 0;
21 | }
22 |
23 | /* HTML5 display definitions
24 | ========================================================================== */
25 |
26 | /**
27 | * Correct `block` display not defined for any HTML5 element in IE 8/9.
28 | * Correct `block` display not defined for `details` or `summary` in IE 10/11
29 | * and Firefox.
30 | * Correct `block` display not defined for `main` in IE 11.
31 | */
32 |
33 | article,
34 | aside,
35 | details,
36 | figcaption,
37 | figure,
38 | footer,
39 | header,
40 | hgroup,
41 | main,
42 | menu,
43 | nav,
44 | section,
45 | summary {
46 | display: block;
47 | }
48 |
49 | /**
50 | * 1. Correct `inline-block` display not defined in IE 8/9.
51 | * 2. Normalize vertical alignment of `progress` in Chrome, Firefox, and Opera.
52 | */
53 |
54 | audio,
55 | canvas,
56 | progress,
57 | video {
58 | display: inline-block; /* 1 */
59 | vertical-align: baseline; /* 2 */
60 | }
61 |
62 | /**
63 | * Prevent modern browsers from displaying `audio` without controls.
64 | * Remove excess height in iOS 5 devices.
65 | */
66 |
67 | audio:not([controls]) {
68 | display: none;
69 | height: 0;
70 | }
71 |
72 | /**
73 | * Address `[hidden]` styling not present in IE 8/9/10.
74 | * Hide the `template` element in IE 8/9/11, Safari, and Firefox < 22.
75 | */
76 |
77 | [hidden],
78 | template {
79 | display: none;
80 | }
81 |
82 | /* Links
83 | ========================================================================== */
84 |
85 | /**
86 | * Remove the gray background color from active links in IE 10.
87 | */
88 |
89 | a {
90 | background-color: transparent;
91 | }
92 |
93 | /**
94 | * Improve readability when focused and also mouse hovered in all browsers.
95 | */
96 |
97 | a:active,
98 | a:hover {
99 | outline: 0;
100 | }
101 |
102 | /* Text-level semantics
103 | ========================================================================== */
104 |
105 | /**
106 | * Address styling not present in IE 8/9/10/11, Safari, and Chrome.
107 | */
108 |
109 | abbr[title] {
110 | border-bottom: 1px dotted;
111 | }
112 |
113 | /**
114 | * Address style set to `bolder` in Firefox 4+, Safari, and Chrome.
115 | */
116 |
117 | b,
118 | strong {
119 | font-weight: bold;
120 | }
121 |
122 | /**
123 | * Address styling not present in Safari and Chrome.
124 | */
125 |
126 | dfn {
127 | font-style: italic;
128 | }
129 |
130 | /**
131 | * Address variable `h1` font-size and margin within `section` and `article`
132 | * contexts in Firefox 4+, Safari, and Chrome.
133 | */
134 |
135 | h1 {
136 | font-size: 2em;
137 | margin: 0.67em 0;
138 | }
139 |
140 | /**
141 | * Address styling not present in IE 8/9.
142 | */
143 |
144 | mark {
145 | background: #ff0;
146 | color: #000;
147 | }
148 |
149 | /**
150 | * Address inconsistent and variable font size in all browsers.
151 | */
152 |
153 | small {
154 | font-size: 80%;
155 | }
156 |
157 | /**
158 | * Prevent `sub` and `sup` affecting `line-height` in all browsers.
159 | */
160 |
161 | sub,
162 | sup {
163 | font-size: 75%;
164 | line-height: 0;
165 | position: relative;
166 | vertical-align: baseline;
167 | }
168 |
169 | sup {
170 | top: -0.5em;
171 | }
172 |
173 | sub {
174 | bottom: -0.25em;
175 | }
176 |
177 | /* Embedded content
178 | ========================================================================== */
179 |
180 | /**
181 | * Remove border when inside `a` element in IE 8/9/10.
182 | */
183 |
184 | img {
185 | border: 0;
186 | }
187 |
188 | /**
189 | * Correct overflow not hidden in IE 9/10/11.
190 | */
191 |
192 | svg:not(:root) {
193 | overflow: hidden;
194 | }
195 |
196 | /* Grouping content
197 | ========================================================================== */
198 |
199 | /**
200 | * Address margin not present in IE 8/9 and Safari.
201 | */
202 |
203 | figure {
204 | margin: 1em 40px;
205 | }
206 |
207 | /**
208 | * Address differences between Firefox and other browsers.
209 | */
210 |
211 | hr {
212 | -moz-box-sizing: content-box;
213 | box-sizing: content-box;
214 | height: 0;
215 | }
216 |
217 | /**
218 | * Contain overflow in all browsers.
219 | */
220 |
221 | pre {
222 | overflow: auto;
223 | }
224 |
225 | /**
226 | * Address odd `em`-unit font size rendering in all browsers.
227 | */
228 |
229 | code,
230 | kbd,
231 | pre,
232 | samp {
233 | font-family: monospace, monospace;
234 | font-size: 1em;
235 | }
236 |
237 | /* Forms
238 | ========================================================================== */
239 |
240 | /**
241 | * Known limitation: by default, Chrome and Safari on OS X allow very limited
242 | * styling of `select`, unless a `border` property is set.
243 | */
244 |
245 | /**
246 | * 1. Correct color not being inherited.
247 | * Known issue: affects color of disabled elements.
248 | * 2. Correct font properties not being inherited.
249 | * 3. Address margins set differently in Firefox 4+, Safari, and Chrome.
250 | */
251 |
252 | button,
253 | input,
254 | optgroup,
255 | select,
256 | textarea {
257 | color: inherit; /* 1 */
258 | font: inherit; /* 2 */
259 | margin: 0; /* 3 */
260 | }
261 |
262 | /**
263 | * Address `overflow` set to `hidden` in IE 8/9/10/11.
264 | */
265 |
266 | button {
267 | overflow: visible;
268 | }
269 |
270 | /**
271 | * Address inconsistent `text-transform` inheritance for `button` and `select`.
272 | * All other form control elements do not inherit `text-transform` values.
273 | * Correct `button` style inheritance in Firefox, IE 8/9/10/11, and Opera.
274 | * Correct `select` style inheritance in Firefox.
275 | */
276 |
277 | button,
278 | select {
279 | text-transform: none;
280 | }
281 |
282 | /**
283 | * 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio`
284 | * and `video` controls.
285 | * 2. Correct inability to style clickable `input` types in iOS.
286 | * 3. Improve usability and consistency of cursor style between image-type
287 | * `input` and others.
288 | */
289 |
290 | button,
291 | html input[type="button"], /* 1 */
292 | input[type="reset"],
293 | input[type="submit"] {
294 | -webkit-appearance: button; /* 2 */
295 | cursor: pointer; /* 3 */
296 | }
297 |
298 | /**
299 | * Re-set default cursor for disabled elements.
300 | */
301 |
302 | button[disabled],
303 | html input[disabled] {
304 | cursor: default;
305 | }
306 |
307 | /**
308 | * Remove inner padding and border in Firefox 4+.
309 | */
310 |
311 | button::-moz-focus-inner,
312 | input::-moz-focus-inner {
313 | border: 0;
314 | padding: 0;
315 | }
316 |
317 | /**
318 | * Address Firefox 4+ setting `line-height` on `input` using `!important` in
319 | * the UA stylesheet.
320 | */
321 |
322 | input {
323 | line-height: normal;
324 | }
325 |
326 | /**
327 | * It's recommended that you don't attempt to style these elements.
328 | * Firefox's implementation doesn't respect box-sizing, padding, or width.
329 | *
330 | * 1. Address box sizing set to `content-box` in IE 8/9/10.
331 | * 2. Remove excess padding in IE 8/9/10.
332 | */
333 |
334 | input[type="checkbox"],
335 | input[type="radio"] {
336 | box-sizing: border-box; /* 1 */
337 | padding: 0; /* 2 */
338 | }
339 |
340 | /**
341 | * Fix the cursor style for Chrome's increment/decrement buttons. For certain
342 | * `font-size` values of the `input`, it causes the cursor style of the
343 | * decrement button to change from `default` to `text`.
344 | */
345 |
346 | input[type="number"]::-webkit-inner-spin-button,
347 | input[type="number"]::-webkit-outer-spin-button {
348 | height: auto;
349 | }
350 |
351 | /**
352 | * 1. Address `appearance` set to `searchfield` in Safari and Chrome.
353 | * 2. Address `box-sizing` set to `border-box` in Safari and Chrome
354 | * (include `-moz` to future-proof).
355 | */
356 |
357 | input[type="search"] {
358 | -webkit-appearance: textfield; /* 1 */
359 | -moz-box-sizing: content-box;
360 | -webkit-box-sizing: content-box; /* 2 */
361 | box-sizing: content-box;
362 | }
363 |
364 | /**
365 | * Remove inner padding and search cancel button in Safari and Chrome on OS X.
366 | * Safari (but not Chrome) clips the cancel button when the search input has
367 | * padding (and `textfield` appearance).
368 | */
369 |
370 | input[type="search"]::-webkit-search-cancel-button,
371 | input[type="search"]::-webkit-search-decoration {
372 | -webkit-appearance: none;
373 | }
374 |
375 | /**
376 | * Define consistent border, margin, and padding.
377 | */
378 |
379 | fieldset {
380 | border: 1px solid #c0c0c0;
381 | margin: 0 2px;
382 | padding: 0.35em 0.625em 0.75em;
383 | }
384 |
385 | /**
386 | * 1. Correct `color` not being inherited in IE 8/9/10/11.
387 | * 2. Remove padding so people aren't caught out if they zero out fieldsets.
388 | */
389 |
390 | legend {
391 | border: 0; /* 1 */
392 | padding: 0; /* 2 */
393 | }
394 |
395 | /**
396 | * Remove default vertical scrollbar in IE 8/9/10/11.
397 | */
398 |
399 | textarea {
400 | overflow: auto;
401 | }
402 |
403 | /**
404 | * Don't inherit the `font-weight` (applied by a rule above).
405 | * NOTE: the default cannot safely be changed in Chrome and Safari on OS X.
406 | */
407 |
408 | optgroup {
409 | font-weight: bold;
410 | }
411 |
412 | /* Tables
413 | ========================================================================== */
414 |
415 | /**
416 | * Remove most spacing between table cells.
417 | */
418 |
419 | table {
420 | border-collapse: collapse;
421 | border-spacing: 0;
422 | }
423 |
424 | td,
425 | th {
426 | padding: 0;
427 | }
428 |
--------------------------------------------------------------------------------
/views/support.md:
--------------------------------------------------------------------------------
1 | Support
2 | ===
3 |
4 | Commercial Support
5 | ---
6 |
7 | [Redis Ltd.](https://redis.com) is the official sponsor of open source Redis.
8 |
9 | In addition, Redis Ltd. provides commercial support with Redis Enterprise, available as [a serverless fully-managed cloud service](https://redis.com/products/redis-cloud/) and an [on-premises software deployment](https://redis.com/redis-enterprise/advantages).
10 |
11 | Redis Enterprise simplifies the management of Redis at scale and includes [advanced security](https://redisl.com/enterprise-grade-redis-security/), [active-active geo distribution](https://redis.com/redis-enterprise/technology/active-active-geo-distribution/), and on-call customer support.
12 |
13 | Redis Ltd. also develops several Redis modules that extend the functionality of Redis. These include [RediSearch](https://oss.redis.com/redisearch/) (querying, indexing, and full-text search for Redis), [RedisJSON](https://oss.redis.com/redisjson/) (JSON for Redis), [RedisAI] (https://oss.redis.com/redisai/) (running AI inferencing closer to where you data lives, in Redis), [RedisGraph] (https://oss.redis.com/redisgraph) (a Graph database on Redis), [RedisTimeSeries](https://oss.redis.com/redistimeseries/) (a time series database on Redis), [RedisBloom](https://oss.redis.com/redisbloom/) (advanced probabilistic data-structures for Redis: bloom and cuckoo filters, count-min sketch, top-k) and [RedisGears](https://oss.redis.com/redisgears/) (a distributed programmability engine for Redis).
14 |
15 |
16 | Community Support
17 | ---
18 |
19 | Visit the [Community](/community) page for public support options and details on how to contribute to Redis.
20 |
--------------------------------------------------------------------------------
/views/topics/name.haml:
--------------------------------------------------------------------------------
1 | .site-content
2 | .text
3 |
4 | %article#topic
5 |
6 | - unless @related_commands.nil? || @related_commands.empty?
7 | %aside
8 | %h2
9 | Related commands
10 |
11 | %ul
12 | - @related_commands.each do |command|
13 | %li
14 | %a(href="/commands/#{command.to_param}")= command.name
15 |
16 | ~ @body
17 |
--------------------------------------------------------------------------------