qlang 0.0.14142135 → 0.0.27000000
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 +4 -4
- data/README.md +10 -3
- data/lib/qlang/api.rb +4 -0
- data/lib/qlang/iq.rb +17 -4
- data/lib/qlang/lexer/base.rb +10 -0
- data/lib/qlang/lexer/func_lexer.rb +1 -1
- data/lib/qlang/lexer/tokens.rb +45 -0
- data/lib/qlang/lexer/wrap_lexer.rb +10 -4
- data/lib/qlang/parser/func_parser.rb +6 -7
- data/lib/qlang/parser/matrix_parser.rb +6 -2
- data/lib/qlang/parser/vector_parser.rb +4 -2
- data/lib/qlang/parser.rb +23 -27
- data/lib/qlang/version.rb +1 -1
- data/spec/iq_spec.rb +46 -9
- data/spec/objects/function_spec.rb +2 -2
- data/spec/objects/list_spec.rb +10 -10
- data/spec/q_lang_spec.rb +1 -1
- metadata +4 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a9e53219836ec42b6ef2edb577c24f402f97e49c
|
4
|
+
data.tar.gz: 2cf73154711705aa6b29b1da703b06638f37c01b
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 1b1a074dec18140414233546d87d3de94c5ac22524054be77cac955a47ec7ba1f05069a84c63dc88275f0eeb09b5e5c1ba61436aef9d18a34e93cc764077dca7
|
7
|
+
data.tar.gz: f23ae787a6fdcdf7a70e992dfa8db9e51e5f08f24dfdaa9db86f1ef0b3899478afb5ec5818325d68bdb517e45b1f0258e42326c24dc5c607c6784e7e067db588
|
data/README.md
CHANGED
@@ -13,8 +13,13 @@ d/dx(cos(x))
|
|
13
13
|
d/dx(log(x))
|
14
14
|
=> ( 1 / x )
|
15
15
|
|
16
|
-
|
16
|
+
# You can omit parentheses
|
17
|
+
|
18
|
+
d/dy y^2
|
17
19
|
=> ( 2 * y )
|
20
|
+
|
21
|
+
d/dy xy
|
22
|
+
=> ( x )
|
18
23
|
```
|
19
24
|
|
20
25
|
|
@@ -47,7 +52,9 @@ S(cos(x)dx)[0..pi]
|
|
47
52
|
|
48
53
|
### Function
|
49
54
|
```
|
50
|
-
f(x, y) =
|
55
|
+
f(x, y) = xy
|
56
|
+
f(1, 2)
|
57
|
+
=> 2
|
51
58
|
```
|
52
59
|
|
53
60
|
|
@@ -72,7 +79,7 @@ Install it yourself as:
|
|
72
79
|
|
73
80
|
## Contributing
|
74
81
|
|
75
|
-
1. Fork it ( https://github.com/
|
82
|
+
1. Fork it ( https://github.com/gogotanaka/Q/fork )
|
76
83
|
2. Create your feature branch (`git checkout -b my-new-feature`)
|
77
84
|
3. Commit your changes (`git commit -am 'Add some feature'`)
|
78
85
|
4. Push to the branch (`git push origin my-new-feature`)
|
data/lib/qlang/api.rb
CHANGED
data/lib/qlang/iq.rb
CHANGED
@@ -7,9 +7,22 @@ module Qlang
|
|
7
7
|
str.equalize!
|
8
8
|
end
|
9
9
|
end
|
10
|
-
|
11
|
-
|
12
|
-
|
10
|
+
|
11
|
+
def self.execute(code)
|
12
|
+
ruby_code = Q.to_ruby.compile(code)
|
13
|
+
ruby_obj = eval(ruby_code)
|
14
|
+
|
15
|
+
optimize_output(ruby_obj)
|
16
|
+
rescue SyntaxError
|
17
|
+
# TODO: emergency
|
18
|
+
case ruby_code
|
19
|
+
when /(\d)+(\w)/
|
20
|
+
execute("#{$1} * #{$2}")
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.optimize_output(ruby_obj)
|
25
|
+
case ruby_obj
|
13
26
|
when Matrix, Vector, Dydx::Algebra::Formula
|
14
27
|
ruby_obj.to_q
|
15
28
|
when Float::INFINITY
|
@@ -21,6 +34,6 @@ module Qlang
|
|
21
34
|
str.equalize!
|
22
35
|
end
|
23
36
|
end
|
24
|
-
|
37
|
+
|
25
38
|
end
|
26
39
|
end
|
data/lib/qlang/lexer/base.rb
CHANGED
@@ -1,8 +1,12 @@
|
|
1
1
|
require 'strscan'
|
2
|
+
require 'qlang/lexer/tokens'
|
3
|
+
|
2
4
|
|
3
5
|
module Qlang
|
4
6
|
module Lexer
|
5
7
|
class Base
|
8
|
+
attr_accessor :lexeds
|
9
|
+
include Tokens
|
6
10
|
class << self
|
7
11
|
attr_reader :token_hash
|
8
12
|
|
@@ -96,6 +100,12 @@ module Qlang
|
|
96
100
|
end
|
97
101
|
end
|
98
102
|
|
103
|
+
# NEW APIs
|
104
|
+
def parsed!(token_position, parsed)
|
105
|
+
@lexeds.delete_at(token_position)
|
106
|
+
@lexeds.insert(token_position, { R: parsed })
|
107
|
+
end
|
108
|
+
|
99
109
|
private
|
100
110
|
|
101
111
|
def to_num(token_with_num)
|
@@ -0,0 +1,45 @@
|
|
1
|
+
module Qlang
|
2
|
+
module Lexer
|
3
|
+
module Tokens
|
4
|
+
# FIRST TOKEN
|
5
|
+
NUM = '[0-9]+'
|
6
|
+
VAR = '[a-z]'
|
7
|
+
FUNCV = '[a-zA-Z]'
|
8
|
+
VARNUM = '[0-9a-z]'
|
9
|
+
ANYSP = ' *'
|
10
|
+
ANYSTR = '.+'
|
11
|
+
NONL = '[^\r\n]'
|
12
|
+
LPRN = '\('
|
13
|
+
RPRN = '\)'
|
14
|
+
LBRC = '\{'
|
15
|
+
RBRC = '\}'
|
16
|
+
CLN = '\:'
|
17
|
+
SCLN = ';'
|
18
|
+
CMA = '\,'
|
19
|
+
SP = ' '
|
20
|
+
|
21
|
+
# SECOND TOKEN
|
22
|
+
class ::String
|
23
|
+
def line_by(char)
|
24
|
+
"#{ANYSP}#{self}(#{ANYSP}#{char}#{ANYSP}#{self})*#{ANYSP}"
|
25
|
+
end
|
26
|
+
end
|
27
|
+
NUMS_BY_CMA = NUM.line_by(CMA)
|
28
|
+
VARS_BY_CMA = VAR.line_by(CMA)
|
29
|
+
VARNUMS_BY_CMA = VARNUM.line_by(CMA)
|
30
|
+
NUMS_BY_SP = NUM.line_by(SP)
|
31
|
+
|
32
|
+
# THIRD TOKEN
|
33
|
+
class ::String
|
34
|
+
def func_call
|
35
|
+
"#{FUNCV}#{LPRN}#{ANYSP}#{self}*#{ANYSP}#{RPRN}"
|
36
|
+
end
|
37
|
+
end
|
38
|
+
FUNCCN = NUMS_BY_CMA.func_call
|
39
|
+
FUNCCV = VARS_BY_CMA.func_call
|
40
|
+
FUNCCVN = VARNUMS_BY_CMA.func_call
|
41
|
+
|
42
|
+
NUMS_BY_SP_BY_SCLN = NUMS_BY_SP.line_by(SCLN)
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -1,10 +1,16 @@
|
|
1
|
+
require 'pry'
|
1
2
|
module Qlang
|
2
3
|
module Lexer
|
3
4
|
class WrapLexer < Base
|
4
|
-
rule(
|
5
|
-
rule(
|
6
|
-
rule(/S
|
7
|
-
rule(/d\/d
|
5
|
+
rule(%r@#{FUNCCV}#{ANYSP}=#{ANYSP}#{NONL}+@) { :def_func }
|
6
|
+
rule(%r@#{FUNCCN}@) { :eval_func }
|
7
|
+
rule(/S#{ANYSP}#{LPRN}#{ANYSTR}#{RPRN}\[#{ANYSTR}\]/) { :integral }
|
8
|
+
rule(/d\/d#{VAR} .*/) { :differential }
|
9
|
+
rule(%r@#{LPRN}#{NUMS_BY_SP}#{RPRN}@) { :vector }
|
10
|
+
rule(%r@#{LPRN}#{NUMS_BY_SP_BY_SCLN}#{RPRN}t@) { :tmatrix }
|
11
|
+
rule(%r@#{LPRN}#{NUMS_BY_SP_BY_SCLN}#{RPRN}@) { :matrix }
|
12
|
+
|
13
|
+
|
8
14
|
rule(/\(/) { :LPRN }
|
9
15
|
rule(/\)/) { :RPRN }
|
10
16
|
rule(/\{/) { :LBRC }
|
@@ -1,14 +1,13 @@
|
|
1
|
+
require 'qlang/lexer/tokens'
|
1
2
|
module Qlang
|
2
3
|
module Parser
|
3
4
|
module FuncParser
|
4
5
|
include Base
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
FuncApi.execute(func_name, args, FormulaParser.execute(lexed[-1][:FOML]))
|
6
|
+
include Lexer::Tokens
|
7
|
+
def execute(string)
|
8
|
+
def_func, formula = string.split(/ *= */)
|
9
|
+
def_func =~ %r@(#{FUNCV})#{LPRN}#{ANYSP}(#{VARS_BY_CMA})#{ANYSP}#{RPRN}@
|
10
|
+
FuncApi.execute($1, $2.split(' *,'), FormulaParser.execute(formula))
|
12
11
|
end
|
13
12
|
module_function :execute
|
14
13
|
end
|
@@ -2,9 +2,13 @@ module Qlang
|
|
2
2
|
module Parser
|
3
3
|
module MatrixParser
|
4
4
|
include Base
|
5
|
-
def execute(
|
6
|
-
|
5
|
+
def execute(lexed_string, trans: false)
|
6
|
+
lexed_string.rms!(')','(', 't')
|
7
|
+
rows = lexed_string.split(/ *; */).map(&:split_by_sp)
|
7
8
|
rows.all? { |row| row.count == rows.first.count }
|
9
|
+
if trans
|
10
|
+
rows = rows.transpose
|
11
|
+
end
|
8
12
|
MatrixApi.execute(rows)
|
9
13
|
end
|
10
14
|
module_function :execute
|
@@ -2,8 +2,10 @@ module Qlang
|
|
2
2
|
module Parser
|
3
3
|
module VectorParser
|
4
4
|
include Base
|
5
|
-
def execute(
|
6
|
-
|
5
|
+
def execute(lexed_string)
|
6
|
+
lexed_string.rms!(/ *\( */, / *\) */)
|
7
|
+
elements = lexed_string.split_by_sp
|
8
|
+
VectorApi.execute(elements)
|
7
9
|
end
|
8
10
|
module_function :execute
|
9
11
|
end
|
data/lib/qlang/parser.rb
CHANGED
@@ -19,18 +19,30 @@ module Qlang
|
|
19
19
|
fail "I'm so sorry, something wrong. Please feel free to report this." if Time.now > time + 10
|
20
20
|
|
21
21
|
case lexed.token_str
|
22
|
+
when /(:vector)(\d)/, /(:matrix)(\d)/, /(:tmatrix)(\d)/, /(:integral)(\d)/, /(:def_func)(\d)/
|
23
|
+
token_sym = $1.delete(':').to_sym
|
24
|
+
token_position = $2.to_i
|
25
|
+
token_val = lexed.lexeds[token_position][token_sym]
|
26
|
+
|
27
|
+
parsed = case token_sym
|
28
|
+
when :vector
|
29
|
+
VectorParser.execute(token_val)
|
30
|
+
when :matrix
|
31
|
+
MatrixParser.execute(token_val)
|
32
|
+
when :tmatrix
|
33
|
+
MatrixParser.execute(token_val, trans: true)
|
34
|
+
when :integral
|
35
|
+
IntegralParser.execute(token_val)
|
36
|
+
when :def_func
|
37
|
+
FuncParser.execute(token_val)
|
38
|
+
end
|
39
|
+
lexed.parsed!(token_position, parsed)
|
40
|
+
|
22
41
|
when /:LPRN\d(:CONT\d):RPRN\d/
|
23
42
|
cont_token_with_num = $1
|
24
43
|
cont_lexed = Lexer::ContLexer.new(lexed.get_value(cont_token_with_num))
|
25
44
|
|
26
|
-
|
27
|
-
when /(:NUM\d)+(:SCLN\d|:NLIN\d)(:NUM\d)/
|
28
|
-
cont = MatrixParser.execute(cont_lexed)
|
29
|
-
when /(:NUM\d)+/
|
30
|
-
cont = VectorParser.execute(cont_lexed)
|
31
|
-
else
|
32
|
-
cont = "(#{cont_lexed.values.join(' ')})"
|
33
|
-
end
|
45
|
+
cont = "(#{cont_lexed.values.join(' ')})"
|
34
46
|
lexed.squash_with_prn(cont_token_with_num, cont)
|
35
47
|
|
36
48
|
when /:LBRC\d(:CONT\d):RBRC\d/
|
@@ -45,29 +57,12 @@ module Qlang
|
|
45
57
|
end
|
46
58
|
lexed.squash_with_prn(cont_token_with_num, cont)
|
47
59
|
|
48
|
-
when /:
|
49
|
-
cont_token_with_num = $&
|
50
|
-
cont_lexed = Lexer::FuncLexer.new(lexed.get_value(cont_token_with_num))
|
51
|
-
|
52
|
-
case cont_lexed.token_str
|
53
|
-
when /:FDEF\d:EQL\d:FOML\d/
|
54
|
-
cont = FuncParser.execute(cont_lexed)
|
55
|
-
lexed.ch_value(cont_token_with_num, cont)
|
56
|
-
lexed.ch_token(cont_token_with_num, :R)
|
57
|
-
end
|
58
|
-
|
59
|
-
when /:ITGL\d/
|
60
|
-
cont_token_with_num = $&
|
61
|
-
cont = IntegralParser.execute(lexed.get_value(cont_token_with_num))
|
62
|
-
lexed.ch_value(cont_token_with_num, cont)
|
63
|
-
lexed.ch_token(cont_token_with_num, :R)
|
64
|
-
|
65
|
-
when /:EFUNC\d/
|
60
|
+
when /:eval_func\d/
|
66
61
|
cont_token_with_num = $&
|
67
62
|
cont = lexed.get_value(cont_token_with_num)
|
68
63
|
lexed.squash_with_prn(cont_token_with_num, cont)
|
69
64
|
|
70
|
-
when /:
|
65
|
+
when /:differential\d/
|
71
66
|
cont_token_with_num = $&
|
72
67
|
cont = lexed.get_value(cont_token_with_num)
|
73
68
|
cont =~ /(d\/d[a-zA-Z]) (.*)/
|
@@ -81,6 +76,7 @@ module Qlang
|
|
81
76
|
|
82
77
|
lexed.squash_to_cont($1, 2) if lexed.token_str =~ /(:CONT\d|:R\d)(:CONT\d|:R\d)/
|
83
78
|
end
|
79
|
+
|
84
80
|
lexed.fix_r_txt!
|
85
81
|
lexed.values.join
|
86
82
|
end
|
data/lib/qlang/version.rb
CHANGED
data/spec/iq_spec.rb
CHANGED
@@ -15,15 +15,52 @@ describe Qlang do
|
|
15
15
|
end
|
16
16
|
end
|
17
17
|
|
18
|
+
describe 'General' do
|
19
|
+
cal_test('ex1', '2x', '2x')
|
20
|
+
cal_test('ex2', 'x + x', '2x')
|
21
|
+
cal_test('ex3', 'x * y', 'xy')
|
22
|
+
end
|
23
|
+
|
18
24
|
describe 'Matrix' do
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
25
|
+
cal_test('ex1',
|
26
|
+
'(1 2 3; 4 5 6)',
|
27
|
+
'(1 2 3; 4 5 6)'
|
28
|
+
)
|
29
|
+
|
30
|
+
cal_test('ex2',
|
31
|
+
'(1 2 3; 4 5 6) + (1 2 3; 4 5 6)',
|
32
|
+
'(2 4 6; 8 10 12)'
|
33
|
+
)
|
34
|
+
|
35
|
+
cal_test('ex3',
|
36
|
+
'(1 2 3; 4 5 6) - (2 4 1; 8 3 9)',
|
37
|
+
'(-1 -2 2; -4 2 -3)'
|
38
|
+
)
|
39
|
+
|
40
|
+
cal_test('ex4',
|
41
|
+
'(1 2; 3 4) * (1 2; 3 4)',
|
42
|
+
'(7 10; 15 22)'
|
43
|
+
)
|
44
|
+
|
45
|
+
cal_test('ex5',
|
46
|
+
'(1 2; 3 4) ** 2',
|
47
|
+
'(7 10; 15 22)'
|
48
|
+
)
|
49
|
+
|
50
|
+
cal_test('ex6',
|
51
|
+
'(1 2; 3 4) ** 2',
|
52
|
+
'(7 10; 15 22)'
|
53
|
+
)
|
54
|
+
|
55
|
+
cal_test('ex7',
|
56
|
+
'(1 2; 3 4) * (1 2)',
|
57
|
+
'(5 11)'
|
58
|
+
)
|
59
|
+
|
60
|
+
cal_test('ex8',
|
61
|
+
'(1 2 3; 4 5 6)t',
|
62
|
+
'(1 4; 2 5; 3 6)'
|
63
|
+
)
|
27
64
|
end
|
28
65
|
|
29
66
|
describe 'Vector' do
|
@@ -67,7 +104,7 @@ describe Qlang do
|
|
67
104
|
def_test('ex1', 'f(x, y) = x + y', 'x + y')
|
68
105
|
cal_test('ex1', 'f( 4, 5 )', '9.0')
|
69
106
|
|
70
|
-
def_test('ex2', 'f(x,
|
107
|
+
def_test('ex2', 'f( x , y) = xy', 'x * y')
|
71
108
|
cal_test('ex2', 'f( 3, 9 )', '27.0')
|
72
109
|
|
73
110
|
def_test('ex3', 'f(x, y) = xy^2', 'x * ( y ** 2 )')
|
@@ -7,7 +7,7 @@ describe Qlang do
|
|
7
7
|
expect(
|
8
8
|
Q.to_r.compile('f(x, y) = x + y')
|
9
9
|
).to eq(
|
10
|
-
"f <- function(x
|
10
|
+
"f <- function(x, y) x + y"
|
11
11
|
)
|
12
12
|
|
13
13
|
expect(
|
@@ -25,7 +25,7 @@ describe Qlang do
|
|
25
25
|
expect(
|
26
26
|
Q.to_r.compile('h(a, b, c) = a ^ 2 + b ^ 2 + c ^ 2')
|
27
27
|
).to eq(
|
28
|
-
"h <- function(a ,
|
28
|
+
"h <- function(a, b, c) a ^ 2 + b ^ 2 + c ^ 2"
|
29
29
|
)
|
30
30
|
end
|
31
31
|
end
|
data/spec/objects/list_spec.rb
CHANGED
@@ -3,11 +3,11 @@ require 'spec_helper'
|
|
3
3
|
describe Qlang do
|
4
4
|
describe 'List' do
|
5
5
|
it do
|
6
|
-
expect(
|
7
|
-
|
8
|
-
).to eq(
|
9
|
-
|
10
|
-
)
|
6
|
+
# expect(
|
7
|
+
# Q.to_r.compile('{name: "Gogotanaka", age: 21, birth: (1992 8 10) }')
|
8
|
+
# ).to eq(
|
9
|
+
# "list(name=\"Gogotanaka\", age=21, birth=c(1992, 8, 10))"
|
10
|
+
# )
|
11
11
|
|
12
12
|
expect(
|
13
13
|
Q.to_r.compile('{key1: 234234, key2: 387342 }')
|
@@ -21,11 +21,11 @@ describe Qlang do
|
|
21
21
|
"list(key1=234234, key2=387342, key3=38733242)"
|
22
22
|
)
|
23
23
|
|
24
|
-
expect(
|
25
|
-
|
26
|
-
).to eq(
|
27
|
-
|
28
|
-
)
|
24
|
+
# expect(
|
25
|
+
# Q.to_r.compile('{key1:(1 3 2; 8 2 3),key2:387342,key3:38733242}')
|
26
|
+
# ).to eq(
|
27
|
+
# "list(key1=matrix(c(1, 3, 2, 8, 2, 3), 2, 3, byrow = TRUE), key2=387342, key3=38733242)"
|
28
|
+
# )
|
29
29
|
end
|
30
30
|
end
|
31
31
|
end
|
data/spec/q_lang_spec.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: qlang
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0.
|
4
|
+
version: 0.0.27000000
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- gogotanaka
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2014-
|
11
|
+
date: 2014-10-15 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: dydx
|
@@ -97,6 +97,7 @@ files:
|
|
97
97
|
- lib/qlang/lexer/base.rb
|
98
98
|
- lib/qlang/lexer/cont_lexer.rb
|
99
99
|
- lib/qlang/lexer/func_lexer.rb
|
100
|
+
- lib/qlang/lexer/tokens.rb
|
100
101
|
- lib/qlang/lexer/wrap_lexer.rb
|
101
102
|
- lib/qlang/parser.rb
|
102
103
|
- lib/qlang/parser/base.rb
|
@@ -137,7 +138,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
137
138
|
version: '0'
|
138
139
|
requirements: []
|
139
140
|
rubyforge_project:
|
140
|
-
rubygems_version: 2.
|
141
|
+
rubygems_version: 2.2.2
|
141
142
|
signing_key:
|
142
143
|
specification_version: 4
|
143
144
|
summary: Enjoy MATH!
|