This repository has been archived by the owner on Jun 10, 2018. It is now read-only.
forked from rails/execjs
-
Notifications
You must be signed in to change notification settings - Fork 46
/
external_runtime.rb
158 lines (136 loc) · 4.02 KB
/
external_runtime.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
require "tempfile"
module ExecJS
class ExternalRuntime
class Context
def initialize(runtime, source = "")
source = source.encode('UTF-8') if source.respond_to?(:encode)
@runtime = runtime
@source = source
end
def eval(source, options = {})
source = source.encode('UTF-8') if source.respond_to?(:encode)
if /\S/ =~ source
exec("return eval(#{MultiJson.encode("(#{source})")})")
end
end
def exec(source, options = {})
source = source.encode('UTF-8') if source.respond_to?(:encode)
compile_to_tempfile([@source, source].join("\n")) do |file|
extract_result(@runtime.send(:exec_runtime, file.path))
end
end
def call(identifier, *args)
eval "#{identifier}.apply(this, #{MultiJson.encode(args)})"
end
protected
def compile_to_tempfile(source)
tempfile = Tempfile.open(['execjs', '.js'])
tempfile.write compile(source)
tempfile.close
yield tempfile
ensure
tempfile.close!
end
def compile(source)
@runtime.send(:runner_source).dup.tap do |output|
output.sub!('#{source}') do
source
end
output.sub!('#{encoded_source}') do
encoded_source = encode_unicode_codepoints(source)
MultiJson.encode("(function(){ #{encoded_source} })()")
end
output.sub!('#{json2_source}') do
IO.read(ExecJS.root + "/support/json2.js")
end
end
end
def extract_result(output)
status, value = output.empty? ? [] : MultiJson.decode(output)
if status == "ok"
value
elsif value == "SyntaxError: Parse error"
raise RuntimeError, value
else
raise ProgramError, value
end
end
if "".respond_to?(:codepoints)
def encode_unicode_codepoints(str)
str.gsub(/[\u0080-\uffff]/) do |ch|
"\\u%04x" % ch.codepoints.to_a
end
end
else
def encode_unicode_codepoints(str)
str.gsub(/([\xC0-\xDF][\x80-\xBF]|
[\xE0-\xEF][\x80-\xBF]{2}|
[\xF0-\xF7][\x80-\xBF]{3})+/nx) do |ch|
"\\u%04x" % ch.unpack("U*")
end
end
end
end
attr_reader :name
def initialize(options)
@name = options[:name]
@command = options[:command]
@runner_path = options[:runner_path]
@test_args = options[:test_args]
@test_match = options[:test_match]
@binary = locate_binary
end
def exec(source)
context = Context.new(self)
context.exec(source)
end
def eval(source)
context = Context.new(self)
context.eval(source)
end
def compile(source)
Context.new(self, source)
end
def available?
require "multi_json"
@binary ? true : false
end
protected
def runner_source
@runner_source ||= IO.read(@runner_path)
end
def exec_runtime(filename)
output = nil
IO.popen("#{@binary} #{filename} 2>&1") { |f| output = f.read }
if $?.success?
output
else
raise RuntimeError, output
end
end
def locate_binary
if binary = which(@command)
if @test_args
output = `#{binary} #{@test_args} 2>&1`
binary if output.match(@test_match)
else
binary
end
end
end
def which(command)
Array(command).each do |name|
name, args = name.split(/\s+/, 2)
result = if ExecJS.windows?
`#{ExecJS.root}/support/which.bat #{name}`
else
`command -v #{name} 2>/dev/null`
end
if path = result.strip.split("\n").first
return args ? "#{path} #{args}" : path
end
end
nil
end
end
end