tildeath 0.0.1
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/LICENSE +674 -0
- data/README.md +4 -0
- data/bin/tildeath +28 -0
- data/bin/tildeath~ +28 -0
- data/bin/~ath~ +434 -0
- data/lib/tildeath.rb +2 -0
- data/lib/tildeath/ast_nodes.rb +10 -0
- data/lib/tildeath/ast_nodes/bang.rb +17 -0
- data/lib/tildeath/ast_nodes/bifurcate.rb +25 -0
- data/lib/tildeath/ast_nodes/dot_die.rb +18 -0
- data/lib/tildeath/ast_nodes/import.rb +22 -0
- data/lib/tildeath/ast_nodes/null.rb +11 -0
- data/lib/tildeath/ast_nodes/program.rb +22 -0
- data/lib/tildeath/ast_nodes/split.rb +25 -0
- data/lib/tildeath/ast_nodes/statements.rb +23 -0
- data/lib/tildeath/ast_nodes/tildeath.rb +29 -0
- data/lib/tildeath/ast_nodes/value.rb +17 -0
- data/lib/tildeath/imminently_deceased_object.rb +19 -0
- data/lib/tildeath/interpreter.rb +22 -0
- data/lib/tildeath/lexer.rb +70 -0
- data/lib/tildeath/parser.rb +136 -0
- data/lib/tildeath/tildeath_error.rb +11 -0
- data/lib/tildeath/token.rb +12 -0
- data/lib/tildeath/version.rb +3 -0
- metadata +69 -0
data/lib/tildeath.rb
ADDED
@@ -0,0 +1,10 @@
|
|
1
|
+
require_relative 'ast_nodes/program'
|
2
|
+
require_relative 'ast_nodes/statements'
|
3
|
+
require_relative 'ast_nodes/bifurcate'
|
4
|
+
require_relative 'ast_nodes/split'
|
5
|
+
require_relative 'ast_nodes/import'
|
6
|
+
require_relative 'ast_nodes/tildeath'
|
7
|
+
require_relative 'ast_nodes/null'
|
8
|
+
require_relative 'ast_nodes/dot_die'
|
9
|
+
require_relative 'ast_nodes/bang'
|
10
|
+
require_relative 'ast_nodes/value'
|
@@ -0,0 +1,25 @@
|
|
1
|
+
require_relative '../imminently_deceased_object'
|
2
|
+
|
3
|
+
module Tildeath
|
4
|
+
module ASTNodes
|
5
|
+
class Bifurcate
|
6
|
+
def initialize(orig, parts)
|
7
|
+
@orig = orig
|
8
|
+
@parts = parts
|
9
|
+
end
|
10
|
+
|
11
|
+
def execute(context)
|
12
|
+
type = context[@orig].type
|
13
|
+
@parts.each do |part|
|
14
|
+
context[part] = ImminentlyDeceasedObject.new(type, part)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
def to_s
|
19
|
+
@parts[1..-1].reduce("bifurcate #{@orig}[#{@parts[0]}") {|memo, part|
|
20
|
+
memo << ', ' << part.to_s
|
21
|
+
} << ']'
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module Tildeath
|
2
|
+
module ASTNodes
|
3
|
+
class DotDie
|
4
|
+
def initialize(victim)
|
5
|
+
@victim = victim
|
6
|
+
end
|
7
|
+
|
8
|
+
def execute(context)
|
9
|
+
return unless context[:THIS].alive?
|
10
|
+
context[@victim].die
|
11
|
+
end
|
12
|
+
|
13
|
+
def to_s
|
14
|
+
"#{@victim}.DIE()"
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
require_relative '../imminently_deceased_object'
|
2
|
+
|
3
|
+
module Tildeath
|
4
|
+
module ASTNodes
|
5
|
+
class Import
|
6
|
+
def initialize(type, name)
|
7
|
+
@type = type
|
8
|
+
@name = name
|
9
|
+
end
|
10
|
+
|
11
|
+
def execute(context)
|
12
|
+
return unless context[:THIS].alive?
|
13
|
+
# Create new object of the specified type and name and store it in context
|
14
|
+
context[@name] = ImminentlyDeceasedObject.new(@type, @name)
|
15
|
+
end
|
16
|
+
|
17
|
+
def to_s
|
18
|
+
"import #{@type} #{@name}"
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
require_relative '../imminently_deceased_object'
|
2
|
+
|
3
|
+
module Tildeath
|
4
|
+
module ASTNodes
|
5
|
+
class Program
|
6
|
+
def initialize(statements)
|
7
|
+
@statements = statements
|
8
|
+
end
|
9
|
+
|
10
|
+
def execute
|
11
|
+
context = {
|
12
|
+
THIS: ImminentlyDeceasedObject.new(:program, :THIS)
|
13
|
+
}
|
14
|
+
@statements.execute(context)
|
15
|
+
end
|
16
|
+
|
17
|
+
def to_s
|
18
|
+
@statements.to_s
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
require_relative '../imminently_deceased_object'
|
2
|
+
|
3
|
+
module Tildeath
|
4
|
+
module ASTNodes
|
5
|
+
class Split
|
6
|
+
def initialize(orig, parts)
|
7
|
+
@orig = orig
|
8
|
+
@parts = parts
|
9
|
+
end
|
10
|
+
|
11
|
+
def execute(context)
|
12
|
+
type = context[@orig].type
|
13
|
+
@parts.each do |part|
|
14
|
+
context[part] = ImminentlyDeceasedObject.new(type, part)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
def to_s
|
19
|
+
@parts[1..-1].reduce("split #{@orig}[#{@parts[0]}") {|memo, part|
|
20
|
+
memo << ', ' << part.to_s
|
21
|
+
} << ']'
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module Tildeath
|
2
|
+
module ASTNodes
|
3
|
+
class Statements
|
4
|
+
def initialize(statements=[])
|
5
|
+
@statements = statements
|
6
|
+
end
|
7
|
+
|
8
|
+
def execute(context)
|
9
|
+
return unless context[:THIS].alive?
|
10
|
+
@statements.each do |statement|
|
11
|
+
return unless context[:THIS].alive?
|
12
|
+
statement.execute(context)
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
def to_s
|
17
|
+
@statements.reduce('') do |memo, stmt|
|
18
|
+
memo << stmt.to_s << ";\n"
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
module Tildeath
|
2
|
+
module ASTNodes
|
3
|
+
class Tildeath
|
4
|
+
def initialize(victim, tildeath_body, execute_body)
|
5
|
+
@victim = victim
|
6
|
+
@tildeath_body = tildeath_body
|
7
|
+
@execute_body = execute_body
|
8
|
+
end
|
9
|
+
|
10
|
+
def execute(context)
|
11
|
+
fail "error: no such object: #{@victim}" unless context[@victim]
|
12
|
+
# loop over first set of statements while victim is alive
|
13
|
+
while context[@victim].alive?
|
14
|
+
return unless context[:THIS].alive?
|
15
|
+
@tildeath_body.execute(context)
|
16
|
+
end
|
17
|
+
# run second set of statements when victim dies
|
18
|
+
return unless context[:THIS].alive?
|
19
|
+
@execute_body.execute(context)
|
20
|
+
end
|
21
|
+
|
22
|
+
def to_s
|
23
|
+
"~ATH(#{@victim}) {
|
24
|
+
#{@tildeath_body}
|
25
|
+
} EXECUTE(#{@execute_body})"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
module Tildeath
|
2
|
+
class ImminentlyDeceasedObject
|
3
|
+
attr_reader :type, :name
|
4
|
+
|
5
|
+
def initialize(type, name)
|
6
|
+
@type = type
|
7
|
+
@name = name
|
8
|
+
@alive = true
|
9
|
+
end
|
10
|
+
|
11
|
+
def die
|
12
|
+
@alive = false
|
13
|
+
end
|
14
|
+
|
15
|
+
def alive?
|
16
|
+
@alive
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
require_relative 'tildeath_error'
|
2
|
+
require_relative 'lexer'
|
3
|
+
require_relative 'parser'
|
4
|
+
|
5
|
+
module Tildeath
|
6
|
+
module Interpreter
|
7
|
+
def self.interpret(script, filename:, verbose: false)
|
8
|
+
# discard shebang line if present
|
9
|
+
script.slice!(0, script.index("\n") + 1) if script[0..1] == "#!"
|
10
|
+
# scan string into tokens
|
11
|
+
tokens = Lexer.lex(script)
|
12
|
+
# parse tokens into abstract syntax tree
|
13
|
+
tree = Parser.parse(tokens)
|
14
|
+
# show gussed original source based on AST
|
15
|
+
puts tree if verbose
|
16
|
+
# execute AST starting at its root
|
17
|
+
tree.execute
|
18
|
+
rescue TildeathError => ex
|
19
|
+
puts "#{filename}:#{ex.line_number}:#{ex.column}: #{ex.message}"
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,70 @@
|
|
1
|
+
require_relative 'tildeath_error'
|
2
|
+
require_relative 'token'
|
3
|
+
|
4
|
+
module Tildeath
|
5
|
+
module Lexer
|
6
|
+
# token types
|
7
|
+
TOKENS = {
|
8
|
+
COMMENT: /\/\/.*$/,
|
9
|
+
IMPORT: /import/,
|
10
|
+
TILDEATH: /~ATH/,
|
11
|
+
EXECUTE: /EXECUTE/,
|
12
|
+
THIS: /THIS/,
|
13
|
+
BIFURC8: /bifurcate/,
|
14
|
+
SPLIT: /split/,
|
15
|
+
LBRACE: /\{/,
|
16
|
+
RBRACE: /\}/,
|
17
|
+
LPAREN: /\(/,
|
18
|
+
RPAREN: /\)/,
|
19
|
+
LBRACKET: /\[/,
|
20
|
+
RBRACKET: /\]/,
|
21
|
+
DOT: /\./,
|
22
|
+
DIE: /DIE/,
|
23
|
+
NULL: /NULL/,
|
24
|
+
COMMA: /,/,
|
25
|
+
SEMI: /;/,
|
26
|
+
BANG: /!/,
|
27
|
+
# one or more whitespace characters
|
28
|
+
WS: /\s+/,
|
29
|
+
# a letter or underscore followed by zero or more letters, underscores, and digits
|
30
|
+
IDENT: /[a-z_][a-z_0-9]*/i
|
31
|
+
}
|
32
|
+
|
33
|
+
def self.lex(input)
|
34
|
+
# tokens found so far
|
35
|
+
tokens = []
|
36
|
+
# current position in script
|
37
|
+
pos = 0
|
38
|
+
line_number = 0
|
39
|
+
column = 0
|
40
|
+
match = nil
|
41
|
+
# while pos isn't outside the script...
|
42
|
+
while pos < input.length
|
43
|
+
if input[pos] != "\n"
|
44
|
+
column += 1
|
45
|
+
else
|
46
|
+
line_number += 1
|
47
|
+
column = 0
|
48
|
+
end
|
49
|
+
# for each token type...
|
50
|
+
good = TOKENS.any? do |sym, regex|
|
51
|
+
# try to match it at the current position in the script
|
52
|
+
match = input.match(regex, pos)
|
53
|
+
# skip to next token type unless it matched at the current position
|
54
|
+
next unless match && match.begin(0) == pos
|
55
|
+
# ignore whitespace and comments
|
56
|
+
unless [:WS, :COMMENT].include?(sym)
|
57
|
+
# add new token to list of found tokens
|
58
|
+
# if it's an identifier, save the actual text found as well
|
59
|
+
tokens << Token.new(sym, line_number, column, sym == :IDENT ? match[0] : nil)
|
60
|
+
end
|
61
|
+
# move current position to just after the end of the found token
|
62
|
+
pos += match[0].length
|
63
|
+
true
|
64
|
+
end
|
65
|
+
fail TildeathError.new(line_number, column), "error: unrecognized token #{input[pos]}" unless good
|
66
|
+
end
|
67
|
+
return tokens
|
68
|
+
end
|
69
|
+
end
|
70
|
+
end
|
@@ -0,0 +1,136 @@
|
|
1
|
+
require_relative 'tildeath_error'
|
2
|
+
require_relative 'ast_nodes'
|
3
|
+
|
4
|
+
module Tildeath
|
5
|
+
module Parser
|
6
|
+
# null: NULL
|
7
|
+
def self.parse_null(tokens)
|
8
|
+
# shift the NULL off the token queue
|
9
|
+
tokens.shift
|
10
|
+
# return a new Null object
|
11
|
+
ASTNodes::Null.new
|
12
|
+
end
|
13
|
+
|
14
|
+
# import: IMPORT IDENT IDENT
|
15
|
+
def self.parse_import(tokens)
|
16
|
+
# Shift the IMPORT IDENT IDENT tokens off the queue, saving the type and
|
17
|
+
# name as symbols
|
18
|
+
type, name = tokens.shift(3)[1..2].map{|token| token.value.to_sym}
|
19
|
+
# return a new Import with the given type and name
|
20
|
+
ASTNodes::Import.new(type, name)
|
21
|
+
end
|
22
|
+
|
23
|
+
# value: THIS | IDENT
|
24
|
+
def self.parse_value(tokens)
|
25
|
+
ASTNodes::Value.new(tokens.shift.value.to_sym)
|
26
|
+
end
|
27
|
+
|
28
|
+
# bang: BANG value
|
29
|
+
def self.parse_bang(tokens)
|
30
|
+
# shift off BANG
|
31
|
+
tokens.shift
|
32
|
+
value = parse_value(tokens)
|
33
|
+
ASTNodes::Bang.new(operand)
|
34
|
+
end
|
35
|
+
|
36
|
+
# expression: bang | value
|
37
|
+
def self.parse_expression(tokens)
|
38
|
+
return parse_bang(tokens) if tokens[0].name == :BANG
|
39
|
+
parse_value(tokens)
|
40
|
+
end
|
41
|
+
|
42
|
+
# tildeath: TILDEATH LPAREN expression RPAREN LBRACE statements RBRACE EXECUTE LPAREN statements RPAREN
|
43
|
+
def self.parse_tildeath(tokens)
|
44
|
+
# shift off the first five tokens, saving the IDENT
|
45
|
+
victim = tokens.shift(5)[2]
|
46
|
+
victim = victim == :THIS ? victim.name : victim.value.to_sym
|
47
|
+
# parse the first statements
|
48
|
+
tildeath_body = parse_statements(tokens)
|
49
|
+
# shift off some punctuation
|
50
|
+
tokens.shift(3)
|
51
|
+
# parse the EXECUTE statements (or NULL)
|
52
|
+
execute_body = tokens[0].name == :NULL ? parse_null(tokens) : parse_statements(tokens)
|
53
|
+
# shift off the last RPAREN
|
54
|
+
tokens.shift
|
55
|
+
# return a new Tildeath with the parsed victim and statements
|
56
|
+
ASTNodes::Tildeath.new(victim, tildeath_body, execute_body)
|
57
|
+
end
|
58
|
+
|
59
|
+
# array: LBRACKET expression (COMMA expression)* RBRACKET
|
60
|
+
def self.parse_array(tokens)
|
61
|
+
elements = []
|
62
|
+
elements << tokens.shift(2)[1].value.to_sym
|
63
|
+
ASTNodes::Array.new(elements)
|
64
|
+
end
|
65
|
+
|
66
|
+
# dot_die: expression DOT DIE LPAREN RPAREN
|
67
|
+
def self.parse_dot_die(tokens)
|
68
|
+
# shift off all the tokens, keeping the IDENT's value
|
69
|
+
victim = tokens.shift(5)[0]
|
70
|
+
victim = victim.name == :THIS ? victim.name.to_sym : victim.value.to_sym
|
71
|
+
ASTNodes::DotDie.new(victim)
|
72
|
+
end
|
73
|
+
|
74
|
+
# bifurcate: BIFURC8 IDENT LBRACKET IDENT COMMA IDENT RBRACKET
|
75
|
+
def self.parse_bifurcate(tokens)
|
76
|
+
orig = tokens.shift(3)[1].value.to_sym
|
77
|
+
parts = tokens.shift(4).values_at(0, 2).map do |token|
|
78
|
+
token.value.to_sym
|
79
|
+
end
|
80
|
+
ASTNodes::Bifurcate.new(orig, parts)
|
81
|
+
end
|
82
|
+
|
83
|
+
# split: SPLIT IDENT LBRACKET IDENT (COMMA IDENT)* RBRACKET
|
84
|
+
def self.parse_split(tokens)
|
85
|
+
orig = tokens.shift(3)[1].value.to_sym
|
86
|
+
parts = [tokens.shift.value.to_sym]
|
87
|
+
while true
|
88
|
+
part = tokens.shift
|
89
|
+
case part.name
|
90
|
+
when :COMMA
|
91
|
+
parts << tokens.shift.value.to_sym
|
92
|
+
when :RBRACKET
|
93
|
+
break
|
94
|
+
end
|
95
|
+
end
|
96
|
+
ASTNodes::Split.new(orig, parts)
|
97
|
+
end
|
98
|
+
|
99
|
+
# statement: (import | tildeath | dot_die | bifurcate | split) SEMI
|
100
|
+
# TODO: make [THIS, THIS].DIE() legal
|
101
|
+
def self.parse_statement(tokens)
|
102
|
+
# Determine statement type based on first token, and parse it
|
103
|
+
token = tokens[0].name
|
104
|
+
ret = case token
|
105
|
+
when :IMPORT then parse_import(tokens)
|
106
|
+
when :TILDEATH then parse_tildeath(tokens)
|
107
|
+
when :THIS, :IDENT then parse_dot_die(tokens)
|
108
|
+
when :BIFURC8 then parse_bifucate(tokens)
|
109
|
+
when :SPLIT then parse_split(tokens)
|
110
|
+
when :RBRACE then return
|
111
|
+
else fail TildeathError.new(token.line_number, token.column), "error: unexpected token #{token}"
|
112
|
+
end
|
113
|
+
# shift off SEMI
|
114
|
+
fail TildeathError.new(tokens[0].line_number, tokens[0].column), 'missing semicolon' unless tokens[0].name == :SEMI
|
115
|
+
tokens.shift
|
116
|
+
ret
|
117
|
+
end
|
118
|
+
|
119
|
+
# statements: statement*
|
120
|
+
def self.parse_statements(tokens)
|
121
|
+
statements = []
|
122
|
+
# while there are tokens left and parse_statement returns non-nil...
|
123
|
+
while tokens.length > 0 && statement = parse_statement(tokens)
|
124
|
+
# add parsed statement to list of parsed statements
|
125
|
+
statements << statement
|
126
|
+
end
|
127
|
+
# return a new Statements object with the list of parsed statements
|
128
|
+
ASTNodes::Statements.new(statements)
|
129
|
+
end
|
130
|
+
|
131
|
+
# program: statements
|
132
|
+
def self.parse(tokens)
|
133
|
+
ASTNodes::Program.new(parse_statements(tokens))
|
134
|
+
end
|
135
|
+
end
|
136
|
+
end
|