codemodels 0.2.2-java → 0.2.3-java

Sign up to get free protection for your applications and to get access to all the features.
data/LICENSE CHANGED
@@ -1,20 +1,191 @@
1
- The MIT License (MIT)
2
-
3
- Copyright (c) 2013 ftomassetti
4
-
5
- Permission is hereby granted, free of charge, to any person obtaining a copy of
6
- this software and associated documentation files (the "Software"), to deal in
7
- the Software without restriction, including without limitation the rights to
8
- use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9
- the Software, and to permit persons to whom the Software is furnished to do so,
10
- subject to the following conditions:
11
-
12
- The above copyright notice and this permission notice shall be included in all
13
- copies or substantial portions of the Software.
14
-
15
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
17
- FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
18
- COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
19
- IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20
- CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction, and
10
+ distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by the copyright
13
+ owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all other entities
16
+ that control, are controlled by, or are under common control with that entity.
17
+ For the purposes of this definition, "control" means (i) the power, direct or
18
+ indirect, to cause the direction or management of such entity, whether by
19
+ contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the
20
+ outstanding shares, or (iii) beneficial ownership of such entity.
21
+
22
+ "You" (or "Your") shall mean an individual or Legal Entity exercising
23
+ permissions granted by this License.
24
+
25
+ "Source" form shall mean the preferred form for making modifications, including
26
+ but not limited to software source code, documentation source, and configuration
27
+ files.
28
+
29
+ "Object" form shall mean any form resulting from mechanical transformation or
30
+ translation of a Source form, including but not limited to compiled object code,
31
+ generated documentation, and conversions to other media types.
32
+
33
+ "Work" shall mean the work of authorship, whether in Source or Object form, made
34
+ available under the License, as indicated by a copyright notice that is included
35
+ in or attached to the work (an example is provided in the Appendix below).
36
+
37
+ "Derivative Works" shall mean any work, whether in Source or Object form, that
38
+ is based on (or derived from) the Work and for which the editorial revisions,
39
+ annotations, elaborations, or other modifications represent, as a whole, an
40
+ original work of authorship. For the purposes of this License, Derivative Works
41
+ shall not include works that remain separable from, or merely link (or bind by
42
+ name) to the interfaces of, the Work and Derivative Works thereof.
43
+
44
+ "Contribution" shall mean any work of authorship, including the original version
45
+ of the Work and any modifications or additions to that Work or Derivative Works
46
+ thereof, that is intentionally submitted to Licensor for inclusion in the Work
47
+ by the copyright owner or by an individual or Legal Entity authorized to submit
48
+ on behalf of the copyright owner. For the purposes of this definition,
49
+ "submitted" means any form of electronic, verbal, or written communication sent
50
+ to the Licensor or its representatives, including but not limited to
51
+ communication on electronic mailing lists, source code control systems, and
52
+ issue tracking systems that are managed by, or on behalf of, the Licensor for
53
+ the purpose of discussing and improving the Work, but excluding communication
54
+ that is conspicuously marked or otherwise designated in writing by the copyright
55
+ owner as "Not a Contribution."
56
+
57
+ "Contributor" shall mean Licensor and any individual or Legal Entity on behalf
58
+ of whom a Contribution has been received by Licensor and subsequently
59
+ incorporated within the Work.
60
+
61
+ 2. Grant of Copyright License.
62
+
63
+ Subject to the terms and conditions of this License, each Contributor hereby
64
+ grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
65
+ irrevocable copyright license to reproduce, prepare Derivative Works of,
66
+ publicly display, publicly perform, sublicense, and distribute the Work and such
67
+ Derivative Works in Source or Object form.
68
+
69
+ 3. Grant of Patent License.
70
+
71
+ Subject to the terms and conditions of this License, each Contributor hereby
72
+ grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
73
+ irrevocable (except as stated in this section) patent license to make, have
74
+ made, use, offer to sell, sell, import, and otherwise transfer the Work, where
75
+ such license applies only to those patent claims licensable by such Contributor
76
+ that are necessarily infringed by their Contribution(s) alone or by combination
77
+ of their Contribution(s) with the Work to which such Contribution(s) was
78
+ submitted. If You institute patent litigation against any entity (including a
79
+ cross-claim or counterclaim in a lawsuit) alleging that the Work or a
80
+ Contribution incorporated within the Work constitutes direct or contributory
81
+ patent infringement, then any patent licenses granted to You under this License
82
+ for that Work shall terminate as of the date such litigation is filed.
83
+
84
+ 4. Redistribution.
85
+
86
+ You may reproduce and distribute copies of the Work or Derivative Works thereof
87
+ in any medium, with or without modifications, and in Source or Object form,
88
+ provided that You meet the following conditions:
89
+
90
+ You must give any other recipients of the Work or Derivative Works a copy of
91
+ this License; and
92
+ You must cause any modified files to carry prominent notices stating that You
93
+ changed the files; and
94
+ You must retain, in the Source form of any Derivative Works that You distribute,
95
+ all copyright, patent, trademark, and attribution notices from the Source form
96
+ of the Work, excluding those notices that do not pertain to any part of the
97
+ Derivative Works; and
98
+ If the Work includes a "NOTICE" text file as part of its distribution, then any
99
+ Derivative Works that You distribute must include a readable copy of the
100
+ attribution notices contained within such NOTICE file, excluding those notices
101
+ that do not pertain to any part of the Derivative Works, in at least one of the
102
+ following places: within a NOTICE text file distributed as part of the
103
+ Derivative Works; within the Source form or documentation, if provided along
104
+ with the Derivative Works; or, within a display generated by the Derivative
105
+ Works, if and wherever such third-party notices normally appear. The contents of
106
+ the NOTICE file are for informational purposes only and do not modify the
107
+ License. You may add Your own attribution notices within Derivative Works that
108
+ You distribute, alongside or as an addendum to the NOTICE text from the Work,
109
+ provided that such additional attribution notices cannot be construed as
110
+ modifying the License.
111
+ You may add Your own copyright statement to Your modifications and may provide
112
+ additional or different license terms and conditions for use, reproduction, or
113
+ distribution of Your modifications, or for any such Derivative Works as a whole,
114
+ provided Your use, reproduction, and distribution of the Work otherwise complies
115
+ with the conditions stated in this License.
116
+
117
+ 5. Submission of Contributions.
118
+
119
+ Unless You explicitly state otherwise, any Contribution intentionally submitted
120
+ for inclusion in the Work by You to the Licensor shall be under the terms and
121
+ conditions of this License, without any additional terms or conditions.
122
+ Notwithstanding the above, nothing herein shall supersede or modify the terms of
123
+ any separate license agreement you may have executed with Licensor regarding
124
+ such Contributions.
125
+
126
+ 6. Trademarks.
127
+
128
+ This License does not grant permission to use the trade names, trademarks,
129
+ service marks, or product names of the Licensor, except as required for
130
+ reasonable and customary use in describing the origin of the Work and
131
+ reproducing the content of the NOTICE file.
132
+
133
+ 7. Disclaimer of Warranty.
134
+
135
+ Unless required by applicable law or agreed to in writing, Licensor provides the
136
+ Work (and each Contributor provides its Contributions) on an "AS IS" BASIS,
137
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
138
+ including, without limitation, any warranties or conditions of TITLE,
139
+ NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are
140
+ solely responsible for determining the appropriateness of using or
141
+ redistributing the Work and assume any risks associated with Your exercise of
142
+ permissions under this License.
143
+
144
+ 8. Limitation of Liability.
145
+
146
+ In no event and under no legal theory, whether in tort (including negligence),
147
+ contract, or otherwise, unless required by applicable law (such as deliberate
148
+ and grossly negligent acts) or agreed to in writing, shall any Contributor be
149
+ liable to You for damages, including any direct, indirect, special, incidental,
150
+ or consequential damages of any character arising as a result of this License or
151
+ out of the use or inability to use the Work (including but not limited to
152
+ damages for loss of goodwill, work stoppage, computer failure or malfunction, or
153
+ any and all other commercial damages or losses), even if such Contributor has
154
+ been advised of the possibility of such damages.
155
+
156
+ 9. Accepting Warranty or Additional Liability.
157
+
158
+ While redistributing the Work or Derivative Works thereof, You may choose to
159
+ offer, and charge a fee for, acceptance of support, warranty, indemnity, or
160
+ other liability obligations and/or rights consistent with this License. However,
161
+ in accepting such obligations, You may act only on Your own behalf and on Your
162
+ sole responsibility, not on behalf of any other Contributor, and only if You
163
+ agree to indemnify, defend, and hold each Contributor harmless for any liability
164
+ incurred by, or claims asserted against, such Contributor by reason of your
165
+ accepting any such warranty or additional liability.
166
+
167
+ END OF TERMS AND CONDITIONS
168
+
169
+ APPENDIX: How to apply the Apache License to your work
170
+
171
+ To apply the Apache License to your work, attach the following boilerplate
172
+ notice, with the fields enclosed by brackets "[]" replaced with your own
173
+ identifying information. (Don't include the brackets!) The text should be
174
+ enclosed in the appropriate comment syntax for the file format. We also
175
+ recommend that a file or class name and description of purpose be included on
176
+ the same "printed page" as the copyright notice for easier identification within
177
+ third-party archives.
178
+
179
+ Copyright [yyyy] [name of copyright owner]
180
+
181
+ Licensed under the Apache License, Version 2.0 (the "License");
182
+ you may not use this file except in compliance with the License.
183
+ You may obtain a copy of the License at
184
+
185
+ http://www.apache.org/licenses/LICENSE-2.0
186
+
187
+ Unless required by applicable law or agreed to in writing, software
188
+ distributed under the License is distributed on an "AS IS" BASIS,
189
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
190
+ See the License for the specific language governing permissions and
191
+ limitations under the License.
data/README.md CHANGED
@@ -7,10 +7,10 @@ It is based on [RGen](http://github.com/mthiede/rgen) and it supportes the conve
7
7
 
8
8
  There are different gems which transform source code in models of the code. Currently they are:
9
9
  * [html-lightmodels](http://github.com/ftomassetti/html-lightmodels)
10
- * [java-lightmodels](http://github.com/ftomassetti/java-lightmodels)
10
+ * [codemodels-java](http://github.com/ftomassetti/codemodels-java)
11
11
  * [js-lightmodels](http://github.com/ftomassetti/js-lightmodels)
12
12
  * [properties-lightmodels](http://github.com/ftomassetti/properties-lightmodels)
13
- * [ruby-lightmodels](http://github.com/ftomassetti/ruby-lightmodels)
13
+ * [codemodels-ruby](http://github.com/ftomassetti/codemodels-ruby)
14
14
  * [xml-lightmodels](http://github.com/ftomassetti/xml-lightmodels)
15
15
 
16
16
  Therefore it can be used to perform analysis on different languages.
data/codemodels.gemspec CHANGED
@@ -7,13 +7,12 @@ Gem::Specification.new do |s|
7
7
  s.platform = 'java'
8
8
  s.name = 'codemodels'
9
9
  s.version = CodeModels::VERSION
10
- s.date = '2013-08-27'
11
10
  s.summary = "Library to build models of code"
12
11
  s.description = "Library to build models of code of different languages in a uniform way."
13
12
  s.authors = ["Federico Tomassetti"]
14
13
  s.email = 'f.tomassetti@gmail.com'
15
14
  s.homepage = 'https://github.com/ftomassetti/codemodels'
16
- s.license = "APACHE 2"
15
+ s.license = "Apache v2"
17
16
 
18
17
  s.files = `git ls-files`.split($/)
19
18
  s.executables = s.files.grep(%r{^bin/}) { |f| File.basename(f) }
@@ -26,4 +25,4 @@ Gem::Specification.new do |s|
26
25
  s.add_development_dependency "bundler", "1.3.5"
27
26
  s.add_development_dependency "rake"
28
27
  s.add_development_dependency "simplecov"
29
- end
28
+ end
@@ -131,9 +131,5 @@ module InfoExtractionFunctionalities
131
131
 
132
132
  end
133
133
 
134
- class ::RGen::MetamodelBuilder::MMBase
135
- include InfoExtractionFunctionalities
136
- end
137
-
138
134
  end
139
135
  end
@@ -2,34 +2,6 @@ module CodeModels
2
2
 
3
3
  @@languages = []
4
4
 
5
- class Parser
6
-
7
- def parse_file(path)
8
- code = IO.read(path)
9
- parse_code(code)
10
- end
11
-
12
- end
13
-
14
- module CodeModelsReflectionInfo
15
- attr_accessor :source
16
- attr_accessor :language
17
- end
18
-
19
- class Position
20
- attr_accessor :line, :column
21
- end
22
-
23
- class SourceInfo
24
- attr_accessor :filename
25
- attr_accessor :begin_pos, :end_pos
26
-
27
- def to_code
28
- raise "Unimplemented"
29
- end
30
-
31
- end
32
-
33
5
  class Language
34
6
  attr_reader :name
35
7
  attr_reader :extensions
@@ -48,7 +20,9 @@ class Language
48
20
 
49
21
  end
50
22
 
23
+ # It avoids multiple registration of the same class
51
24
  def self.register_language(language)
25
+ return if @@languages.find {|l| l.is_a?(language.class)}
52
26
  @@languages << language
53
27
  end
54
28
 
@@ -56,9 +30,17 @@ def self.registered_languages
56
30
  @@languages
57
31
  end
58
32
 
33
+ class NoLanguageRegistered < Exception
34
+ attr_reader :path
35
+
36
+ def initialize(path)
37
+ @path = path
38
+ end
39
+ end
40
+
59
41
  def self.parse_file(path)
60
42
  l = @@languages.find {|l| l.can_parse?(path) }
61
- raise "I don't know how to parse #{path}. Languages known: #{@@languages.map(&:name)}" unless l
43
+ raise NoLanguageRegistered.new(path) unless l
62
44
  l.parser.parse_file(path)
63
45
  end
64
46
 
@@ -1,11 +1,19 @@
1
1
  require 'rgen/metamodel_builder'
2
- require 'codemodels/language'
2
+ require 'codemodels/source_info'
3
+ require 'codemodels/navigation'
4
+ require 'codemodels/info_extraction'
5
+ require 'codemodels/serialization'
3
6
 
4
7
  module CodeModels
5
8
 
6
- # All nodes should derive from this one
9
+ # All AST nodes built with CodeModels should derive from this one
7
10
  class CodeModelsAstNode < RGen::MetamodelBuilder::MMBase
8
- include CodeModelsReflectionInfo
11
+ include SourceInfoExtensions
12
+ include ForeignAstExtensions
13
+ include HostPositionExtensions
14
+ include NavigationExtensions
15
+ include InfoExtraction::InfoExtractionFunctionalities
16
+ include Serialization::SerializationFunctionalities
9
17
  end
10
18
 
11
19
  end
@@ -2,75 +2,71 @@ require 'codemodels/serialization'
2
2
 
3
3
  module CodeModels
4
4
 
5
+ class FilesSource
6
+
7
+ def initialize(dir,extension)
8
+ @dir = dir
9
+ @extension = extension
10
+ end
11
+
12
+ def each(&block)
13
+ Dir["#{@dir}/**/*.#{@extension}"].each do |f|
14
+ block.call(f)
15
+ end
16
+ end
17
+ end
18
+
19
+ class FileMapper
20
+
21
+ def initialize(src,dest,src_extension,dest_extension)
22
+ @src = src
23
+ @dest = dest
24
+ @src_ext = src_extension
25
+ @dest_ext = dest_extension
26
+ end
27
+
28
+ def map(f)
29
+ f_in_dest_dir = File.join(@dest,f.remove_prefix(@src))
30
+ f_in_dest_dir.remove_postfix(@src_ext)+@dest_ext
31
+ end
32
+
33
+ end
34
+
5
35
  module ModelBuilding
6
36
 
7
37
  class << self
8
38
  attr_accessor :verbose
39
+ attr_accessor :max_nesting
9
40
  end
10
41
 
42
+ @@verbose = false
43
+ @@max_nesting = 500
44
+
11
45
  def self.verbose_warn(msg)
12
46
  warn(msg) if verbose
13
47
  end
14
48
 
15
- def self.handle_serialized_models_in_dir(src,src_extension,error_handler=nil,model_handler,&model_creator)
16
- Dir["#{src}/**/*.#{src_extension}"].each do |fd|
17
- verbose_warn "== #{fd} =="
18
- handle_serialized_model_per_file(fd,error_handler,model_handler,&model_creator)
19
- end
20
- end
21
-
22
- def self.handle_models_in_dir(src,src_extension,error_handler=nil,model_handler,&model_creator)
23
- Dir["#{src}/**/*.#{src_extension}"].each do |fd|
24
- verbose_warn "== #{fd} =="
25
- handle_model_per_file(fd,error_handler,model_handler,&model_creator)
26
- end
27
- end
28
-
29
- def self.generate_models_in_dir(src,dest,src_extension,dest_extension,max_nesting=500,error_handler=nil,&model_creator)
49
+ def self.generate_models_in_dir(src,dest,src_extension,dest_extension,&model_creator)
30
50
  verbose_warn "== #{src} -> #{dest} =="
31
- Dir["#{src}/**/*.#{src_extension}"].each do |fd|
32
- if File.directory? fd
33
- basename = File.basename(fd)
34
- generate_models_in_dir("#{src}/#{basename}","#{dest}/#{basename}",src_extension,dest_extension,max_nesting,error_handler,&model_creator)
35
- else
36
- if File.extname(fd)==".#{src_extension}"
37
- translated_simple_name = "#{File.basename(fd, ".#{src_extension}")}.#{dest_extension}"
38
- translated_name = "#{dest}/#{translated_simple_name}"
39
- verbose_warn "* #{fd} --> #{translated_name}"
40
- generate_model_per_file(fd,translated_name,max_nesting,error_handler,&model_creator)
41
- end
42
- end
51
+ FilesSource.new(src,src_extension).each do |f|
52
+ dest_name = FileMapper.new(src,dest,src_extension,dest_extension).map(f)
53
+ verbose_warn "* #{f} --> #{dest_name}"
54
+ generate_model_per_file(f,dest_name,&model_creator)
43
55
  end
44
56
  end
45
57
 
46
- def self.handle_model_per_file(src,error_handler=nil,model_handler,&models_generator)
58
+ def self.handle_model_per_file(src,model_handler,&models_generator)
47
59
  verbose_warn "<Model from #{src}>"
48
- begin
49
- m = models_generator.call(src)
50
- model_handler.call(src,m)
51
- rescue => e
52
- if error_handler
53
- error_handler.call(src,e)
54
- else
55
- raise e
56
- end
57
- end
60
+
61
+ m = models_generator.call(src)
62
+ model_handler.call(src,m) if m
58
63
  end
59
64
 
60
- def self.generate_model_per_file(src,dest,max_nesting=500,error_handler=nil,&models_generator)
65
+ def self.generate_model_per_file(src,dest,&models_generator)
61
66
  if not File.exist? dest
62
- verbose_warn "<Model from #{src}>"
63
-
64
- begin
65
- m = models_generator.call(src)
66
- LightModels::Serialization.save_model(m,dest,max_nesting)
67
- rescue Exception => e
68
- if error_handler
69
- error_handler.call(src,e)
70
- else
71
- raise e
72
- end
73
- end
67
+ verbose_warn "<Model from #{src}>"
68
+ m = models_generator.call(src)
69
+ LightModels::Serialization.save_model(m,dest,@@max_nesting) if m
74
70
  else
75
71
  verbose_warn "skipping #{src} because #{dest} found"
76
72
  end
@@ -0,0 +1,121 @@
1
+ module CodeModels
2
+
3
+ module NavigationExtensions
4
+
5
+ def all_children(flag=nil)
6
+ also_foreign = (flag==:also_foreign)
7
+ arr = []
8
+ ecore = self.class.ecore
9
+ # Awful hack to forbid the same reference is visited twice when
10
+ # two references with the same name are found
11
+ already_used_references = []
12
+ ecore.eAllReferences.select {|r| r.containment}.each do |ref|
13
+ #raise "Too many features with name #{ref.name}. Count: #{features_by_name(ref.name).count}" if features_by_name(ref.name).count!=1
14
+ unless already_used_references.include?(ref.name)
15
+ res = self.send(ref.name.to_sym)
16
+ if ref.many
17
+ d = arr.count
18
+ res.each do |el|
19
+ arr << el unless res==nil
20
+ end
21
+ elsif res!=nil
22
+ d = arr.count
23
+ arr << res
24
+ end
25
+ already_used_references << ref.name
26
+ end
27
+ end
28
+ if also_foreign
29
+ arr.concat(self.foreign_asts)
30
+ end
31
+ arr
32
+ end
33
+
34
+ def all_children_deep(flag=nil)
35
+ arr = []
36
+ all_children(flag).each do |c|
37
+ arr << c
38
+ c.all_children_deep(flag).each do |cc|
39
+ arr << cc
40
+ end
41
+ end
42
+ arr
43
+ end
44
+
45
+ def traverse(flag=nil,&op)
46
+ op.call(self)
47
+ all_children_deep(flag).each do |c|
48
+ op.call(c)
49
+ end
50
+ end
51
+
52
+ def collect_values_with_count
53
+ values = Hash.new {|h,k| h[k]=0}
54
+ self.class.ecore.eAllAttributes.each do |a|
55
+ v = self.send(:"#{a.name}")
56
+ if v!=nil
57
+ if a.many
58
+ v.each {|el| values[el]+=1}
59
+ else
60
+ values[v]+=1
61
+ end
62
+ end
63
+ end
64
+ values
65
+ end
66
+
67
+ def collect_values_with_count_subtree(flag=nil)
68
+ values = collect_values_with_count
69
+ all_children_deep(flag).each do |c|
70
+ c.collect_values_with_count.each do |k,v|
71
+ values[k]+=v
72
+ end
73
+ end
74
+ values
75
+ end
76
+
77
+ def all_children_of_type(flag=nil,type)
78
+ all_children(flag).select {|c| c and c.is_a?(type)}
79
+ end
80
+
81
+ def all_children_deep_of_type(flag=nil,type)
82
+ all_children_deep(flag).select {|c| c and c.is_a?(type)}
83
+ end
84
+
85
+ def only_child_of_type(flag=nil,type)
86
+ selected = all_children_of_type(flag,type)
87
+ raise "Exactly one child of type #{type} expected, #{selected.count} found on #{self}" unless selected.count==1
88
+ selected[0]
89
+ end
90
+
91
+ def only_child_deep_of_type(flag=nil,type)
92
+ selected = all_children_deep_of_type(flag,type)
93
+ raise "Exactly one child of type #{type} expected, #{selected.count} found on #{self}" unless selected.count==1
94
+ selected[0]
95
+ end
96
+
97
+ def container(flag=nil)
98
+ also_foreign = (flag==:also_foreign)
99
+ return self.eContainer if self.eContainer
100
+ if also_foreign
101
+ self.foreign_container
102
+ else
103
+ nil
104
+ end
105
+ end
106
+
107
+ def all_children_also_foreign
108
+ all_children(:also_foreign)
109
+ end
110
+
111
+ def all_children_deep_also_foreign
112
+ all_children_deep(:also_foreign)
113
+ end
114
+
115
+ def traverse_also_foreign(&block)
116
+ traverse(:also_foreign,block)
117
+ end
118
+
119
+ end
120
+
121
+ end