kinetic_sdk 0.1.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/GettingStarted.md +64 -0
- data/README.md +383 -0
- data/bin/console +12 -0
- data/bin/setup +8 -0
- data/gems/kontena-websocket-client-0.1.1/Gemfile +4 -0
- data/gems/kontena-websocket-client-0.1.1/LICENSE +190 -0
- data/gems/kontena-websocket-client-0.1.1/README.md +138 -0
- data/gems/kontena-websocket-client-0.1.1/Rakefile +6 -0
- data/gems/kontena-websocket-client-0.1.1/benchmark/benchmark-client.rb +45 -0
- data/gems/kontena-websocket-client-0.1.1/benchmark/benchmark-em.rb +66 -0
- data/gems/kontena-websocket-client-0.1.1/benchmark/benchmark.rb +161 -0
- data/gems/kontena-websocket-client-0.1.1/benchmark/benchmark.sh +17 -0
- data/gems/kontena-websocket-client-0.1.1/benchmark/websocket-echo-server.go +207 -0
- data/gems/kontena-websocket-client-0.1.1/examples/websocket-echo-client.rb +80 -0
- data/gems/kontena-websocket-client-0.1.1/kontena-websocket-client.gemspec +24 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/client/connection.rb +119 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/client/version.rb +13 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/client.rb +848 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/error.rb +81 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/logging.rb +55 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena/websocket/openssl_patch.rb +10 -0
- data/gems/kontena-websocket-client-0.1.1/lib/kontena-websocket-client.rb +15 -0
- data/gems/mime-types-3.1/Code-of-Conduct.rdoc +74 -0
- data/gems/mime-types-3.1/Contributing.rdoc +130 -0
- data/gems/mime-types-3.1/History.rdoc +658 -0
- data/gems/mime-types-3.1/Licence.rdoc +25 -0
- data/gems/mime-types-3.1/Manifest.txt +31 -0
- data/gems/mime-types-3.1/README.rdoc +202 -0
- data/gems/mime-types-3.1/Rakefile +254 -0
- data/gems/mime-types-3.1/lib/mime/type/columnar.rb +55 -0
- data/gems/mime-types-3.1/lib/mime/type.rb +573 -0
- data/gems/mime-types-3.1/lib/mime/types/_columnar.rb +135 -0
- data/gems/mime-types-3.1/lib/mime/types/cache.rb +56 -0
- data/gems/mime-types-3.1/lib/mime/types/columnar.rb +1 -0
- data/gems/mime-types-3.1/lib/mime/types/container.rb +30 -0
- data/gems/mime-types-3.1/lib/mime/types/deprecations.rb +32 -0
- data/gems/mime-types-3.1/lib/mime/types/full.rb +17 -0
- data/gems/mime-types-3.1/lib/mime/types/loader.rb +148 -0
- data/gems/mime-types-3.1/lib/mime/types/logger.rb +37 -0
- data/gems/mime-types-3.1/lib/mime/types/registry.rb +81 -0
- data/gems/mime-types-3.1/lib/mime/types.rb +228 -0
- data/gems/mime-types-3.1/lib/mime-types.rb +1 -0
- data/gems/mime-types-3.1/test/bad-fixtures/malformed +9 -0
- data/gems/mime-types-3.1/test/fixture/json.json +1 -0
- data/gems/mime-types-3.1/test/fixture/old-data +9 -0
- data/gems/mime-types-3.1/test/fixture/yaml.yaml +55 -0
- data/gems/mime-types-3.1/test/minitest_helper.rb +13 -0
- data/gems/mime-types-3.1/test/test_mime_type.rb +603 -0
- data/gems/mime-types-3.1/test/test_mime_types.rb +161 -0
- data/gems/mime-types-3.1/test/test_mime_types_cache.rb +109 -0
- data/gems/mime-types-3.1/test/test_mime_types_class.rb +155 -0
- data/gems/mime-types-3.1/test/test_mime_types_lazy.rb +43 -0
- data/gems/mime-types-3.1/test/test_mime_types_loader.rb +32 -0
- data/gems/mime-types-data-3.2016.0521/Code-of-Conduct.md +75 -0
- data/gems/mime-types-data-3.2016.0521/Contributing.md +157 -0
- data/gems/mime-types-data-3.2016.0521/History.md +351 -0
- data/gems/mime-types-data-3.2016.0521/Licence.md +25 -0
- data/gems/mime-types-data-3.2016.0521/Manifest.txt +18 -0
- data/gems/mime-types-data-3.2016.0521/README.md +61 -0
- data/gems/mime-types-data-3.2016.0521/Rakefile +80 -0
- data/gems/mime-types-data-3.2016.0521/data/mime-types.json +1 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.content_type.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.docs.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.encoding.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.flags.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.friendly.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.pext.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.use_instead.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/data/mime.xrefs.column +1964 -0
- data/gems/mime-types-data-3.2016.0521/lib/mime/types/data.rb +21 -0
- data/gems/mime-types-data-3.2016.0521/lib/mime-types-data.rb +3 -0
- data/gems/multipart-post-2.0.0/Gemfile +14 -0
- data/gems/multipart-post-2.0.0/History.txt +60 -0
- data/gems/multipart-post-2.0.0/Manifest.txt +9 -0
- data/gems/multipart-post-2.0.0/README.md +77 -0
- data/gems/multipart-post-2.0.0/Rakefile +9 -0
- data/gems/multipart-post-2.0.0/lib/composite_io.rb +108 -0
- data/gems/multipart-post-2.0.0/lib/multipart_post.rb +9 -0
- data/gems/multipart-post-2.0.0/lib/multipartable.rb +29 -0
- data/gems/multipart-post-2.0.0/lib/net/http/post/multipart.rb +27 -0
- data/gems/multipart-post-2.0.0/lib/parts.rb +96 -0
- data/gems/multipart-post-2.0.0/multipart-post.gemspec +22 -0
- data/gems/multipart-post-2.0.0/test/multibyte.txt +1 -0
- data/gems/multipart-post-2.0.0/test/net/http/post/test_multipart.rb +110 -0
- data/gems/multipart-post-2.0.0/test/test_composite_io.rb +115 -0
- data/gems/multipart-post-2.0.0/test/test_parts.rb +86 -0
- data/gems/parallel-1.12.1/MIT-LICENSE.txt +20 -0
- data/gems/parallel-1.12.1/lib/parallel/processor_count.rb +93 -0
- data/gems/parallel-1.12.1/lib/parallel/version.rb +3 -0
- data/gems/parallel-1.12.1/lib/parallel.rb +500 -0
- data/gems/ruby-progressbar-1.9.0/LICENSE.txt +19 -0
- data/gems/ruby-progressbar-1.9.0/README.md +38 -0
- data/gems/ruby-progressbar-1.9.0/Rakefile +2 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/base.rb +183 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/calculators/length.rb +99 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/calculators/running_average.rb +9 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components/bar.rb +96 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components/percentage.rb +29 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components/rate.rb +43 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components/time.rb +107 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components/title.rb +13 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/components.rb +5 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/errors/invalid_progress_error.rb +4 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/format/formatter.rb +27 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/format/molecule.rb +59 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/format/string.rb +36 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/format.rb +3 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/output.rb +68 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/outputs/non_tty.rb +47 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/outputs/null.rb +33 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/outputs/tty.rb +32 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/progress.rb +118 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/refinements/enumerator.rb +25 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/refinements.rb +1 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/throttle.rb +25 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/time.rb +30 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/timer.rb +72 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar/version.rb +3 -0
- data/gems/ruby-progressbar-1.9.0/lib/ruby-progressbar.rb +18 -0
- data/gems/slugify-1.0.7/README +18 -0
- data/gems/slugify-1.0.7/lib/slugify.rb +1012 -0
- data/gems/slugify-1.0.7/lib/slugify_string.rb +11 -0
- data/gems/slugify-1.0.7/tests/slugify_test.rb +116 -0
- data/gems/slugify-1.0.7/tests/string_slugify_test.rb +23 -0
- data/gems/websocket-driver-0.6.5/CHANGELOG.md +123 -0
- data/gems/websocket-driver-0.6.5/LICENSE.md +22 -0
- data/gems/websocket-driver-0.6.5/README.md +369 -0
- data/gems/websocket-driver-0.6.5/examples/tcp_server.rb +28 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/Makefile +264 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/WebsocketMaskService.java +55 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/extconf.rb +4 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/websocket_mask.bundle +0 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/websocket_mask.c +41 -0
- data/gems/websocket-driver-0.6.5/ext/websocket-driver/websocket_mask.o +0 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/client.rb +140 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/draft75.rb +102 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/draft76.rb +96 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/event_emitter.rb +54 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/headers.rb +45 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/hybi/frame.rb +20 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/hybi/message.rb +31 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/hybi.rb +406 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/proxy.rb +68 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/server.rb +80 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver/stream_reader.rb +55 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/driver.rb +199 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/http/headers.rb +112 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/http/request.rb +45 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/http/response.rb +29 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/http.rb +15 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/mask.rb +14 -0
- data/gems/websocket-driver-0.6.5/lib/websocket/websocket_mask.rb +2 -0
- data/gems/websocket-driver-0.6.5/lib/websocket_mask.bundle +0 -0
- data/gems/websocket-driver-0.6.5-java/CHANGELOG.md +123 -0
- data/gems/websocket-driver-0.6.5-java/LICENSE.md +22 -0
- data/gems/websocket-driver-0.6.5-java/README.md +369 -0
- data/gems/websocket-driver-0.6.5-java/examples/tcp_server.rb +28 -0
- data/gems/websocket-driver-0.6.5-java/ext/websocket-driver/WebsocketMaskService.java +55 -0
- data/gems/websocket-driver-0.6.5-java/ext/websocket-driver/extconf.rb +4 -0
- data/gems/websocket-driver-0.6.5-java/ext/websocket-driver/websocket_mask.c +41 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/client.rb +140 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/draft75.rb +102 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/draft76.rb +96 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/event_emitter.rb +54 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/headers.rb +45 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/hybi/frame.rb +20 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/hybi/message.rb +31 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/hybi.rb +406 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/proxy.rb +68 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/server.rb +80 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver/stream_reader.rb +55 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/driver.rb +199 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/http/headers.rb +112 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/http/request.rb +45 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/http/response.rb +29 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/http.rb +15 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/mask.rb +14 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket/websocket_mask.rb +2 -0
- data/gems/websocket-driver-0.6.5-java/lib/websocket_mask.jar +0 -0
- data/gems/websocket-extensions-0.1.3/CHANGELOG.md +15 -0
- data/gems/websocket-extensions-0.1.3/LICENSE.md +20 -0
- data/gems/websocket-extensions-0.1.3/README.md +313 -0
- data/gems/websocket-extensions-0.1.3/lib/websocket/extensions/parser.rb +111 -0
- data/gems/websocket-extensions-0.1.3/lib/websocket/extensions.rb +181 -0
- data/lib/kinetic-sdk.rb +1 -0
- data/lib/kinetic_sdk/bridgehub/bridgehub-sdk.rb +80 -0
- data/lib/kinetic_sdk/bridgehub/lib/access_keys.rb +67 -0
- data/lib/kinetic_sdk/bridgehub/lib/bridge.rb +69 -0
- data/lib/kinetic_sdk/discussions/discussions-sdk.rb +165 -0
- data/lib/kinetic_sdk/discussions/lib/discussions.rb +107 -0
- data/lib/kinetic_sdk/discussions/lib/invitations.rb +120 -0
- data/lib/kinetic_sdk/discussions/lib/messages.rb +190 -0
- data/lib/kinetic_sdk/discussions/lib/meta.rb +14 -0
- data/lib/kinetic_sdk/discussions/lib/participants.rb +64 -0
- data/lib/kinetic_sdk/discussions/lib/related_items.rb +54 -0
- data/lib/kinetic_sdk/discussions/lib/websockets.rb +96 -0
- data/lib/kinetic_sdk/filehub/filehub-sdk.rb +80 -0
- data/lib/kinetic_sdk/filehub/lib/access_keys.rb +67 -0
- data/lib/kinetic_sdk/filehub/lib/filestores.rb +67 -0
- data/lib/kinetic_sdk/request_ce/lib/attribute_definitions.rb +153 -0
- data/lib/kinetic_sdk/request_ce/lib/bridges.rb +66 -0
- data/lib/kinetic_sdk/request_ce/lib/categories.rb +34 -0
- data/lib/kinetic_sdk/request_ce/lib/datastore_form.rb +110 -0
- data/lib/kinetic_sdk/request_ce/lib/datastore_submissions.rb +157 -0
- data/lib/kinetic_sdk/request_ce/lib/form.rb +99 -0
- data/lib/kinetic_sdk/request_ce/lib/form_types.rb +51 -0
- data/lib/kinetic_sdk/request_ce/lib/jwt.rb +55 -0
- data/lib/kinetic_sdk/request_ce/lib/kapp.rb +93 -0
- data/lib/kinetic_sdk/request_ce/lib/meta.rb +14 -0
- data/lib/kinetic_sdk/request_ce/lib/oauth.rb +37 -0
- data/lib/kinetic_sdk/request_ce/lib/security_policy_definitions.rb +157 -0
- data/lib/kinetic_sdk/request_ce/lib/space.rb +89 -0
- data/lib/kinetic_sdk/request_ce/lib/submissions.rb +215 -0
- data/lib/kinetic_sdk/request_ce/lib/system_api.rb +69 -0
- data/lib/kinetic_sdk/request_ce/lib/teams.rb +113 -0
- data/lib/kinetic_sdk/request_ce/lib/users.rb +245 -0
- data/lib/kinetic_sdk/request_ce/lib/webhook_jobs.rb +142 -0
- data/lib/kinetic_sdk/request_ce/lib/webhooks.rb +192 -0
- data/lib/kinetic_sdk/request_ce/request-ce-sdk.rb +153 -0
- data/lib/kinetic_sdk/task/lib/access_keys.rb +94 -0
- data/lib/kinetic_sdk/task/lib/categories.rb +190 -0
- data/lib/kinetic_sdk/task/lib/config.rb +202 -0
- data/lib/kinetic_sdk/task/lib/engine.rb +42 -0
- data/lib/kinetic_sdk/task/lib/environment.rb +14 -0
- data/lib/kinetic_sdk/task/lib/errors.rb +165 -0
- data/lib/kinetic_sdk/task/lib/groups.rb +112 -0
- data/lib/kinetic_sdk/task/lib/handlers.rb +105 -0
- data/lib/kinetic_sdk/task/lib/health.rb +28 -0
- data/lib/kinetic_sdk/task/lib/license.rb +52 -0
- data/lib/kinetic_sdk/task/lib/policy_rules.rb +166 -0
- data/lib/kinetic_sdk/task/lib/setup.rb +56 -0
- data/lib/kinetic_sdk/task/lib/sources.rb +134 -0
- data/lib/kinetic_sdk/task/lib/tasks.rb +17 -0
- data/lib/kinetic_sdk/task/lib/trees.rb +274 -0
- data/lib/kinetic_sdk/task/lib/users.rb +75 -0
- data/lib/kinetic_sdk/task/task-sdk.rb +92 -0
- data/lib/kinetic_sdk/utils/kinetic-http-headers.rb +150 -0
- data/lib/kinetic_sdk/utils/kinetic-http-response.rb +75 -0
- data/lib/kinetic_sdk/utils/kinetic-http.rb +552 -0
- data/lib/kinetic_sdk/utils/logger.rb +70 -0
- data/lib/kinetic_sdk/utils/random.rb +30 -0
- data/lib/kinetic_sdk/version.rb +7 -0
- data/lib/kinetic_sdk.rb +57 -0
- metadata +414 -0
@@ -0,0 +1,313 @@
|
|
1
|
+
# websocket-extensions [![Build status](https://secure.travis-ci.org/faye/websocket-extensions-ruby.svg)](http://travis-ci.org/faye/websocket-extensions-ruby)
|
2
|
+
|
3
|
+
A minimal framework that supports the implementation of WebSocket extensions in
|
4
|
+
a way that's decoupled from the main protocol. This library aims to allow a
|
5
|
+
WebSocket extension to be written and used with any protocol library, by
|
6
|
+
defining abstract representations of frames and messages that allow modules to
|
7
|
+
co-operate.
|
8
|
+
|
9
|
+
`websocket-extensions` provides a container for registering extension plugins,
|
10
|
+
and provides all the functions required to negotiate which extensions to use
|
11
|
+
during a session via the `Sec-WebSocket-Extensions` header. By implementing the
|
12
|
+
APIs defined in this document, an extension may be used by any WebSocket library
|
13
|
+
based on this framework.
|
14
|
+
|
15
|
+
## Installation
|
16
|
+
|
17
|
+
```
|
18
|
+
$ gem install websocket-extensions
|
19
|
+
```
|
20
|
+
|
21
|
+
## Usage
|
22
|
+
|
23
|
+
There are two main audiences for this library: authors implementing the
|
24
|
+
WebSocket protocol, and authors implementing extensions. End users of a
|
25
|
+
WebSocket library or an extension should be able to use any extension by passing
|
26
|
+
it as an argument to their chosen protocol library, without needing to know how
|
27
|
+
either of them work, or how the `websocket-extensions` framework operates.
|
28
|
+
|
29
|
+
The library is designed with the aim that any protocol implementation and any
|
30
|
+
extension can be used together, so long as they support the same abstract
|
31
|
+
representation of frames and messages.
|
32
|
+
|
33
|
+
### Data types
|
34
|
+
|
35
|
+
The APIs provided by the framework rely on two data types; extensions will
|
36
|
+
expect to be given data and to be able to return data in these formats:
|
37
|
+
|
38
|
+
#### *Frame*
|
39
|
+
|
40
|
+
*Frame* is a structure representing a single WebSocket frame of any type. Frames
|
41
|
+
are simple objects that must have at least the following properties, which
|
42
|
+
represent the data encoded in the frame:
|
43
|
+
|
44
|
+
| property | description |
|
45
|
+
| ------------ | ------------------------------------------------------------------ |
|
46
|
+
| `final` | `true` if the `FIN` bit is set, `false` otherwise |
|
47
|
+
| `rsv1` | `true` if the `RSV1` bit is set, `false` otherwise |
|
48
|
+
| `rsv2` | `true` if the `RSV2` bit is set, `false` otherwise |
|
49
|
+
| `rsv3` | `true` if the `RSV3` bit is set, `false` otherwise |
|
50
|
+
| `opcode` | the numeric opcode (`0`, `1`, `2`, `8`, `9`, or `10`) of the frame |
|
51
|
+
| `masked` | `true` if the `MASK` bit is set, `false` otherwise |
|
52
|
+
| `masking_key` | a 4-byte string if `masked` is `true`, otherwise `nil` |
|
53
|
+
| `payload` | a string containing the (unmasked) application data |
|
54
|
+
|
55
|
+
#### *Message*
|
56
|
+
|
57
|
+
A *Message* represents a complete application message, which can be formed from
|
58
|
+
text, binary and continuation frames. It has the following properties:
|
59
|
+
|
60
|
+
| property | description |
|
61
|
+
| -------- | ----------------------------------------------------------------- |
|
62
|
+
| `rsv1` | `true` if the first frame of the message has the `RSV1` bit set |
|
63
|
+
| `rsv2` | `true` if the first frame of the message has the `RSV2` bit set |
|
64
|
+
| `rsv3` | `true` if the first frame of the message has the `RSV3` bit set |
|
65
|
+
| `opcode` | the numeric opcode (`1` or `2`) of the first frame of the message |
|
66
|
+
| `data` | the concatenation of all the frame payloads in the message |
|
67
|
+
|
68
|
+
### For driver authors
|
69
|
+
|
70
|
+
A driver author is someone implementing the WebSocket protocol proper, and who
|
71
|
+
wishes end users to be able to use WebSocket extensions with their library.
|
72
|
+
|
73
|
+
At the start of a WebSocket session, on both the client and the server side,
|
74
|
+
they should begin by creating an extension container and adding whichever
|
75
|
+
extensions they want to use.
|
76
|
+
|
77
|
+
```rb
|
78
|
+
require 'websocket/extensions'
|
79
|
+
require 'permessage_deflate'
|
80
|
+
|
81
|
+
exts = WebSocket::Extensions.new
|
82
|
+
exts.add(PermessageDeflate)
|
83
|
+
```
|
84
|
+
|
85
|
+
In the following examples, `exts` refers to this `Extensions` instance.
|
86
|
+
|
87
|
+
#### Client sessions
|
88
|
+
|
89
|
+
Clients will use the methods `generate_offer` and `activate(header)`.
|
90
|
+
|
91
|
+
As part of the handshake process, the client must send a
|
92
|
+
`Sec-WebSocket-Extensions` header to advertise that it supports the registered
|
93
|
+
extensions. This header should be generated using:
|
94
|
+
|
95
|
+
```rb
|
96
|
+
request_headers['Sec-WebSocket-Extensions'] = exts.generate_offer
|
97
|
+
```
|
98
|
+
|
99
|
+
This returns a string, for example `"permessage-deflate;
|
100
|
+
client_max_window_bits"`, that represents all the extensions the client is
|
101
|
+
offering to use, and their parameters. This string may contain multiple offers
|
102
|
+
for the same extension.
|
103
|
+
|
104
|
+
When the client receives the handshake response from the server, it should pass
|
105
|
+
the incoming `Sec-WebSocket-Extensions` header in to `exts` to activate the
|
106
|
+
extensions the server has accepted:
|
107
|
+
|
108
|
+
```rb
|
109
|
+
exts.activate(response_headers['Sec-WebSocket-Extensions'])
|
110
|
+
```
|
111
|
+
|
112
|
+
If the server has sent any extension responses that the client does not
|
113
|
+
recognize, or are in conflict with one another for use of RSV bits, or that use
|
114
|
+
invalid parameters for the named extensions, then `exts.activate` will `raise`.
|
115
|
+
In this event, the client driver should fail the connection with closing code
|
116
|
+
`1010`.
|
117
|
+
|
118
|
+
#### Server sessions
|
119
|
+
|
120
|
+
Servers will use the method `generate_response(header)`.
|
121
|
+
|
122
|
+
A server session needs to generate a `Sec-WebSocket-Extensions` header to send
|
123
|
+
in its handshake response:
|
124
|
+
|
125
|
+
```rb
|
126
|
+
client_offer = request_env['HTTP_SEC_WEBSOCKET_EXTENSIONS']
|
127
|
+
ext_response = exts.generate_response(client_offer)
|
128
|
+
|
129
|
+
response_headers['Sec-WebSocket-Extensions'] = ext_response
|
130
|
+
```
|
131
|
+
|
132
|
+
Calling `exts.generate_response(header)` activates those extensions the client
|
133
|
+
has asked to use, if they are registered, asks each extension for a set of
|
134
|
+
response parameters, and returns a string containing the response parameters for
|
135
|
+
all accepted extensions.
|
136
|
+
|
137
|
+
#### In both directions
|
138
|
+
|
139
|
+
Both clients and servers will use the methods `valid_frame_rsv(frame)`,
|
140
|
+
`process_incoming_message(message)` and `process_outgoing_message(message)`.
|
141
|
+
|
142
|
+
The WebSocket protocol requires that frames do not have any of the `RSV` bits
|
143
|
+
set unless there is an extension in use that allows otherwise. When processing
|
144
|
+
an incoming frame, sessions should pass a *Frame* object to:
|
145
|
+
|
146
|
+
```rb
|
147
|
+
exts.valid_frame_rsv(frame)
|
148
|
+
```
|
149
|
+
|
150
|
+
If this method returns `false`, the session should fail the WebSocket connection
|
151
|
+
with closing code `1002`.
|
152
|
+
|
153
|
+
To pass incoming messages through the extension stack, a session should
|
154
|
+
construct a *Message* object according to the above datatype definitions, and
|
155
|
+
call:
|
156
|
+
|
157
|
+
```rb
|
158
|
+
message = exts.process_incoming_message(message)
|
159
|
+
```
|
160
|
+
|
161
|
+
If any extensions fail to process the message, then this call will `raise` an
|
162
|
+
error and the session should fail the WebSocket connection with closing code
|
163
|
+
`1010`. Otherwise, `message` should be passed on to the application.
|
164
|
+
|
165
|
+
To pass outgoing messages through the extension stack, a session should
|
166
|
+
construct a *Message* as before, and call:
|
167
|
+
|
168
|
+
```rb
|
169
|
+
message = exts.process_outgoing_message(message)
|
170
|
+
```
|
171
|
+
|
172
|
+
If any extensions fail to process the message, then this call will `raise` an
|
173
|
+
error and the session should fail the WebSocket connection with closing code
|
174
|
+
`1010`. Otherwise, `message` should be converted into frames (with the message's
|
175
|
+
`rsv1`, `rsv2`, `rsv3` and `opcode` set on the first frame) and written to the
|
176
|
+
transport.
|
177
|
+
|
178
|
+
At the end of the WebSocket session (either when the protocol is explicitly
|
179
|
+
ended or the transport connection disconnects), the driver should call:
|
180
|
+
|
181
|
+
```rb
|
182
|
+
exts.close
|
183
|
+
```
|
184
|
+
|
185
|
+
### For extension authors
|
186
|
+
|
187
|
+
An extension author is someone implementing an extension that transforms
|
188
|
+
WebSocket messages passing between the client and server. They would like to
|
189
|
+
implement their extension once and have it work with any protocol library.
|
190
|
+
|
191
|
+
Extension authors will not install `websocket-extensions` or call it directly.
|
192
|
+
Instead, they should implement the following API to allow their extension to
|
193
|
+
plug into the `websocket-extensions` framework.
|
194
|
+
|
195
|
+
An `Extension` is any object that has the following properties:
|
196
|
+
|
197
|
+
| property | description |
|
198
|
+
| -------- | ---------------------------------------------------------------------------- |
|
199
|
+
| `name` | a string containing the name of the extension as used in negotiation headers |
|
200
|
+
| `type` | a string, must be `"permessage"` |
|
201
|
+
| `rsv1` | either `true` if the extension uses the RSV1 bit, `false` otherwise |
|
202
|
+
| `rsv2` | either `true` if the extension uses the RSV2 bit, `false` otherwise |
|
203
|
+
| `rsv3` | either `true` if the extension uses the RSV3 bit, `false` otherwise |
|
204
|
+
|
205
|
+
It must also implement the following methods:
|
206
|
+
|
207
|
+
```rb
|
208
|
+
ext.create_client_session
|
209
|
+
```
|
210
|
+
|
211
|
+
This returns a *ClientSession*, whose interface is defined below.
|
212
|
+
|
213
|
+
```rb
|
214
|
+
ext.create_server_session(offers)
|
215
|
+
```
|
216
|
+
|
217
|
+
This takes an array of offer params and returns a *ServerSession*, whose
|
218
|
+
interface is defined below. For example, if the client handshake contains the
|
219
|
+
offer header:
|
220
|
+
|
221
|
+
```
|
222
|
+
Sec-WebSocket-Extensions: permessage-deflate; server_no_context_takeover; server_max_window_bits=8, \
|
223
|
+
permessage-deflate; server_max_window_bits=15
|
224
|
+
```
|
225
|
+
|
226
|
+
then the `permessage-deflate` extension will receive the call:
|
227
|
+
|
228
|
+
```rb
|
229
|
+
ext.create_server_session([
|
230
|
+
{'server_no_context_takeover' => true, 'server_max_window_bits' => 8},
|
231
|
+
{'server_max_window_bits' => 15}
|
232
|
+
])
|
233
|
+
```
|
234
|
+
|
235
|
+
The extension must decide which set of parameters it wants to accept, if any,
|
236
|
+
and return a *ServerSession* if it wants to accept the parameters and `nil`
|
237
|
+
otherwise.
|
238
|
+
|
239
|
+
#### *ClientSession*
|
240
|
+
|
241
|
+
A *ClientSession* is the type returned by `ext.create_client_session`. It must
|
242
|
+
implement the following methods, as well as the *Session* API listed below.
|
243
|
+
|
244
|
+
```rb
|
245
|
+
client_session.generate_offer
|
246
|
+
# e.g. -> [
|
247
|
+
# {'server_no_context_takeover' => true, 'server_max_window_bits' => 8},
|
248
|
+
# {'server_max_window_bits' => 15}
|
249
|
+
# ]
|
250
|
+
```
|
251
|
+
|
252
|
+
This must return a set of parameters to include in the client's
|
253
|
+
`Sec-WebSocket-Extensions` offer header. If the session wants to offer multiple
|
254
|
+
configurations, it can return an array of sets of parameters as shown above.
|
255
|
+
|
256
|
+
```rb
|
257
|
+
client_session.activate(params) # -> true
|
258
|
+
```
|
259
|
+
|
260
|
+
This must take a single set of parameters from the server's handshake response
|
261
|
+
and use them to configure the client session. If the client accepts the given
|
262
|
+
parameters, then this method must return `true`. If it returns any other value,
|
263
|
+
the framework will interpret this as the client rejecting the response, and will
|
264
|
+
`raise`.
|
265
|
+
|
266
|
+
#### *ServerSession*
|
267
|
+
|
268
|
+
A *ServerSession* is the type returned by `ext.create_server_session(offers)`. It
|
269
|
+
must implement the following methods, as well as the *Session* API listed below.
|
270
|
+
|
271
|
+
```rb
|
272
|
+
server_session.generate_response
|
273
|
+
# e.g. -> {'server_max_window_bits' => 8}
|
274
|
+
```
|
275
|
+
|
276
|
+
This returns the set of parameters the server session wants to send in its
|
277
|
+
`Sec-WebSocket-Extensions` response header. Only one set of parameters is
|
278
|
+
returned to the client per extension. Server sessions that would confict on
|
279
|
+
their use of RSV bits are not activated.
|
280
|
+
|
281
|
+
#### *Session*
|
282
|
+
|
283
|
+
The *Session* API must be implemented by both client and server sessions. It
|
284
|
+
contains three methods: `process_incoming_message(message)` and
|
285
|
+
`process_outgoing_message(message)`.
|
286
|
+
|
287
|
+
```rb
|
288
|
+
message = session.process_incoming_message(message)
|
289
|
+
```
|
290
|
+
|
291
|
+
The session must implement this method to take an incoming *Message* as defined
|
292
|
+
above, transform it in any way it needs, then return it. If there is an error
|
293
|
+
processing the message, this method should `raise` an error.
|
294
|
+
|
295
|
+
```rb
|
296
|
+
message = session.process_outgoing_message(message)
|
297
|
+
```
|
298
|
+
|
299
|
+
The session must implement this method to take an outgoing *Message* as defined
|
300
|
+
above, transform it in any way it needs, then return it. If there is an error
|
301
|
+
processing the message, this method should `raise` an error.
|
302
|
+
|
303
|
+
```rb
|
304
|
+
session.close
|
305
|
+
```
|
306
|
+
|
307
|
+
The framework will call this method when the WebSocket session ends, allowing
|
308
|
+
the session to release any resources it's using.
|
309
|
+
|
310
|
+
## Examples
|
311
|
+
|
312
|
+
* Consumer: [websocket-driver](https://github.com/faye/websocket-driver-ruby)
|
313
|
+
* Provider: [permessage-deflate](https://github.com/faye/permessage-deflate-ruby)
|
@@ -0,0 +1,111 @@
|
|
1
|
+
require 'strscan'
|
2
|
+
|
3
|
+
module WebSocket
|
4
|
+
class Extensions
|
5
|
+
|
6
|
+
class Parser
|
7
|
+
TOKEN = /([!#\$%&'\*\+\-\.\^_`\|~0-9A-Za-z]+)/
|
8
|
+
NOTOKEN = /([^!#\$%&'\*\+\-\.\^_`\|~0-9A-Za-z])/
|
9
|
+
QUOTED = /"((?:\\[\x00-\x7f]|[^\x00-\x08\x0a-\x1f\x7f"])*)"/
|
10
|
+
PARAM = %r{#{TOKEN.source}(?:=(?:#{TOKEN.source}|#{QUOTED.source}))?}
|
11
|
+
EXT = %r{#{TOKEN.source}(?: *; *#{PARAM.source})*}
|
12
|
+
EXT_LIST = %r{^#{EXT.source}(?: *, *#{EXT.source})*$}
|
13
|
+
NUMBER = /^-?(0|[1-9][0-9]*)(\.[0-9]+)?$/
|
14
|
+
|
15
|
+
ParseError = Class.new(ArgumentError)
|
16
|
+
|
17
|
+
def self.parse_header(header)
|
18
|
+
offers = Offers.new
|
19
|
+
return offers if header == '' or header.nil?
|
20
|
+
|
21
|
+
unless header =~ EXT_LIST
|
22
|
+
raise ParseError, "Invalid Sec-WebSocket-Extensions header: #{header}"
|
23
|
+
end
|
24
|
+
|
25
|
+
scanner = StringScanner.new(header)
|
26
|
+
value = scanner.scan(EXT)
|
27
|
+
|
28
|
+
until value.nil?
|
29
|
+
params = value.scan(PARAM)
|
30
|
+
name = params.shift.first
|
31
|
+
offer = {}
|
32
|
+
|
33
|
+
params.each do |key, unquoted, quoted|
|
34
|
+
if unquoted
|
35
|
+
data = unquoted
|
36
|
+
elsif quoted
|
37
|
+
data = quoted.gsub(/\\/, '')
|
38
|
+
else
|
39
|
+
data = true
|
40
|
+
end
|
41
|
+
if data =~ NUMBER
|
42
|
+
data = data =~ /\./ ? data.to_f : data.to_i(10)
|
43
|
+
end
|
44
|
+
|
45
|
+
if offer.has_key?(key)
|
46
|
+
offer[key] = [*offer[key]] + [data]
|
47
|
+
else
|
48
|
+
offer[key] = data
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
offers.push(name, offer)
|
53
|
+
|
54
|
+
scanner.scan(/ *, */)
|
55
|
+
value = scanner.scan(EXT)
|
56
|
+
end
|
57
|
+
offers
|
58
|
+
end
|
59
|
+
|
60
|
+
def self.serialize_params(name, params)
|
61
|
+
values = []
|
62
|
+
|
63
|
+
print = lambda do |key, value|
|
64
|
+
case value
|
65
|
+
when Array then value.each { |v| print[key, v] }
|
66
|
+
when true then values.push(key)
|
67
|
+
when Numeric then values.push(key + '=' + value.to_s)
|
68
|
+
else
|
69
|
+
if value =~ NOTOKEN
|
70
|
+
values.push(key + '="' + value.gsub(/"/, '\"') + '"')
|
71
|
+
else
|
72
|
+
values.push(key + '=' + value)
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
params.keys.sort.each { |key| print[key, params[key]] }
|
78
|
+
|
79
|
+
([name] + values).join('; ')
|
80
|
+
end
|
81
|
+
end
|
82
|
+
|
83
|
+
class Offers
|
84
|
+
def initialize
|
85
|
+
@by_name = {}
|
86
|
+
@in_order = []
|
87
|
+
end
|
88
|
+
|
89
|
+
def push(name, params)
|
90
|
+
@by_name[name] ||= []
|
91
|
+
@by_name[name].push(params)
|
92
|
+
@in_order.push(:name => name, :params => params)
|
93
|
+
end
|
94
|
+
|
95
|
+
def each_offer(&block)
|
96
|
+
@in_order.each do |offer|
|
97
|
+
block.call(offer[:name], offer[:params])
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
def by_name(name)
|
102
|
+
@by_name[name] || []
|
103
|
+
end
|
104
|
+
|
105
|
+
def to_a
|
106
|
+
@in_order.dup
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
end
|
111
|
+
end
|
@@ -0,0 +1,181 @@
|
|
1
|
+
module WebSocket
|
2
|
+
class Extensions
|
3
|
+
|
4
|
+
autoload :Parser, File.expand_path('../extensions/parser', __FILE__)
|
5
|
+
|
6
|
+
ExtensionError = Class.new(ArgumentError)
|
7
|
+
|
8
|
+
MESSAGE_OPCODES = [1, 2]
|
9
|
+
|
10
|
+
def initialize
|
11
|
+
@rsv1 = @rsv2 = @rsv3 = nil
|
12
|
+
|
13
|
+
@by_name = {}
|
14
|
+
@in_order = []
|
15
|
+
@sessions = []
|
16
|
+
@index = {}
|
17
|
+
end
|
18
|
+
|
19
|
+
def add(ext)
|
20
|
+
unless ext.respond_to?(:name) and ext.name.is_a?(String)
|
21
|
+
raise TypeError, 'extension.name must be a string'
|
22
|
+
end
|
23
|
+
|
24
|
+
unless ext.respond_to?(:type) and ext.type == 'permessage'
|
25
|
+
raise TypeError, 'extension.type must be "permessage"'
|
26
|
+
end
|
27
|
+
|
28
|
+
unless ext.respond_to?(:rsv1) and [true, false].include?(ext.rsv1)
|
29
|
+
raise TypeError, 'extension.rsv1 must be true or false'
|
30
|
+
end
|
31
|
+
|
32
|
+
unless ext.respond_to?(:rsv2) and [true, false].include?(ext.rsv2)
|
33
|
+
raise TypeError, 'extension.rsv2 must be true or false'
|
34
|
+
end
|
35
|
+
|
36
|
+
unless ext.respond_to?(:rsv3) and [true, false].include?(ext.rsv3)
|
37
|
+
raise TypeError, 'extension.rsv3 must be true or false'
|
38
|
+
end
|
39
|
+
|
40
|
+
if @by_name.has_key?(ext.name)
|
41
|
+
raise TypeError, %Q{An extension with name "#{ext.name}" is already registered}
|
42
|
+
end
|
43
|
+
|
44
|
+
@by_name[ext.name] = ext
|
45
|
+
@in_order.push(ext)
|
46
|
+
end
|
47
|
+
|
48
|
+
def generate_offer
|
49
|
+
sessions = []
|
50
|
+
offer = []
|
51
|
+
index = {}
|
52
|
+
|
53
|
+
@in_order.each do |ext|
|
54
|
+
session = ext.create_client_session
|
55
|
+
next unless session
|
56
|
+
|
57
|
+
record = [ext, session]
|
58
|
+
sessions.push(record)
|
59
|
+
index[ext.name] = record
|
60
|
+
|
61
|
+
offers = session.generate_offer
|
62
|
+
offers = offers ? [offers].flatten : []
|
63
|
+
|
64
|
+
offers.each do |off|
|
65
|
+
offer.push(Parser.serialize_params(ext.name, off))
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
@sessions = sessions
|
70
|
+
@index = index
|
71
|
+
|
72
|
+
offer.size > 0 ? offer.join(', ') : nil
|
73
|
+
end
|
74
|
+
|
75
|
+
def activate(header)
|
76
|
+
responses = Parser.parse_header(header)
|
77
|
+
@sessions = []
|
78
|
+
|
79
|
+
responses.each_offer do |name, params|
|
80
|
+
unless record = @index[name]
|
81
|
+
raise ExtensionError, %Q{Server sent am extension response for unknown extension "#{name}"}
|
82
|
+
end
|
83
|
+
|
84
|
+
ext, session = *record
|
85
|
+
|
86
|
+
if reserved = reserved?(ext)
|
87
|
+
raise ExtensionError, %Q{Server sent two extension responses that use the RSV#{reserved[0]} } +
|
88
|
+
%Q{ bit: "#{reserved[1]}" and "#{ext.name}"}
|
89
|
+
end
|
90
|
+
|
91
|
+
unless session.activate(params) == true
|
92
|
+
raise ExtensionError, %Q{Server send unacceptable extension parameters: #{Parser.serialize_params(name, params)}}
|
93
|
+
end
|
94
|
+
|
95
|
+
reserve(ext)
|
96
|
+
@sessions.push(record)
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
100
|
+
def generate_response(header)
|
101
|
+
sessions = []
|
102
|
+
response = []
|
103
|
+
offers = Parser.parse_header(header)
|
104
|
+
|
105
|
+
@in_order.each do |ext|
|
106
|
+
offer = offers.by_name(ext.name)
|
107
|
+
next if offer.empty? or reserved?(ext)
|
108
|
+
|
109
|
+
next unless session = ext.create_server_session(offer)
|
110
|
+
|
111
|
+
reserve(ext)
|
112
|
+
sessions.push([ext, session])
|
113
|
+
response.push(Parser.serialize_params(ext.name, session.generate_response))
|
114
|
+
end
|
115
|
+
|
116
|
+
@sessions = sessions
|
117
|
+
response.size > 0 ? response.join(', ') : nil
|
118
|
+
end
|
119
|
+
|
120
|
+
def valid_frame_rsv(frame)
|
121
|
+
allowed = {:rsv1 => false, :rsv2 => false, :rsv3 => false}
|
122
|
+
|
123
|
+
if MESSAGE_OPCODES.include?(frame.opcode)
|
124
|
+
@sessions.each do |ext, session|
|
125
|
+
allowed[:rsv1] ||= ext.rsv1
|
126
|
+
allowed[:rsv2] ||= ext.rsv2
|
127
|
+
allowed[:rsv3] ||= ext.rsv3
|
128
|
+
end
|
129
|
+
end
|
130
|
+
|
131
|
+
(allowed[:rsv1] || !frame.rsv1) &&
|
132
|
+
(allowed[:rsv2] || !frame.rsv2) &&
|
133
|
+
(allowed[:rsv3] || !frame.rsv3)
|
134
|
+
end
|
135
|
+
alias :valid_frame_rsv? :valid_frame_rsv
|
136
|
+
|
137
|
+
def process_incoming_message(message)
|
138
|
+
@sessions.reverse.inject(message) do |msg, (ext, session)|
|
139
|
+
begin
|
140
|
+
session.process_incoming_message(msg)
|
141
|
+
rescue => error
|
142
|
+
raise ExtensionError, [ext.name, error.message].join(': ')
|
143
|
+
end
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
def process_outgoing_message(message)
|
148
|
+
@sessions.inject(message) do |msg, (ext, session)|
|
149
|
+
begin
|
150
|
+
session.process_outgoing_message(msg)
|
151
|
+
rescue => error
|
152
|
+
raise ExtensionError, [ext.name, error.message].join(': ')
|
153
|
+
end
|
154
|
+
end
|
155
|
+
end
|
156
|
+
|
157
|
+
def close
|
158
|
+
return unless @sessions
|
159
|
+
|
160
|
+
@sessions.each do |ext, session|
|
161
|
+
session.close rescue nil
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
private
|
166
|
+
|
167
|
+
def reserve(ext)
|
168
|
+
@rsv1 ||= ext.rsv1 && ext.name
|
169
|
+
@rsv2 ||= ext.rsv2 && ext.name
|
170
|
+
@rsv3 ||= ext.rsv3 && ext.name
|
171
|
+
end
|
172
|
+
|
173
|
+
def reserved?(ext)
|
174
|
+
return [1, @rsv1] if @rsv1 and ext.rsv1
|
175
|
+
return [2, @rsv2] if @rsv2 and ext.rsv2
|
176
|
+
return [3, @rsv3] if @rsv3 and ext.rsv3
|
177
|
+
false
|
178
|
+
end
|
179
|
+
|
180
|
+
end
|
181
|
+
end
|
data/lib/kinetic-sdk.rb
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), 'kinetic_sdk'))
|
@@ -0,0 +1,80 @@
|
|
1
|
+
Dir[File.join(File.dirname(File.expand_path(__FILE__)), "lib", "**", "*.rb")].each {|file| require file }
|
2
|
+
|
3
|
+
module KineticSdk
|
4
|
+
|
5
|
+
# Bridgehub is a Ruby class that acts as a wrapper for the Kinetic BridgeHub REST API
|
6
|
+
# without having to make explicit HTTP requests.
|
7
|
+
#
|
8
|
+
class Bridgehub
|
9
|
+
|
10
|
+
# Include the KineticHttpUtils module
|
11
|
+
include KineticSdk::Utils::KineticHttpUtils
|
12
|
+
|
13
|
+
attr_reader :api_url, :username, :options, :password, :server, :version
|
14
|
+
|
15
|
+
# Initalize the BridgeHub SDK with the web server URL and configuration user
|
16
|
+
# credentials, along with any custom option values.
|
17
|
+
#
|
18
|
+
# @param opts [Hash] Kinetic BridgeHub properties
|
19
|
+
# @option opts [String] :config_file optional - path to the YAML configuration file
|
20
|
+
#
|
21
|
+
# * Ex: /opt/config/bridgehub-configuration1.yaml
|
22
|
+
#
|
23
|
+
# @option opts [String] :app_server_url the URL to the Kinetic Bridgehub web application.
|
24
|
+
#
|
25
|
+
# * Ex: <http://192.168.0.1:8080/kinetic-bridgehub>
|
26
|
+
#
|
27
|
+
# @option opts [String] :username the username for the user
|
28
|
+
# @option opts [String] :password the password for the user
|
29
|
+
# @option opts [Hash<Symbol, Object>] :options ({}) optional settings
|
30
|
+
#
|
31
|
+
# * :log_level (String) (_defaults to: off_) level of logging - off | info | debug | trace
|
32
|
+
# * :max_redirects (Fixnum) (_defaults to: 10_) maximum number of redirects to follow
|
33
|
+
# * :ssl_ca_file (String) full path to PEM certificate used to verify the server
|
34
|
+
# * :ssl_verify_mode (String) (_defaults to: none_) - none | peer
|
35
|
+
#
|
36
|
+
# Example: using a configuration file
|
37
|
+
#
|
38
|
+
# KineticSdk::Bridgehub.new({
|
39
|
+
# config_file: "/opt/config1.yaml"
|
40
|
+
# })
|
41
|
+
#
|
42
|
+
# Example: using a properties hash
|
43
|
+
#
|
44
|
+
# KineticSdk::Bridgehub.new({
|
45
|
+
# app_server_url: "http://localhost:8080/kinetic-bridgehub",
|
46
|
+
# username: "admin",
|
47
|
+
# password: "admin",
|
48
|
+
# options: {
|
49
|
+
# log_level: "debug",
|
50
|
+
# ssl_verify_mode: "peer",
|
51
|
+
# ssl_ca_file: "/usr/local/self_signing_ca.pem"
|
52
|
+
# }
|
53
|
+
# })
|
54
|
+
#
|
55
|
+
# If the +config_file+ option is present, it will be loaded first, and any additional
|
56
|
+
# options will overwrite any values in the config file
|
57
|
+
#
|
58
|
+
def initialize(opts)
|
59
|
+
# initialize any variables
|
60
|
+
options = {}
|
61
|
+
|
62
|
+
# process the configuration file if it was provided
|
63
|
+
unless opts[:config_file].nil?
|
64
|
+
options.merge!(YAML::load_file opts[:config_file])
|
65
|
+
end
|
66
|
+
|
67
|
+
# process the configuration hash if it was provided
|
68
|
+
options.merge!(opts)
|
69
|
+
|
70
|
+
# process any individual options
|
71
|
+
@options = options.delete(:options) || {}
|
72
|
+
@username = options[:username]
|
73
|
+
@password = options[:password]
|
74
|
+
@server = options[:app_server_url].chomp('/')
|
75
|
+
@api_url = "#{@server}/app/manage-api/v1"
|
76
|
+
@version = 1
|
77
|
+
end
|
78
|
+
|
79
|
+
end
|
80
|
+
end
|