web-console 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/MIT-LICENSE +20 -0
- data/README.markdown +121 -0
- data/Rakefile +48 -0
- data/app/assets/javascripts/web_console/application.js +4 -0
- data/app/assets/javascripts/web_console/console_sessions.js +24 -0
- data/app/assets/stylesheets/web_console/application.css +13 -0
- data/app/assets/stylesheets/web_console/console_sessions.css +8 -0
- data/app/controllers/web_console/application_controller.rb +12 -0
- data/app/controllers/web_console/console_sessions_controller.rb +23 -0
- data/app/helpers/web_console/application_helper.rb +4 -0
- data/app/helpers/web_console/console_session_helper.rb +4 -0
- data/app/models/web_console/console_session.rb +107 -0
- data/app/views/layouts/web_console/application.html.erb +14 -0
- data/app/views/web_console/console_sessions/index.html.erb +4 -0
- data/config/routes.rb +5 -0
- data/lib/web-console.rb +1 -0
- data/lib/web_console.rb +7 -0
- data/lib/web_console/engine.rb +38 -0
- data/lib/web_console/fiber.rb +48 -0
- data/lib/web_console/repl.rb +59 -0
- data/lib/web_console/repl/dummy.rb +38 -0
- data/lib/web_console/repl/irb.rb +61 -0
- data/lib/web_console/stream.rb +27 -0
- data/lib/web_console/version.rb +3 -0
- data/test/controllers/web_console/console_sessions_controller_test.rb +57 -0
- data/test/dummy/README.rdoc +28 -0
- data/test/dummy/Rakefile +6 -0
- data/test/dummy/app/assets/javascripts/application.js +13 -0
- data/test/dummy/app/assets/stylesheets/application.css +13 -0
- data/test/dummy/app/controllers/application_controller.rb +5 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/views/layouts/application.html.erb +14 -0
- data/test/dummy/bin/bundle +3 -0
- data/test/dummy/bin/rails +4 -0
- data/test/dummy/bin/rake +4 -0
- data/test/dummy/config.ru +4 -0
- data/test/dummy/config/application.rb +15 -0
- data/test/dummy/config/boot.rb +5 -0
- data/test/dummy/config/database.yml +25 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +29 -0
- data/test/dummy/config/environments/production.rb +80 -0
- data/test/dummy/config/environments/test.rb +36 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/filter_parameter_logging.rb +4 -0
- data/test/dummy/config/initializers/inflections.rb +16 -0
- data/test/dummy/config/initializers/mime_types.rb +5 -0
- data/test/dummy/config/initializers/secret_token.rb +12 -0
- data/test/dummy/config/initializers/session_store.rb +3 -0
- data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
- data/test/dummy/config/locales/en.yml +23 -0
- data/test/dummy/config/routes.rb +2 -0
- data/test/dummy/db/development.sqlite3 +0 -0
- data/test/dummy/db/schema.rb +16 -0
- data/test/dummy/db/test.sqlite3 +0 -0
- data/test/dummy/log/development.log +26591 -0
- data/test/dummy/log/test.log +78368 -0
- data/test/dummy/public/404.html +58 -0
- data/test/dummy/public/422.html +58 -0
- data/test/dummy/public/500.html +57 -0
- data/test/dummy/public/favicon.ico +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/0280bb38c5058cc31c4fcd8c392a5ec4 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/05bd7b24c0a86010ebb28b50ac7cad52 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/2a48be7daff14cf56911b263cbe017c7 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/3a37adb5ebd079cf67dae597b8c2e4f8 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/4f79e5f341043e081becefe4952395c5 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/508ab3c25833ea537a5e3fc90df33595 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/5a8ab22e707dfc7ba00691f90e054d7e +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/5adcb7569cdd03204d650e285f19351f +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/654a1bde557d359b957dc0aa12b0dfa0 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/71f57313d6c92f5483915a6c2d79a506 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/7ec0041a47c34b44e52e836c01454a1a +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/a1534e0c08b73ad82c2edb8c364caa66 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/a841a3af20a321912acfed87036438fb +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/ad8eea7f774b674c29708ca90952764f +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/b0741545b4917192ba7b5e803c2e323d +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/b3159b06164dd474ff8efc3c84aefbba +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/b40b7c5b2003544010f30f0bd3fb81b2 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/bbbe6a3ce382662666a355d708c83d2d +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/c4e26d8dbebb3afd7013acfefa564dd1 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/ca53fb2717d5aac2f1c3939d9444fe3b +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/d6b37d10680a997662c379d0ff7cad27 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/e1d89809967e81220dca66770c50aa67 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/e3d5cafc071e8f9a8efc88fddf721947 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/f44e2a41bd51a92a84f99847ea675ba3 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/f4a21ed9cebe2c83a9d988504dc6720b +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/f4d45273ff5b44879dab0a16805f4309 +0 -0
- data/test/dummy/tmp/cache/assets/development/sprockets/f5deea0ae9671fdb035aefc6c4ba1109 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/0280bb38c5058cc31c4fcd8c392a5ec4 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/05bd7b24c0a86010ebb28b50ac7cad52 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/2a48be7daff14cf56911b263cbe017c7 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/3824a9e25e846a4916b3ac1d67060782 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/3a37adb5ebd079cf67dae597b8c2e4f8 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/4f79e5f341043e081becefe4952395c5 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/508ab3c25833ea537a5e3fc90df33595 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/5a8ab22e707dfc7ba00691f90e054d7e +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/5adcb7569cdd03204d650e285f19351f +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/632346eda030b596f513fff2de181743 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/654a1bde557d359b957dc0aa12b0dfa0 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/71f57313d6c92f5483915a6c2d79a506 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/9b9aee85f29dc573732fbb4001ceda00 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/a841a3af20a321912acfed87036438fb +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/bbbe6a3ce382662666a355d708c83d2d +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/ca53fb2717d5aac2f1c3939d9444fe3b +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/d6b37d10680a997662c379d0ff7cad27 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/e1d89809967e81220dca66770c50aa67 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/e3d5cafc071e8f9a8efc88fddf721947 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/f44e2a41bd51a92a84f99847ea675ba3 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/f4a21ed9cebe2c83a9d988504dc6720b +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/f4d45273ff5b44879dab0a16805f4309 +0 -0
- data/test/dummy/tmp/cache/assets/test/sprockets/f5deea0ae9671fdb035aefc6c4ba1109 +0 -0
- data/test/dummy/tmp/pids/server.pid +1 -0
- data/test/helpers/web_console/console_session_helper_test.rb +6 -0
- data/test/models/console_session_test.rb +110 -0
- data/test/test_helper.rb +15 -0
- data/test/web_console/repl/dummy_test.rb +54 -0
- data/test/web_console/repl/irb_test.rb +108 -0
- data/test/web_console/repl_test.rb +15 -0
- data/test/web_console_test.rb +91 -0
- data/vendor/assets/javascripts/jquery.console.js +727 -0
- metadata +303 -0
checksums.yaml
ADDED
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
---
|
|
2
|
+
SHA1:
|
|
3
|
+
metadata.gz: 852f83191e70586981bb3ead966b52a03a69d98e
|
|
4
|
+
data.tar.gz: cdaf295bc404be9d8c83d7749a06b81036349f52
|
|
5
|
+
SHA512:
|
|
6
|
+
metadata.gz: b8709b25d8e845212ccf470c10ac4ccbca756ebe7491dbf4a1e468c9a7b3ae4b9459ed6ae3c976fe1c02303124f47cd3008827bd5d264117680828966c576cae
|
|
7
|
+
data.tar.gz: 8707a5e85486c63ff7c7e79e23943333fb8c53d21b4549f2c7cb1b557600c9c85bd6ddf095c66eb84ecc081841373f73bfc72f6e186a006f3da2c2d46362cf76
|
data/MIT-LICENSE
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
Copyright 2013 Genadi Samokovarov and Guillermo Iguaran
|
|
2
|
+
|
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
|
4
|
+
a copy of this software and associated documentation files (the
|
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
|
9
|
+
the following conditions:
|
|
10
|
+
|
|
11
|
+
The above copyright notice and this permission notice shall be
|
|
12
|
+
included in all copies or substantial portions of the Software.
|
|
13
|
+
|
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.markdown
ADDED
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
[](https://travis-ci.org/gsamokovarov/web-console)
|
|
2
|
+
|
|
3
|
+
Web Console
|
|
4
|
+
===========
|
|
5
|
+
|
|
6
|
+
There is no doubt that `rails console` is one of the most useful commands,
|
|
7
|
+
Rails has to offer. However, sometimes you can't easily access it, or maybe
|
|
8
|
+
you want to share it with a coworker without configuring remote desktop
|
|
9
|
+
server.
|
|
10
|
+
|
|
11
|
+
This is where _Web Console_ comes to the rescue. It gives you the same
|
|
12
|
+
`rails console` experience, right in the browser. It's not just a tool that
|
|
13
|
+
let's you evaluate Ruby code, there are a lot of those. It's your IRB session,
|
|
14
|
+
the way you configured it.
|
|
15
|
+
|
|
16
|
+

|
|
17
|
+
|
|
18
|
+
Requirements
|
|
19
|
+
------------
|
|
20
|
+
|
|
21
|
+
To run _Web Console_ you need to be running _Rails 4_ and _MRI Ruby 1.9.3_ and
|
|
22
|
+
above. It may run on _Rubinius_ and _JRuby_, but we haven't tested those yet.
|
|
23
|
+
|
|
24
|
+
Installation
|
|
25
|
+
------------
|
|
26
|
+
|
|
27
|
+
To install it in your current application, add the following to your `Gemfile`.
|
|
28
|
+
|
|
29
|
+
```ruby
|
|
30
|
+
group :development do
|
|
31
|
+
gem 'web-console', '~> 0.1.0'
|
|
32
|
+
end
|
|
33
|
+
```
|
|
34
|
+
|
|
35
|
+
After you save the `Gemfile` changes, make sure to run `bundle install` and
|
|
36
|
+
restart your server for the _Web Console_ to take affect.
|
|
37
|
+
|
|
38
|
+
By default, it should be available in your development environment under
|
|
39
|
+
`/console`. The route is not automatically mounted in a production environment
|
|
40
|
+
and we strongly encourage you to keep it that way.
|
|
41
|
+
|
|
42
|
+
Configuration
|
|
43
|
+
-------------
|
|
44
|
+
|
|
45
|
+
> Today we have learned in the agony of war that great power involves great
|
|
46
|
+
> responsibility.
|
|
47
|
+
>
|
|
48
|
+
> -- <cite>Franklin D. Roosevelt</cite>
|
|
49
|
+
|
|
50
|
+
_Web Console_ is a powerful tool. It allows you to execute arbitrary code on
|
|
51
|
+
the server, so you should be very careful, who you give access to it.
|
|
52
|
+
|
|
53
|
+
### config.web_console.whitelisted_ips
|
|
54
|
+
|
|
55
|
+
By default, only requests coming from `127.0.0.1` are allowed.
|
|
56
|
+
|
|
57
|
+
`config.web_console.whitelisted_ips` lets you control which IP's have access to
|
|
58
|
+
the console.
|
|
59
|
+
|
|
60
|
+
Let's say you want to share your console with just that one roommate, you like
|
|
61
|
+
and his/her IP is `192.168.0.100`.
|
|
62
|
+
|
|
63
|
+
```ruby
|
|
64
|
+
class Application < Rails::Application
|
|
65
|
+
config.web_console.whitelisted_ips = %w( 127.0.0.1 192.168.0.100 )
|
|
66
|
+
end
|
|
67
|
+
```
|
|
68
|
+
|
|
69
|
+
From the example, you can guess that `config.web_console.whitelisted_ips`
|
|
70
|
+
accepts an array of ip addresses, provided as strings. An important thing to
|
|
71
|
+
note here is that, we won't push `127.0.0.1` if you manually set the option!
|
|
72
|
+
|
|
73
|
+
Now let's assume you like all of your roommates. Instead of enumerating their
|
|
74
|
+
IP's, you can whitelist the whole private network. Now every time their IP's
|
|
75
|
+
change, you'll have them covered.
|
|
76
|
+
|
|
77
|
+
```ruby
|
|
78
|
+
class Application < Rails::Application
|
|
79
|
+
config.web_console.whitelisted_ips = '192.168.0.0/16'
|
|
80
|
+
end
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
You can see that `config.web_console.whitelisted_ips` accepts plains strings
|
|
84
|
+
too. More than that, they can cover whole networks.
|
|
85
|
+
|
|
86
|
+
Again, note that this network doesn't allow `127.0.0.1`. If you want to access
|
|
87
|
+
the console, you have to do so from it's external IP or add `127.0.0.1` to the
|
|
88
|
+
mix.
|
|
89
|
+
|
|
90
|
+
### config.web_console.default_mount_path
|
|
91
|
+
|
|
92
|
+
By default, the console will be mounted on `/console`.
|
|
93
|
+
|
|
94
|
+
_(This happens only in the development and test environments!)_.
|
|
95
|
+
|
|
96
|
+
Say you want to mount the console to `/debug`, so you can more easily remember
|
|
97
|
+
where to go, when your application needs debugging.
|
|
98
|
+
|
|
99
|
+
```ruby
|
|
100
|
+
class Application < Rails::Application
|
|
101
|
+
config.web_console.default_mount_path = '/debug'
|
|
102
|
+
end
|
|
103
|
+
```
|
|
104
|
+
|
|
105
|
+
Restart your server and you are done!
|
|
106
|
+
|
|
107
|
+
Test Drive
|
|
108
|
+
----------
|
|
109
|
+
|
|
110
|
+
If you just want to try the web-console, without having to go through the
|
|
111
|
+
trouble of installing it, we provide a [Docker] container that does that for
|
|
112
|
+
you.
|
|
113
|
+
|
|
114
|
+
To try it, install [Docker], clone the project and run the following snippet in
|
|
115
|
+
the git root directory.
|
|
116
|
+
|
|
117
|
+
```bash
|
|
118
|
+
docker build -t gsamokovarov/web-console . && docker run -i -t !#:3
|
|
119
|
+
```
|
|
120
|
+
|
|
121
|
+
[Docker]: http://www.docker.io/
|
data/Rakefile
ADDED
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
begin
|
|
2
|
+
require 'bundler/setup'
|
|
3
|
+
rescue LoadError
|
|
4
|
+
puts 'You must `gem install bundler` and `bundle install` to run rake tasks'
|
|
5
|
+
end
|
|
6
|
+
|
|
7
|
+
require 'socket'
|
|
8
|
+
require 'active_support/core_ext/string/strip'
|
|
9
|
+
require 'rake/testtask'
|
|
10
|
+
|
|
11
|
+
EXPANDED_CWD = File.expand_path(File.dirname(__FILE__))
|
|
12
|
+
|
|
13
|
+
Rake::TestTask.new(:test) do |t|
|
|
14
|
+
t.libs << 'lib'
|
|
15
|
+
t.libs << 'test'
|
|
16
|
+
t.pattern = 'test/**/*_test.rb'
|
|
17
|
+
t.verbose = false
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
# Just ignore this if rake is not runned from the current directory, as is the
|
|
21
|
+
# case with docker's container. BUNDLE_GEMFILE won't do for our case, since the
|
|
22
|
+
# Gemfile references gemspec.
|
|
23
|
+
Bundler::GemHelper.install_tasks if defined? Bundler::GemHelper
|
|
24
|
+
|
|
25
|
+
namespace :docker do
|
|
26
|
+
task :sync do
|
|
27
|
+
Dir.chdir(EXPANDED_CWD) do
|
|
28
|
+
sh "git fetch && git reset --hard origin/master", verbose: false
|
|
29
|
+
end
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
task run: :sync do
|
|
33
|
+
# Generally, the first ipv4 private address is how we would access the
|
|
34
|
+
# docker container from the current machine.
|
|
35
|
+
container_ip = Socket.ip_address_list.find(&:ipv4_private?).ip_address
|
|
36
|
+
|
|
37
|
+
puts <<-NOTICE.strip_heredoc
|
|
38
|
+
Go to http://#{container_ip}:3000/console to preview web-console.
|
|
39
|
+
If it's not showing, please give a few seconds for the server to start.
|
|
40
|
+
NOTICE
|
|
41
|
+
|
|
42
|
+
Dir.chdir("#{EXPANDED_CWD}/test/dummy") do
|
|
43
|
+
sh 'rails server', verbose: false
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
task default: :test
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
$(function() {
|
|
2
|
+
var ERROR_CLASS = 'jquery-console-message-error';
|
|
3
|
+
|
|
4
|
+
var $console = $('#console');
|
|
5
|
+
var instance = $console.console({
|
|
6
|
+
autofocus: true,
|
|
7
|
+
promptLabel: $console.data('initial-prompt'),
|
|
8
|
+
commandHandle: function(line, report) {
|
|
9
|
+
$.ajax({
|
|
10
|
+
url: $console.data('remote-path'),
|
|
11
|
+
type: 'PUT',
|
|
12
|
+
dataType: 'json',
|
|
13
|
+
data: { input: line },
|
|
14
|
+
success: function(response) {
|
|
15
|
+
instance.promptLabel(response.prompt);
|
|
16
|
+
report([{msg: response.output}]);
|
|
17
|
+
},
|
|
18
|
+
error: function(xhr) {
|
|
19
|
+
report([{msg: xhr.responseJSON.error, className: ERROR_CLASS}]);
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
});
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* This is a manifest file that'll be compiled into application.css, which will include all the files
|
|
3
|
+
* listed below.
|
|
4
|
+
*
|
|
5
|
+
* Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
|
|
6
|
+
* or vendor/assets/stylesheets of plugins, if any, can be referenced here using a relative path.
|
|
7
|
+
*
|
|
8
|
+
* You're free to add application-wide styles to this file and they'll appear at the top of the
|
|
9
|
+
* compiled file, but it's generally better to create a new file per style scope.
|
|
10
|
+
*
|
|
11
|
+
*= require_self
|
|
12
|
+
*= require_tree .
|
|
13
|
+
*/
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
body, div, textarea { margin: 0; padding 0 }
|
|
2
|
+
|
|
3
|
+
#console { font-family: Consolas, "Liberation Mono", Courier, monospace; font-size: 13px; position: fixed; width: 98%; height: 98%; left: 1%; top: 1% }
|
|
4
|
+
#console div.jquery-console-inner { height: 100%; background: #333; color: #fff; overflow: auto }
|
|
5
|
+
#console div.jquery-console-focus span.jquery-console-cursor { font-weight: bold; background: #fff }
|
|
6
|
+
#console div.jquery-console-message { white-space: pre-wrap }
|
|
7
|
+
#console div.jquery-console-message-error { font-weight: bold; color: #ff530d }
|
|
8
|
+
#console span.jquery-console-prompt-label { font-weight: bold }
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
module WebConsole
|
|
2
|
+
class ApplicationController < ActionController::Base
|
|
3
|
+
before_action :prevent_unauthorized_requests!
|
|
4
|
+
|
|
5
|
+
private
|
|
6
|
+
def prevent_unauthorized_requests!
|
|
7
|
+
unless request.remote_ip.in?(WebConsole::Engine.config.web_console.whitelisted_ips)
|
|
8
|
+
render nothing: true, status: :unauthorized
|
|
9
|
+
end
|
|
10
|
+
end
|
|
11
|
+
end
|
|
12
|
+
end
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
require_dependency "web_console/application_controller"
|
|
2
|
+
|
|
3
|
+
module WebConsole
|
|
4
|
+
class ConsoleSessionsController < ApplicationController
|
|
5
|
+
rescue_from ConsoleSession::NotFound do |exception|
|
|
6
|
+
render json: exception, status: :gone
|
|
7
|
+
end
|
|
8
|
+
|
|
9
|
+
def index
|
|
10
|
+
@console_session = ConsoleSession.create
|
|
11
|
+
end
|
|
12
|
+
|
|
13
|
+
def update
|
|
14
|
+
@console_session = ConsoleSession.find(params[:id])
|
|
15
|
+
render json: @console_session.save(console_session_params)
|
|
16
|
+
end
|
|
17
|
+
|
|
18
|
+
private
|
|
19
|
+
def console_session_params
|
|
20
|
+
params.permit(:input)
|
|
21
|
+
end
|
|
22
|
+
end
|
|
23
|
+
end
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
module WebConsole
|
|
2
|
+
class ConsoleSession
|
|
3
|
+
include Mutex_m
|
|
4
|
+
|
|
5
|
+
include ActiveModel::Model
|
|
6
|
+
include ActiveModel::Serializers::JSON
|
|
7
|
+
|
|
8
|
+
# In-memory storage for the console sessions. Session preservation is
|
|
9
|
+
# troubled on servers with multiple workers and threads.
|
|
10
|
+
INMEMORY_STORAGE = {}
|
|
11
|
+
|
|
12
|
+
# Store and define the available attributes.
|
|
13
|
+
ATTRIBUTES = [ :id, :input, :output, :prompt ].each do |attr|
|
|
14
|
+
attr_accessor attr
|
|
15
|
+
end
|
|
16
|
+
|
|
17
|
+
# Raised when trying to find a session that is no longer in the in-memory
|
|
18
|
+
# session storage.
|
|
19
|
+
class NotFound < Exception
|
|
20
|
+
def to_json(*)
|
|
21
|
+
{error: message}.to_json
|
|
22
|
+
end
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
class << self
|
|
26
|
+
# Finds a session by its id.
|
|
27
|
+
#
|
|
28
|
+
# Raises WebConsole::ConsoleSession::Expired if there is no such session.
|
|
29
|
+
def find(id)
|
|
30
|
+
INMEMORY_STORAGE[id.to_i] or raise NotFound.new('Session unavailable')
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
# Creates an already persisted consolse session.
|
|
34
|
+
#
|
|
35
|
+
# Use this method if you need to persist a session, without providing it
|
|
36
|
+
# any input.
|
|
37
|
+
def create
|
|
38
|
+
INMEMORY_STORAGE[(model = new).id] = model
|
|
39
|
+
end
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
def initialize(attributes = {})
|
|
43
|
+
@repl = WebConsole::REPL.default.new
|
|
44
|
+
|
|
45
|
+
super
|
|
46
|
+
ensure_consequential_id!
|
|
47
|
+
populate_repl_attributes!(initial: true)
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
# Saves the model into the in-memory storage.
|
|
51
|
+
#
|
|
52
|
+
# Returns false if the model is not valid (e.g. its missing input).
|
|
53
|
+
def save(attributes = {})
|
|
54
|
+
self.attributes = attributes if attributes.present?
|
|
55
|
+
populate_repl_attributes!
|
|
56
|
+
store!
|
|
57
|
+
end
|
|
58
|
+
|
|
59
|
+
# Returns true if the current session is persisted in the in-memory storage.
|
|
60
|
+
def persisted?
|
|
61
|
+
self == INMEMORY_STORAGE[id]
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
# Returns an Enumerable of all key attributes if any is set, regardless if
|
|
65
|
+
# the object is persisted or not.
|
|
66
|
+
def to_key
|
|
67
|
+
super if persisted?
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
protected
|
|
71
|
+
# Returns a hash of the attributes and their values.
|
|
72
|
+
def attributes
|
|
73
|
+
return Hash[ATTRIBUTES.zip([nil])]
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
# Sets model attributes from a hash.
|
|
77
|
+
def attributes=(attributes)
|
|
78
|
+
attributes.each do |attr, value|
|
|
79
|
+
next unless ATTRIBUTES.include?(attr.to_sym)
|
|
80
|
+
public_send(:"#{attr}=", value)
|
|
81
|
+
end
|
|
82
|
+
end
|
|
83
|
+
|
|
84
|
+
private
|
|
85
|
+
def ensure_consequential_id!
|
|
86
|
+
synchronize do
|
|
87
|
+
self.id = begin
|
|
88
|
+
@@counter ||= 0
|
|
89
|
+
@@counter += 1
|
|
90
|
+
end
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
def populate_repl_attributes!(options = {})
|
|
95
|
+
synchronize do
|
|
96
|
+
# Don't send any input on the initial population so we don't bump up
|
|
97
|
+
# the numbers in the dynamic prompts.
|
|
98
|
+
self.output = @repl.send_input(input) unless options[:initial]
|
|
99
|
+
self.prompt = @repl.prompt
|
|
100
|
+
end
|
|
101
|
+
end
|
|
102
|
+
|
|
103
|
+
def store!
|
|
104
|
+
synchronize { INMEMORY_STORAGE[id] = self }
|
|
105
|
+
end
|
|
106
|
+
end
|
|
107
|
+
end
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html>
|
|
3
|
+
<head>
|
|
4
|
+
<title>WebConsole</title>
|
|
5
|
+
<%= stylesheet_link_tag "web_console/application", media: "all" %>
|
|
6
|
+
<%= javascript_include_tag "web_console/application" %>
|
|
7
|
+
<%= csrf_meta_tags %>
|
|
8
|
+
</head>
|
|
9
|
+
<body>
|
|
10
|
+
|
|
11
|
+
<%= yield %>
|
|
12
|
+
|
|
13
|
+
</body>
|
|
14
|
+
</html>
|
data/config/routes.rb
ADDED
data/lib/web-console.rb
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
require 'web_console'
|