pione 0.1.4 → 0.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (270) hide show
  1. data/.gitignore +4 -3
  2. data/.simplecov +7 -0
  3. data/.travis.yml +8 -0
  4. data/Gemfile +3 -0
  5. data/History.txt +13 -0
  6. data/README.md +2 -0
  7. data/Rakefile +16 -6
  8. data/bin/pione-val +5 -0
  9. data/example/AbstractRule/package.yml +1 -0
  10. data/example/HelloWorld/package.yml +2 -0
  11. data/example/HelloWorld/scenario/scenario.yml +2 -0
  12. data/example/LoopByTouch/LoopByTouch.pione +28 -0
  13. data/example/LoopByTouch/package.yml +1 -0
  14. data/example/MakePair/package.yml +2 -0
  15. data/example/MakePair/{data → scenario/input}/1.i +0 -0
  16. data/example/MakePair/{data → scenario/input}/2.i +0 -0
  17. data/example/MakePair/{data → scenario/input}/3.i +0 -0
  18. data/example/MakePair/{data → scenario/input}/4.i +0 -0
  19. data/example/MakePair/{data → scenario/input}/5.i +0 -0
  20. data/example/MakePair/scenario/scenario.yml +1 -0
  21. data/example/ScoreAggregation/ScoreAggregation.pione +122 -0
  22. data/example/ScoreAggregation/bin/apply-template.rb +10 -0
  23. data/example/ScoreAggregation/bin/histgram-graph.sh +18 -0
  24. data/example/ScoreAggregation/bin/mean-summary.rb +10 -0
  25. data/example/ScoreAggregation/bin/personal-bar-graph.sh +18 -0
  26. data/example/ScoreAggregation/bin/personal-pre-statistics.rb +37 -0
  27. data/example/ScoreAggregation/bin/total-mean.rb +10 -0
  28. data/example/ScoreAggregation/bin/total-statistics.rb +43 -0
  29. data/example/ScoreAggregation/generate-data.rb +63 -0
  30. data/example/ScoreAggregation/package.yml +1 -0
  31. data/example/ScoreAggregation/scenario/case1/input/A.score +15 -0
  32. data/example/ScoreAggregation/scenario/case1/input/B.score +15 -0
  33. data/example/ScoreAggregation/scenario/case1/input/C.score +15 -0
  34. data/example/ScoreAggregation/scenario/case1/input/D.score +15 -0
  35. data/example/ScoreAggregation/scenario/case1/input/E.score +15 -0
  36. data/example/ScoreAggregation/scenario/case1/input/F.score +15 -0
  37. data/example/ScoreAggregation/scenario/case1/input/G.score +15 -0
  38. data/example/ScoreAggregation/scenario/case1/input/H.score +15 -0
  39. data/example/ScoreAggregation/scenario/case1/input/I.score +15 -0
  40. data/example/ScoreAggregation/scenario/case1/input/J.score +15 -0
  41. data/example/ScoreAggregation/scenario/case1/input/K.score +15 -0
  42. data/example/ScoreAggregation/scenario/case1/input/L.score +15 -0
  43. data/example/ScoreAggregation/scenario/case1/input/M.score +15 -0
  44. data/example/ScoreAggregation/scenario/case1/input/N.score +15 -0
  45. data/example/ScoreAggregation/scenario/case1/input/O.score +15 -0
  46. data/example/ScoreAggregation/scenario/case1/input/P.score +15 -0
  47. data/example/ScoreAggregation/scenario/case1/input/Q.score +15 -0
  48. data/example/ScoreAggregation/scenario/case1/input/R.score +15 -0
  49. data/example/ScoreAggregation/scenario/case1/input/S.score +15 -0
  50. data/example/ScoreAggregation/scenario/case1/input/T.score +15 -0
  51. data/example/ScoreAggregation/scenario/case1/input/U.score +15 -0
  52. data/example/ScoreAggregation/scenario/case1/input/V.score +15 -0
  53. data/example/ScoreAggregation/scenario/case1/input/W.score +15 -0
  54. data/example/ScoreAggregation/scenario/case1/input/X.score +15 -0
  55. data/example/ScoreAggregation/scenario/case1/input/Y.score +15 -0
  56. data/example/ScoreAggregation/scenario/case1/input/Z.score +15 -0
  57. data/example/ScoreAggregation/scenario/case1/input/stat-template.erb +24 -0
  58. data/example/ScoreAggregation/scenario/case1/scenario.yml +3 -0
  59. data/example/{SequentialProcess/SequentialProcess.pione → SerialProcessing/SerialProcessing.pione} +0 -0
  60. data/example/SerialProcessing/package.yml +1 -0
  61. data/example/WorkflowPatterns/01_Sequence/Sequence.pione +21 -0
  62. data/example/WorkflowPatterns/02_PrallelSplit/ParallelSplit.pione +29 -0
  63. data/example/WorkflowPatterns/03_Synchronization/Synchronization.pione +30 -0
  64. data/example/WorkflowPatterns/04_ExclusiveChoice/ExclusiveChoice.pione +44 -0
  65. data/example/WorkflowPatterns/05_SimpleMerge/SimpleMerge.pione +44 -0
  66. data/example/WorkflowPatterns/06_MultiChoice/MultiChoice.pione +48 -0
  67. data/example/WorkflowPatterns/07_StructuredSynchronizingMerge/StructuredSynchronizingMerge.pione +71 -0
  68. data/example/WorkflowPatterns/08_MultiMerge/MultiMerge.pione +44 -0
  69. data/example/WorkflowPatterns/11_ImplicitTermination/ImplicitTermination.pione +29 -0
  70. data/example/WorkflowPatterns/12_MultipleInstancesWithoutSynchronization/MultipleInstancesWithoutSynchronization.pione +29 -0
  71. data/example/WorkflowPatterns/13_MultipleInstancesWithDesignTimeKnowledge/MultipleInstancesWithDesignTimeKnowledge.pione +28 -0
  72. data/example/WorkflowPatterns/14_MultipleInstancesWithRunTimeKnowledge/MultipleInstancesWithRunTimeKnowledge.pione +33 -0
  73. data/example/WorkflowPatterns/33_GeneralizedANDJoin/GeneralizedANDJoin.pione +21 -0
  74. data/example/WorkflowPatterns/37_LocalSynchronizingMerge/LocalSynchronizingMerge.pione +101 -0
  75. data/example/WorkflowPatterns/41_ThreadMerge/ThreadMerge.pione +43 -0
  76. data/example/WorkflowPatterns/42_ThreadSplit/ThreadSplit.pione +16 -0
  77. data/lib/pione.rb +23 -47
  78. data/lib/pione/agent/broker.rb +1 -1
  79. data/lib/pione/agent/input-generator.rb +1 -1
  80. data/lib/pione/agent/logger.rb +62 -49
  81. data/lib/pione/agent/messenger.rb +38 -0
  82. data/lib/pione/agent/rule-provider.rb +3 -3
  83. data/lib/pione/agent/task-worker.rb +14 -2
  84. data/lib/pione/agent/tuple-space-client.rb +1 -1
  85. data/lib/pione/agent/tuple-space-server-client-life-checker.rb +5 -3
  86. data/lib/pione/command.rb +2 -0
  87. data/lib/pione/command/basic-command.rb +1 -1
  88. data/lib/pione/command/pione-client.rb +57 -16
  89. data/lib/pione/command/pione-log.rb +5 -5
  90. data/lib/pione/command/pione-syntax-checker.rb +22 -7
  91. data/lib/pione/command/pione-task-worker.rb +36 -19
  92. data/lib/pione/command/pione-tuple-space-viewer.rb +3 -3
  93. data/lib/pione/command/pione-val.rb +40 -0
  94. data/lib/pione/component.rb +8 -0
  95. data/lib/pione/{system → component}/document.rb +13 -11
  96. data/lib/pione/component/package.rb +269 -0
  97. data/lib/pione/component/rule.rb +206 -0
  98. data/lib/pione/location/basic-location.rb +49 -7
  99. data/lib/pione/location/ftp-location.rb +19 -6
  100. data/lib/pione/location/local-location.rb +15 -6
  101. data/lib/pione/log.rb +4 -1
  102. data/lib/pione/log/{domain-info.rb → domain-log.rb} +23 -11
  103. data/lib/pione/{util/console-message.rb → log/message-log.rb} +16 -15
  104. data/lib/pione/log/process-log.rb +35 -86
  105. data/lib/pione/log/process-record.rb +24 -11
  106. data/lib/pione/log/system-log.rb +107 -0
  107. data/lib/pione/log/xes-log.rb +54 -48
  108. data/lib/pione/model.rb +1 -3
  109. data/lib/pione/model/assignment.rb +1 -0
  110. data/lib/pione/model/basic-model.rb +31 -15
  111. data/lib/pione/model/block.rb +1 -1
  112. data/lib/pione/model/boolean.rb +10 -23
  113. data/lib/pione/model/constraints.rb +12 -0
  114. data/lib/pione/model/data-expr.rb +31 -36
  115. data/lib/pione/model/feature-expr.rb +6 -23
  116. data/lib/pione/model/float.rb +29 -18
  117. data/lib/pione/model/integer.rb +44 -56
  118. data/lib/pione/model/keyed-sequence.rb +11 -4
  119. data/lib/pione/model/message.rb +4 -4
  120. data/lib/pione/model/ordinal-sequence.rb +12 -14
  121. data/lib/pione/model/package-expr.rb +53 -0
  122. data/lib/pione/model/parameters.rb +12 -11
  123. data/lib/pione/model/pione-method.rb +2 -2
  124. data/lib/pione/model/rule-expr.rb +30 -52
  125. data/lib/pione/model/sequence.rb +26 -26
  126. data/lib/pione/model/string.rb +13 -54
  127. data/lib/pione/model/ticket-expr.rb +12 -53
  128. data/lib/pione/model/type.rb +7 -1
  129. data/lib/pione/model/variable-table.rb +5 -4
  130. data/lib/pione/model/variable.rb +0 -5
  131. data/lib/pione/option/common-option.rb +2 -1
  132. data/lib/pione/option/task-worker-owner-option.rb +1 -1
  133. data/lib/pione/patch/drb-patch.rb +1 -1
  134. data/lib/pione/patch/em-ftpd-patch.rb +7 -0
  135. data/lib/pione/patch/rinda-patch.rb +53 -0
  136. data/lib/pione/patch/uri-patch.rb +5 -0
  137. data/lib/pione/relay/relay-socket.rb +2 -2
  138. data/lib/pione/rule-handler/action-handler.rb +71 -64
  139. data/lib/pione/rule-handler/basic-handler.rb +50 -32
  140. data/lib/pione/rule-handler/flow-handler.rb +51 -34
  141. data/lib/pione/rule-handler/update-criteria.rb +43 -20
  142. data/lib/pione/system.rb +13 -0
  143. data/lib/pione/system/domain-info.rb +34 -0
  144. data/lib/pione/system/global.rb +8 -1
  145. data/lib/pione/system/object.rb +1 -1
  146. data/lib/pione/transformer/expr-transformer.rb +3 -3
  147. data/lib/pione/transformer/literal-transformer.rb +3 -3
  148. data/lib/pione/transformer/rule-definition-transformer.rb +11 -9
  149. data/lib/pione/tuple-space/tuple-space-server-interface.rb +19 -1
  150. data/lib/pione/tuple-space/tuple-space-server.rb +9 -2
  151. data/lib/pione/tuple.rb +30 -0
  152. data/lib/pione/tuple/data-null-tuple.rb +14 -0
  153. data/lib/pione/tuple/message-tuple.rb +7 -0
  154. data/lib/pione/tuple/process-log-tuple.rb +13 -0
  155. data/lib/pione/tuple/touch-tuple.rb +14 -0
  156. data/lib/pione/uri-scheme/myftp-scheme.rb +45 -0
  157. data/lib/pione/util.rb +18 -0
  158. data/lib/pione/util/cpu.rb +20 -0
  159. data/lib/pione/util/error-report.rb +9 -1
  160. data/lib/pione/util/evaluatable.rb +47 -0
  161. data/lib/pione/util/ftp-server.rb +468 -0
  162. data/lib/pione/util/id.rb +39 -0
  163. data/lib/pione/util/indentation.rb +19 -0
  164. data/lib/pione/util/ip-address.rb +35 -0
  165. data/lib/pione/util/misc.rb +0 -47
  166. data/lib/pione/util/uuid.rb +28 -0
  167. data/lib/pione/util/variable-holdable.rb +38 -0
  168. data/lib/pione/version.rb +1 -1
  169. data/pione.gemspec +16 -14
  170. data/test/agent/spec_logger.rb +35 -44
  171. data/test/agent/spec_messenger.rb +25 -0
  172. data/test/agent/spec_rule-provider.rb +1 -1
  173. data/test/agent/spec_task-worker.rb +7 -13
  174. data/test/{system → component}/spec_document.rb +20 -20
  175. data/test/component/spec_package.rb +77 -0
  176. data/test/component/spec_package/TestPackage/Test.pione +14 -0
  177. data/test/component/spec_package/TestPackage/bin/count +3 -0
  178. data/test/component/spec_package/TestPackage/package.yml +7 -0
  179. data/test/component/spec_package/TestPackage/scenario/case1/input/1.txt +1 -0
  180. data/test/component/spec_package/TestPackage/scenario/case1/scenario.yml +1 -0
  181. data/test/component/spec_rule.rb +140 -0
  182. data/test/endurance-test/graph.plt +12 -0
  183. data/test/endurance-test/run.sh +65 -0
  184. data/test/location/location-behavior.rb +125 -0
  185. data/test/location/spec_ftp-location.rb +28 -90
  186. data/test/location/spec_local-location.rb +26 -76
  187. data/test/log/raw-process-log/pione-process.log +710 -0
  188. data/test/log/spec_domain-log.rb +57 -0
  189. data/test/log/spec_message-log.rb +44 -0
  190. data/test/log/spec_process-log.rb +88 -0
  191. data/test/log/spec_process-record.rb +148 -0
  192. data/test/log/spec_system-log.rb +142 -0
  193. data/test/log/spec_xes-log.rb +2 -2
  194. data/test/model/spec_assignment.rb +26 -25
  195. data/test/model/spec_block.rb +51 -97
  196. data/test/model/spec_boolean.rb +0 -5
  197. data/test/model/spec_call-rule.rb +6 -6
  198. data/test/model/spec_data-expr.rb +0 -7
  199. data/test/model/spec_data-expr.yml +8 -0
  200. data/test/model/spec_float.rb +1 -1
  201. data/test/model/spec_float.yml +13 -0
  202. data/test/model/spec_integer.rb +1 -1
  203. data/test/model/spec_integer.yml +12 -1
  204. data/test/model/spec_keyed-sequence.yml +5 -0
  205. data/test/model/spec_message.rb +2 -4
  206. data/test/model/spec_package-expr.rb +19 -0
  207. data/test/model/spec_parameters.rb +56 -88
  208. data/test/model/spec_pione-method.rb +15 -14
  209. data/test/model/spec_rule-expr.rb +4 -52
  210. data/test/model/spec_rule-expr.yml +11 -0
  211. data/test/model/spec_sequence.yml +6 -0
  212. data/test/model/spec_string.rb +21 -13
  213. data/test/model/spec_string.yml +10 -2
  214. data/test/model/spec_ticket-expr.rb +25 -3
  215. data/test/model/spec_variable.rb +8 -8
  216. data/test/rule-handler/spec_action-handler.pione +37 -0
  217. data/test/rule-handler/spec_action-handler.rb +41 -97
  218. data/test/rule-handler/spec_flow-handler.rb +24 -30
  219. data/test/rule-handler/spec_update-criteria.pione +16 -0
  220. data/test/rule-handler/spec_update-criteria.rb +7 -4
  221. data/test/rule-handler/spec_update-criteria.yml +191 -0
  222. data/test/system/spec_domain-info.rb +25 -0
  223. data/test/test-util.rb +38 -228
  224. data/test/transformer/spec_block-transformer.rb +1 -1
  225. data/test/transformer/spec_expr-transformer.rb +30 -39
  226. data/test/transformer/spec_flow-element-transformer.rb +17 -17
  227. data/test/transformer/spec_literal-transformer.rb +3 -3
  228. data/test/transformer/spec_rule-definition-transformer.rb +55 -76
  229. data/test/{spec_data-finder.rb → tuple-space/spec_data-finder.rb} +1 -1
  230. data/test/{spec_data-finder.yml → tuple-space/spec_data-finder.yml} +0 -0
  231. data/test/tuple/spec_data-tuple.rb +5 -76
  232. data/test/tuple/spec_finished-tuple.rb +5 -46
  233. data/test/tuple/spec_message-tuple.rb +10 -0
  234. data/test/tuple/spec_task-tuple.rb +4 -110
  235. data/test/tuple/spec_touch-tuple.rb +14 -0
  236. data/test/tuple/tuple-behavior.rb +14 -0
  237. data/test/uri-scheme/spec_local-scheme.rb +1 -1
  238. data/test/uri-scheme/spec_myftp-scheme.rb +73 -0
  239. data/test/uri-scheme/spec_myftp-scheme.yml +85 -0
  240. data/test/util/spec_cpu.rb +8 -0
  241. data/test/util/spec_error-report.rb +42 -0
  242. data/test/util/spec_evaluatable.rb +18 -0
  243. data/test/util/spec_ftp-server.rb +249 -0
  244. data/test/util/spec_id.pione +14 -0
  245. data/test/util/spec_id.rb +77 -0
  246. data/test/util/spec_indentation.rb +77 -0
  247. data/test/util/spec_ip-address.rb +15 -0
  248. data/test/util/spec_uuid.rb +11 -0
  249. data/test/util/spec_variable-holdable.rb +69 -0
  250. metadata +282 -83
  251. data/bin/pione-eval +0 -111
  252. data/demo/demo.rb +0 -311
  253. data/demo/public/base.css +0 -94
  254. data/demo/public/demo.js +0 -107
  255. data/demo/public/index.html +0 -91
  256. data/demo/public/jquery-1.8.3.min.js +0 -2
  257. data/lib/pione/model/binary-operator.rb +0 -90
  258. data/lib/pione/model/list.rb +0 -108
  259. data/lib/pione/model/package.rb +0 -56
  260. data/lib/pione/model/rule-io.rb +0 -178
  261. data/lib/pione/model/rule.rb +0 -295
  262. data/lib/pione/system/identifier.rb +0 -61
  263. data/lib/pione/tuple/log-tuple.rb +0 -14
  264. data/lib/pione/util/terminal.rb +0 -78
  265. data/test/log/data/sample.log +0 -1003
  266. data/test/model/spec_binary-operator.rb +0 -39
  267. data/test/model/spec_package.rb +0 -15
  268. data/test/model/spec_rule.rb +0 -158
  269. data/test/spec_identifier.rb +0 -29
  270. data/test/spec_log.rb +0 -52
@@ -65,7 +65,7 @@ module Pione
65
65
  start do
66
66
  # print each bags
67
67
  @tuple_space_servers.each do |address, tuple_space_server|
68
- puts "TupleSpaceServer: %s" % Terminal.red(address)
68
+ puts "TupleSpaceServer: %s" % address.color(:red)
69
69
  puts "-"*78
70
70
  if option[:bag_type] == :bag or option[:bag_type].nil?
71
71
  puts "*** bag ***"
@@ -157,8 +157,8 @@ module Pione
157
157
 
158
158
  # show
159
159
  res = PP.pp(tuple, "")
160
- res.gsub!(/\:[a-z]\w+/) {|s| Terminal.red(s) }
161
- res.gsub!(/\#<(\S+)/) {|s| "#<%s" % Terminal.green($1) }
160
+ res.gsub!(/\:[a-z]\w+/) {|s| s.color(:red) }
161
+ res.gsub!(/\#<(\S+)/) {|s| "#<%s" % $1.color(:green) }
162
162
  puts res
163
163
  end
164
164
  end
@@ -0,0 +1,40 @@
1
+ module Pione
2
+ module Command
3
+ # PioneSyntaxChecker is a command for checking syntax tree and model of the
4
+ # PIONE document.
5
+ class PioneEval < BasicCommand
6
+ define_info do
7
+ set_name "pione-val"
8
+ set_banner "Get the evaluation result value of the PIONE expression."
9
+ end
10
+
11
+ define_option do
12
+ default :domain_info, Location["./domain.dump"]
13
+
14
+ option('--domain-info=LOCATION', 'location of Domain info file') do |data, location|
15
+ data[:domain_info] = Location[location]
16
+ end
17
+ end
18
+
19
+ start do
20
+ # get expression string
21
+ str = ARGV[0] || abort("no expressions")
22
+
23
+ # setup domain info
24
+ domain_info = nil
25
+ if option[:domain_info].exist?
26
+ domain_info = System::DomainInfo.read(option[:domain_info])
27
+ end
28
+
29
+ begin
30
+ # evaluate it and print the result
31
+ $stdout.puts Pione.val(str, domain_info)
32
+ exit
33
+ rescue => e
34
+ abort("error: %s" % e)
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
40
+
@@ -0,0 +1,8 @@
1
+ module Pione
2
+ module Component
3
+ end
4
+ end
5
+
6
+ require 'pione/component/rule'
7
+ require 'pione/component/document'
8
+ require 'pione/component/package'
@@ -1,5 +1,5 @@
1
1
  module Pione
2
- module System
2
+ module Component
3
3
  class Document < PioneObject
4
4
  # Add ruby shebang line.
5
5
  def ruby(str, charset=nil)
@@ -9,19 +9,21 @@ module Pione
9
9
  end
10
10
 
11
11
  # Load a rule document and return it.
12
- def self.load(filepath)
13
- parse(File.read(filepath))
12
+ def self.load(filepath, package_name="main")
13
+ filepath = Location[filepath] unless filepath.kind_of?(Location::BasicLocation)
14
+ parse(filepath.read, package_name)
14
15
  end
15
16
 
16
17
  # Parse a rule document string.
17
- def self.parse(src)
18
+ def self.parse(src, package_name="main")
19
+ src = src.read if src.kind_of?(Location::BasicLocation)
18
20
  # parse the document and build the model
19
21
  parser = Parser::DocumentParser.new
20
- transformer = Transformer::DocumentTransformer.new
22
+ transformer = Transformer::DocumentTransformer.new(package_name)
21
23
  toplevels = transformer.apply(parser.parse(src))
22
24
 
23
25
  # rules and assignments
24
- rules = toplevels.select{|elt| elt.kind_of?(Rule)}
26
+ rules = toplevels.select{|elt| elt.kind_of?(Component::Rule)}
25
27
  assignments = toplevels.select{|elt| elt.kind_of?(Assignment)}
26
28
  assignments.each {|assignment| assignment.set_toplevel(true)}
27
29
  user_params = Naming::ParamLine.values(toplevels)
@@ -41,11 +43,11 @@ module Pione
41
43
  end.to_params
42
44
 
43
45
  # set document parameters into rules
44
- rules.each {|rule| rule.params.merge!(params)}
46
+ rules.each {|rule| rule.condition.params.merge!(params)}
45
47
 
46
48
  # make rule table
47
49
  table = rules.inject({}) do |tbl, rule|
48
- tbl.tap{|x| x[rule.rule_path] = rule}
50
+ tbl.tap{|x| x[rule.path] = rule}
49
51
  end
50
52
  return new(table, params)
51
53
  end
@@ -65,7 +67,7 @@ module Pione
65
67
  # rule path
66
68
  # @return [Pione::Model::Rule]
67
69
  def [](name)
68
- @rules[name].params.merge!(@params)
70
+ @rules[name].condition.params.merge!(@params)
69
71
  @rules[name]
70
72
  end
71
73
 
@@ -73,7 +75,7 @@ module Pione
73
75
  # @return [Rule]
74
76
  # main rule of main package
75
77
  def main
76
- @rules["&main:Main"].params.merge!(@params)
78
+ @rules["&main:Main"].condition.params.merge!(@params)
77
79
  @rules["&main:Main"]
78
80
  end
79
81
 
@@ -83,7 +85,7 @@ module Pione
83
85
  # @return [RootRule]
84
86
  # root rule
85
87
  def root_rule(params)
86
- Rule::RootRule.new(main, @params.merge(params))
88
+ Component::RootRule.new(main, @params.merge(params))
87
89
  end
88
90
  end
89
91
  end
@@ -0,0 +1,269 @@
1
+ module Pione
2
+ module Component
3
+ # InvalidPackageError raises when the package is something bad.
4
+ class InvalidPackageError < StandardError
5
+ attr_reader :package
6
+
7
+ def initialize(package, msg)
8
+ @package = package
9
+ @msg = msg
10
+ end
11
+
12
+ def message
13
+ @msg
14
+ end
15
+ end
16
+
17
+ # Package is a container of rules, script, scenario, and etc.
18
+ class Package
19
+ include SimpleIdentity
20
+
21
+ attr_reader :info
22
+ attr_reader :bin
23
+ attr_reader :scenarios
24
+ attr_reader :documents
25
+ attr_reader :rules
26
+ attr_reader :params
27
+
28
+ forward_as_key :@info, "PackageName", :name
29
+
30
+ # @param info [Hash]
31
+ # package information table
32
+ # @param bin [Location]
33
+ # package bin directory
34
+ # @param scenarios [Array<PackageScenario>]
35
+ # scenarios
36
+ # @param documents [Array<Document>]
37
+ # PIONE documents
38
+ def initialize(info, bin, scenarios, documents)
39
+ @info = info
40
+ @bin = bin
41
+ @scenarios = scenarios
42
+ @documents = documents
43
+ build_rules
44
+ build_params
45
+ end
46
+
47
+ # Return the package entry rule.
48
+ #
49
+ # @return [Rule]
50
+ # entry rule
51
+ def main
52
+ @rules["&%s:Main" % name].tap{|x| x.condition.params.merge!(@params)}
53
+ end
54
+
55
+ # Return the named rule.
56
+ #
57
+ # @param [String] name
58
+ # rule path
59
+ # @return [Rule]
60
+ # the rule
61
+ def [](name)
62
+ @rules[name].params.merge!(@params)
63
+ @rules[name]
64
+ end
65
+
66
+ # Return root rule.
67
+ #
68
+ # @param prams [Parameters]
69
+ # root parameters
70
+ # @return [RootRule]
71
+ # root rule
72
+ def root_rule(params)
73
+ Component::RootRule.new(main, @params.merge(params))
74
+ end
75
+
76
+ # Upload the package files to the location.
77
+ #
78
+ # @return [void]
79
+ def upload(dest)
80
+ if @bin.exist?
81
+ # upload bin files
82
+ @bin.entries.each do |entry|
83
+ entry.copy(dest + name + "bin" + entry.basename)
84
+ end
85
+ end
86
+ end
87
+
88
+ # Find scenario that have the name.
89
+ #
90
+ # @param name [String]
91
+ # scenario name
92
+ # @return [PackageScenario]
93
+ # the scenario
94
+ def find_scenario(name)
95
+ if name == :anything
96
+ @scenarios.first
97
+ else
98
+ @scenarios.find {|scenario| scenario.name == name}
99
+ end
100
+ end
101
+
102
+ private
103
+
104
+ # Build rules from all documents in the package.
105
+ def build_rules
106
+ @rules = {}
107
+ @documents.each do |doc|
108
+ doc.rules.each do |name, rule|
109
+ unless @rules[name]
110
+ @rules[name] = rule
111
+ else
112
+ raise InvalidPackage.new(self, "duplicated rules: %s" % name)
113
+ end
114
+ end
115
+ end
116
+ end
117
+
118
+ # Build parameters from all documents in the package.
119
+ def build_params
120
+ @params = Model::Parameters.empty
121
+ @documents.each do |doc|
122
+ doc.params.each do |param|
123
+ @params.merge!(param)
124
+ end
125
+ end
126
+ end
127
+ end
128
+
129
+ # PackageReader is a reader for packages.
130
+ class PackageReader
131
+ # @param location [Location]
132
+ # package location
133
+ def initialize(location)
134
+ @location = location
135
+ end
136
+
137
+ # Read the package.
138
+ #
139
+ # @return [Package]
140
+ # the package
141
+ def read
142
+ infos = read_package_info
143
+ bin = @location + "bin"
144
+ scenarios = find_scenarios
145
+ documents = find_documents(infos["PackageName"])
146
+ Package.new(infos, bin, scenarios, documents)
147
+ end
148
+
149
+ private
150
+
151
+ # Read the informations from the package location.
152
+ #
153
+ # @return [Hash]
154
+ # package information table
155
+ def read_package_info
156
+ YAML.load((@location + "package.yml").read)
157
+ rescue Location::NotFound
158
+ raise InvalidPackageError.new(self, "package.yml not found in %s" % @location.uri)
159
+ end
160
+
161
+ # Find scenarios from the package location.
162
+ #
163
+ # @return [Array<PackageScenario>]
164
+ # scenarios
165
+ def find_scenarios
166
+ if (@location + "scenario" + "scenario.yml").exist?
167
+ [PackageScenarioReader.read(@location + "scenario")]
168
+ else
169
+ if (@location + "scenario").exist? and (@location + "scenario").directory?
170
+ (@location + "scenario").entries.map do |scenario|
171
+ PackageScenarioReader.read(scenario)
172
+ end.compact
173
+ else
174
+ []
175
+ end
176
+ end
177
+ end
178
+
179
+ # Find documents from the packcage location.
180
+ #
181
+ # @return [Array<Document>]
182
+ # documents
183
+ def find_documents(package_name)
184
+ @location.entries.select do |entry|
185
+ entry.file? and entry.path.extname == ".pione"
186
+ end.map {|entry| Document.load(entry, package_name) }
187
+ end
188
+ end
189
+
190
+ # PackageScenario is a class for expected scenario of rule's behavior.
191
+ class PackageScenario
192
+ include SimpleIdentity
193
+
194
+ attr_reader :location
195
+ attr_reader :info
196
+
197
+ forward_as_key :@info, "ScenarioName", :name
198
+
199
+ # @param location [BasicLocation]
200
+ # scenario location
201
+ # @param info [Hash]
202
+ # scenario information table
203
+ def initialize(location, info)
204
+ @location = location
205
+ @info = info
206
+ end
207
+
208
+ # Return the input location. If the scenario doesn't have input location,
209
+ # return nil.
210
+ #
211
+ # @return [BasicLocation]
212
+ # the input location
213
+ def input
214
+ input_location = @location + "input"
215
+ input_location if input_location.exist?
216
+ end
217
+
218
+ # Return the output location.
219
+ #
220
+ # @return [BasicLocation]
221
+ # the output location
222
+ def output
223
+ @location + "output"
224
+ end
225
+ end
226
+
227
+ # PackageScenarioReader is a reader for loading scenarios.
228
+ class PackageScenarioReader
229
+ def self.read(location)
230
+ new(location).read
231
+ end
232
+
233
+ # @param location [Location]
234
+ # the scenario location
235
+ def initialize(location)
236
+ @location = location
237
+ end
238
+
239
+ # Read scenario files.
240
+ #
241
+ # @return [PackageScenario]
242
+ # the scenario
243
+ def read
244
+ begin
245
+ info = read_scenario_informations
246
+ PackageScenario.new(@location, info)
247
+ rescue
248
+ nil
249
+ end
250
+ end
251
+
252
+ private
253
+
254
+ # Read scenario information table.
255
+ #
256
+ # @return [Hash]
257
+ # scenario information table
258
+ def read_scenario_informations
259
+ path = @location + "scenario.yml"
260
+ if path.exist?
261
+ YAML.load(path.read)
262
+ else
263
+ {"ScenarioName" => @location.basename}
264
+ end
265
+ end
266
+ end
267
+ end
268
+ end
269
+
@@ -0,0 +1,206 @@
1
+ module Pione
2
+ module Component
3
+ # RuleCondition represents rule condition.
4
+ class RuleCondition < StructX
5
+ include Util::VariableHoldable
6
+
7
+ member :inputs, default: []
8
+ member :outputs, default: []
9
+ member :params, default: Model::Parameters.empty
10
+ member :features, default: Model::Feature.empty
11
+ member :constraints, default: Model::Constraints.empty
12
+ member :input_ticket_expr, default: Model::TicketExprSequence.empty
13
+ member :output_ticket_expr, default: Model::TicketExprSequence.empty
14
+
15
+ hold_variables members
16
+ end
17
+
18
+ # Rule is a class for PIONE rule model.
19
+ class Rule < PioneObject
20
+ include SimpleIdentity
21
+ include Util::VariableHoldable
22
+
23
+ hold_variables :rule_expr, :condition, :body
24
+
25
+ class << self
26
+ attr_reader :rule_type
27
+ attr_reader :handler_class
28
+
29
+ # Declare rule type of the class.
30
+ #
31
+ # @param rule_type [Symbol]
32
+ # rule type
33
+ # @return [void]
34
+ def set_rule_type(rule_type)
35
+ @rule_type = rule_type
36
+ end
37
+
38
+ # Declare rule handler class of the class.
39
+ #
40
+ # @param handler_class [Class]
41
+ # handler class
42
+ # @return [void]
43
+ def set_handler_class(handler_class)
44
+ @handler_class = handler_class
45
+ end
46
+ end
47
+
48
+ forward! :class, :rule_type, :handler_class
49
+
50
+ attr_reader :package_name
51
+ attr_reader :name
52
+ attr_reader :condition
53
+ attr_reader :body
54
+
55
+ # Create a rule.
56
+ #
57
+ # @param package_name [String]
58
+ # package name
59
+ # @param name [String]
60
+ # rule name
61
+ # @param condition [RuleCondition]
62
+ # rule condition
63
+ # @param [Block] body
64
+ # rule body block
65
+ def initialize(package_name, name, condition, body)
66
+ @package_name = package_name
67
+ @name = name
68
+ @condition = condition
69
+ @body = body
70
+ end
71
+
72
+ # Return true if this is a kind of action rule.
73
+ #
74
+ # @return [Boolean]
75
+ # true if this is a kind of action rule, or false
76
+ def action?
77
+ rule_type == :action
78
+ end
79
+
80
+ # Return true if this is a kind of flow rule.
81
+ #
82
+ # @return [Boolean]
83
+ # true if this is a kind of flow rule, or false
84
+ def flow?
85
+ rule_type == :flow
86
+ end
87
+
88
+ def path
89
+ "&%s:%s" % [@package_name, @name]
90
+ end
91
+
92
+ # Make a task handler object for the rule.
93
+ #
94
+ # @param ts_server [TupleSpaceServer]
95
+ # tuple space server
96
+ # @param inputs [Array<DataTuple, Array<DataTuple>>]
97
+ # input tuples
98
+ # @param params [Parameters]
99
+ # rule parameters
100
+ # @param call_stack [Array<String>]
101
+ # call stack
102
+ # @return [RuleHandler]
103
+ # rule handler object
104
+ def make_handler(ts_server, inputs, params, call_stack, opts={})
105
+ handler_class.new(ts_server, self, inputs, params, call_stack, opts)
106
+ end
107
+ end
108
+
109
+ # ActionRule is a rule that have some actions. This rule makes data
110
+ # processing.
111
+ class ActionRule < Rule
112
+ set_rule_type :action
113
+ set_handler_class RuleHandler::ActionHandler
114
+ end
115
+
116
+ # FlowRule is a rule that have flow elements. This rule makes processing flow.
117
+ class FlowRule < Rule
118
+ set_rule_type :flow
119
+ set_handler_class RuleHandler::FlowHandler
120
+ end
121
+
122
+ # EmptyRule is a rule that have no body. This rule is useful when you need
123
+ # no flows and no actions.
124
+ class EmptyRule < Rule
125
+ set_rule_type :action
126
+ set_handler_class RuleHandler::EmptyHandler
127
+ end
128
+
129
+ # RootRule is a hidden toplevel rule. This rule has same flow as the follow:
130
+ # Rule Root
131
+ # input '*'.all
132
+ # output '*'.all.except('{$INPUT[1]}')
133
+ # Flow
134
+ # rule &main:Main
135
+ # End
136
+ class RootRule < Rule
137
+ set_rule_type :flow
138
+ set_handler_class RuleHandler::RootHandler
139
+
140
+ INPUT_DOMAIN = 'input'
141
+
142
+ # root domain has no digest and no package
143
+ ROOT_DOMAIN = 'Root'
144
+
145
+ attr_reader :main
146
+
147
+ # Make a root rule.
148
+ #
149
+ # @param main [Rule]
150
+ # main rule
151
+ # @param params [Parameters]
152
+ # main rule's parameters
153
+ def initialize(main, params=Parameters.empty)
154
+ @main = main
155
+ @params = params
156
+ @domain = ROOT_DOMAIN
157
+ rule_expr = RuleExpr.new(PackageExpr.new(main.package_name), main.name)
158
+ condition = RuleCondition.new(@main.condition.inputs, @main.condition.outputs)
159
+ block = FlowBlock.new(CallRule.new(rule_expr.set_params(@params)))
160
+ super("Root", "Root", condition, block)
161
+ end
162
+
163
+ def make_handler(ts_server)
164
+ # build parameter
165
+ params = @main.condition.params.merge(@params)
166
+
167
+ # find inputs
168
+ finder = DataFinder.new(ts_server, INPUT_DOMAIN)
169
+ results = finder.find(:input, @main.condition.inputs, params.as_variable_table)
170
+ if results.empty? and not(@main.condition.inputs.empty?)
171
+ return nil
172
+ end
173
+ inputs = @main.condition.inputs.empty? ? [] : results.first.combination
174
+
175
+ # make handler
176
+ handler_class.new(ts_server, self, inputs, params, [], domain: @domain)
177
+ end
178
+ end
179
+
180
+ # SystemRule represents built-in rule definition. System rules belong to
181
+ # 'system' package.
182
+ class SystemRule < Rule
183
+ set_rule_type :action
184
+ set_handler_class RuleHandler::SystemHandler
185
+
186
+ # Create a system rule model.
187
+ #
188
+ # @param name [String]
189
+ # rule name
190
+ # @param [Proc] b
191
+ # rule process
192
+ def initialize(name, &b)
193
+ condition = RuleCondition.new([Model::DataExpr.new('*').to_seq.set_all], [])
194
+ super('System', name, condition, b)
195
+ end
196
+ end
197
+
198
+ # &System:Terminate rule
199
+ SYSTEM_TERMINATE = SystemRule.new('Terminate') do |tuple_space_server|
200
+ user_message "!!! Terminate processing !!!"
201
+ tuple_space_server.write(Tuple[:command].new("terminate"))
202
+ end
203
+
204
+ SYSTEM_RULES = [ SYSTEM_TERMINATE ]
205
+ end
206
+ end