amalgalite 1.8.0-x64-mingw-ucrt

Sign up to get free protection for your applications and to get access to all the features.
Files changed (108) hide show
  1. checksums.yaml +7 -0
  2. data/CONTRIBUTING.md +60 -0
  3. data/HISTORY.md +386 -0
  4. data/LICENSE +31 -0
  5. data/Manifest.txt +105 -0
  6. data/README.md +62 -0
  7. data/Rakefile +27 -0
  8. data/TODO.md +57 -0
  9. data/bin/amalgalite-pack +147 -0
  10. data/examples/a.rb +9 -0
  11. data/examples/blob.rb +88 -0
  12. data/examples/bootstrap.rb +36 -0
  13. data/examples/define_aggregate.rb +75 -0
  14. data/examples/define_function.rb +104 -0
  15. data/examples/fts5.rb +152 -0
  16. data/examples/gem-db.rb +94 -0
  17. data/examples/require_me.rb +11 -0
  18. data/examples/requires.rb +42 -0
  19. data/examples/schema-info.rb +34 -0
  20. data/ext/amalgalite/c/amalgalite.c +355 -0
  21. data/ext/amalgalite/c/amalgalite.h +151 -0
  22. data/ext/amalgalite/c/amalgalite_blob.c +240 -0
  23. data/ext/amalgalite/c/amalgalite_constants.c +1432 -0
  24. data/ext/amalgalite/c/amalgalite_database.c +1188 -0
  25. data/ext/amalgalite/c/amalgalite_requires_bootstrap.c +282 -0
  26. data/ext/amalgalite/c/amalgalite_statement.c +649 -0
  27. data/ext/amalgalite/c/extconf.rb +71 -0
  28. data/ext/amalgalite/c/gen_constants.rb +353 -0
  29. data/ext/amalgalite/c/notes.txt +134 -0
  30. data/ext/amalgalite/c/sqlite3.c +243616 -0
  31. data/ext/amalgalite/c/sqlite3.h +12894 -0
  32. data/ext/amalgalite/c/sqlite3_options.h +4 -0
  33. data/ext/amalgalite/c/sqlite3ext.h +705 -0
  34. data/lib/amalgalite/3.1/amalgalite.so +0 -0
  35. data/lib/amalgalite/aggregate.rb +73 -0
  36. data/lib/amalgalite/blob.rb +186 -0
  37. data/lib/amalgalite/boolean.rb +42 -0
  38. data/lib/amalgalite/busy_timeout.rb +47 -0
  39. data/lib/amalgalite/column.rb +99 -0
  40. data/lib/amalgalite/core_ext/kernel/require.rb +21 -0
  41. data/lib/amalgalite/csv_table_importer.rb +75 -0
  42. data/lib/amalgalite/database.rb +933 -0
  43. data/lib/amalgalite/function.rb +61 -0
  44. data/lib/amalgalite/index.rb +43 -0
  45. data/lib/amalgalite/memory_database.rb +15 -0
  46. data/lib/amalgalite/packer.rb +231 -0
  47. data/lib/amalgalite/paths.rb +80 -0
  48. data/lib/amalgalite/profile_tap.rb +131 -0
  49. data/lib/amalgalite/progress_handler.rb +21 -0
  50. data/lib/amalgalite/requires.rb +151 -0
  51. data/lib/amalgalite/schema.rb +225 -0
  52. data/lib/amalgalite/sqlite3/constants.rb +95 -0
  53. data/lib/amalgalite/sqlite3/database/function.rb +48 -0
  54. data/lib/amalgalite/sqlite3/database/status.rb +68 -0
  55. data/lib/amalgalite/sqlite3/status.rb +60 -0
  56. data/lib/amalgalite/sqlite3/version.rb +55 -0
  57. data/lib/amalgalite/sqlite3.rb +6 -0
  58. data/lib/amalgalite/statement.rb +421 -0
  59. data/lib/amalgalite/table.rb +91 -0
  60. data/lib/amalgalite/taps/console.rb +27 -0
  61. data/lib/amalgalite/taps/io.rb +74 -0
  62. data/lib/amalgalite/taps.rb +2 -0
  63. data/lib/amalgalite/trace_tap.rb +35 -0
  64. data/lib/amalgalite/type_map.rb +63 -0
  65. data/lib/amalgalite/type_maps/default_map.rb +166 -0
  66. data/lib/amalgalite/type_maps/storage_map.rb +38 -0
  67. data/lib/amalgalite/type_maps/text_map.rb +21 -0
  68. data/lib/amalgalite/version.rb +8 -0
  69. data/lib/amalgalite/view.rb +26 -0
  70. data/lib/amalgalite.rb +51 -0
  71. data/spec/aggregate_spec.rb +158 -0
  72. data/spec/amalgalite_spec.rb +4 -0
  73. data/spec/blob_spec.rb +78 -0
  74. data/spec/boolean_spec.rb +24 -0
  75. data/spec/busy_handler.rb +157 -0
  76. data/spec/data/iso-3166-country.txt +242 -0
  77. data/spec/data/iso-3166-schema.sql +22 -0
  78. data/spec/data/iso-3166-subcountry.txt +3995 -0
  79. data/spec/data/make-iso-db.sh +12 -0
  80. data/spec/database_spec.rb +505 -0
  81. data/spec/default_map_spec.rb +92 -0
  82. data/spec/function_spec.rb +78 -0
  83. data/spec/integeration_spec.rb +97 -0
  84. data/spec/iso_3166_database.rb +58 -0
  85. data/spec/json_spec.rb +24 -0
  86. data/spec/packer_spec.rb +60 -0
  87. data/spec/paths_spec.rb +28 -0
  88. data/spec/progress_handler_spec.rb +91 -0
  89. data/spec/requires_spec.rb +54 -0
  90. data/spec/rtree_spec.rb +66 -0
  91. data/spec/schema_spec.rb +131 -0
  92. data/spec/spec_helper.rb +48 -0
  93. data/spec/sqlite3/constants_spec.rb +108 -0
  94. data/spec/sqlite3/database_status_spec.rb +36 -0
  95. data/spec/sqlite3/status_spec.rb +22 -0
  96. data/spec/sqlite3/version_spec.rb +28 -0
  97. data/spec/sqlite3_spec.rb +53 -0
  98. data/spec/statement_spec.rb +168 -0
  99. data/spec/storage_map_spec.rb +38 -0
  100. data/spec/tap_spec.rb +57 -0
  101. data/spec/text_map_spec.rb +20 -0
  102. data/spec/type_map_spec.rb +14 -0
  103. data/spec/version_spec.rb +8 -0
  104. data/tasks/custom.rake +101 -0
  105. data/tasks/default.rake +244 -0
  106. data/tasks/extension.rake +28 -0
  107. data/tasks/this.rb +208 -0
  108. metadata +325 -0
@@ -0,0 +1,244 @@
1
+ # vim: syntax=ruby
2
+ require 'rake/clean'
3
+ require 'digest'
4
+ #------------------------------------------------------------------------------
5
+ # If you want to Develop on this project just run 'rake develop' and you'll
6
+ # have all you need to get going. If you want to use bundler for development,
7
+ # then run 'rake develop:using_bundler'
8
+ #------------------------------------------------------------------------------
9
+ namespace :develop do
10
+
11
+ # Install all the development and runtime dependencies of this gem using the
12
+ # gemspec.
13
+ task :default => 'Gemfile' do
14
+ require 'rubygems/dependency_installer'
15
+ installer = ::Gem::DependencyInstaller.new
16
+ puts "Installing bundler..."
17
+ installer.install 'bundler'
18
+ sh 'bundle install'
19
+ puts "\n\nNow run 'rake test'"
20
+ end
21
+
22
+ # Create a Gemfile that just references the gemspec
23
+ file 'Gemfile' => :gemspec do
24
+ File.open( "Gemfile", "w+" ) do |f|
25
+ f.puts "# DO NOT EDIT - This file is automatically generated"
26
+ f.puts "# Make changes to Manifest.txt and/or Rakefile and regenerate"
27
+ f.puts 'source "https://rubygems.org"'
28
+ f.puts 'gemspec'
29
+ end
30
+ end
31
+ end
32
+ desc "Bootstrap development"
33
+ task :develop => "develop:default"
34
+
35
+ #------------------------------------------------------------------------------
36
+ # RSpec - standard TestTask
37
+ #------------------------------------------------------------------------------
38
+ begin
39
+ require 'rspec/core/rake_task'
40
+ RSpec::Core::RakeTask.new( :test ) do |t|
41
+ t.ruby_opts = %w[ -w ]
42
+ t.rspec_opts = %w[
43
+ --color
44
+ --format documentation
45
+ --format RspecJunitFormatter --out tmp/report.xml
46
+ ]
47
+ end
48
+ task :test_requirements
49
+ task :test => :test_requirements
50
+ task :default => :test
51
+ rescue LoadError
52
+ This.task_warning( 'test' )
53
+ end
54
+
55
+ #------------------------------------------------------------------------------
56
+ # RDoc - standard rdoc rake task, although we must make sure to use a more
57
+ # recent version of rdoc since it is the one that has 'tomdoc' markup
58
+ #------------------------------------------------------------------------------
59
+ begin
60
+ gem 'rdoc' # otherwise we get the wrong task from stdlib
61
+ require 'rdoc/task'
62
+ RDoc::Task.new do |t|
63
+ t.markup = 'tomdoc'
64
+ t.rdoc_dir = 'doc'
65
+ t.main = 'README.md'
66
+ t.title = "#{This.name} #{This.version}"
67
+ t.rdoc_files.include( FileList['*.{rdoc,md,txt}'], FileList['ext/**/*.c'],
68
+ FileList['lib/**/*.rb'] )
69
+ end
70
+ rescue StandardError, LoadError
71
+ This.task_warning( 'rdoc' )
72
+ end
73
+
74
+ #------------------------------------------------------------------------------
75
+ # Coverage - optional code coverage, rcov for 1.8 and simplecov for 1.9, so
76
+ # for the moment only rcov is listed.
77
+ #------------------------------------------------------------------------------
78
+ begin
79
+ require 'simplecov'
80
+ desc 'Run tests with code coverage'
81
+ task :coverage do
82
+ ENV['COVERAGE'] = 'true'
83
+ Rake::Task[:test].execute
84
+ end
85
+ CLOBBER << 'coverage' if File.directory?( 'coverage' )
86
+ rescue LoadError
87
+ This.task_warning( 'simplecov' )
88
+ end
89
+
90
+ #------------------------------------------------------------------------------
91
+ # Manifest - We want an explicit list of thos files that are to be packaged in
92
+ # the gem. Most of this is from Hoe.
93
+ #------------------------------------------------------------------------------
94
+ namespace 'manifest' do
95
+ desc "Check the manifest"
96
+ task :check => :clean do
97
+ files = FileList["**/*", ".*"].exclude( This.exclude_from_manifest ).to_a.sort
98
+ files = files.select{ |f| File.file?( f ) }
99
+
100
+ tmp = "Manifest.tmp"
101
+ File.open( tmp, 'w' ) do |f|
102
+ f.puts files.join("\n")
103
+ end
104
+
105
+ begin
106
+ sh "diff -du Manifest.txt #{tmp}"
107
+ ensure
108
+ rm tmp
109
+ end
110
+ puts "Manifest looks good"
111
+ end
112
+
113
+ desc "Generate the manifest"
114
+ task :generate => :clean do
115
+ files = %x[ git ls-files ].split("\n").sort
116
+ files.reject! { |f| f =~ This.exclude_from_manifest }
117
+ File.open( "Manifest.txt", "w" ) do |f|
118
+ f.puts files.join("\n")
119
+ end
120
+ end
121
+ end
122
+
123
+ #------------------------------------------------------------------------------
124
+ # Fixme - look for fixmes and report them
125
+ #------------------------------------------------------------------------------
126
+ namespace :fixme do
127
+ task :default => 'manifest:check' do
128
+ This.manifest.each do |file|
129
+ next if file == __FILE__
130
+ next unless file =~ %r/(txt|rb|md|rdoc|css|html|xml|css)\Z/
131
+ puts "FIXME: Rename #{file}" if file =~ /fixme/i
132
+ IO.readlines( file ).each_with_index do |line, idx|
133
+ prefix = "FIXME: #{file}:#{idx+1}".ljust(42)
134
+ puts "#{prefix} => #{line.strip}" if line =~ /fixme/i
135
+ end
136
+ end
137
+ end
138
+
139
+ def fixme_project_root
140
+ This.project_path( '../fixme' )
141
+ end
142
+
143
+ def fixme_project_path( subtree )
144
+ fixme_project_root.join( subtree )
145
+ end
146
+
147
+ def local_fixme_files
148
+ This.manifest.select { |p| p =~ %r|^tasks/| }
149
+ end
150
+
151
+ def outdated_fixme_files
152
+ local_fixme_files.select do |local|
153
+ upstream = fixme_project_path( local )
154
+ upstream.exist? &&
155
+ ( Digest::SHA256.file( local ) != Digest::SHA256.file( upstream ) )
156
+ end
157
+ end
158
+
159
+ def fixme_up_to_date?
160
+ outdated_fixme_files.empty?
161
+ end
162
+
163
+ desc "See if the fixme tools are outdated"
164
+ task :outdated do
165
+ if fixme_up_to_date? then
166
+ puts "Fixme files are up to date."
167
+ else
168
+ outdated_fixme_files.each do |f|
169
+ puts "#{f} is outdated"
170
+ end
171
+ end
172
+ end
173
+
174
+ desc "Update outdated fixme files"
175
+ task :update do
176
+ if fixme_up_to_date? then
177
+ puts "Fixme files are already up to date."
178
+ else
179
+ puts "Updating fixme files:"
180
+ outdated_fixme_files.each do |local|
181
+ upstream = fixme_project_path( local )
182
+ puts " * #{local}"
183
+ FileUtils.cp( upstream, local )
184
+ end
185
+ puts "Use your git commands as appropriate."
186
+ end
187
+ end
188
+ end
189
+ desc "Look for fixmes and report them"
190
+ task :fixme => "fixme:default"
191
+
192
+ #------------------------------------------------------------------------------
193
+ # Gem Specification
194
+ #------------------------------------------------------------------------------
195
+ # Really this is only here to support those who use bundler
196
+ desc "Build the #{This.name}.gemspec file"
197
+ task :gemspec do
198
+ File.open( This.gemspec_file, "wb+" ) do |f|
199
+ f.puts "# DO NOT EDIT - This file is automatically generated"
200
+ f.puts "# Make changes to Manifest.txt and/or Rakefile and regenerate"
201
+ f.write This.platform_gemspec.to_ruby
202
+ end
203
+ end
204
+
205
+ # .rbc files from ruby 2.0
206
+ CLOBBER << "**/*.rbc"
207
+
208
+ # The standard gem packaging task, everyone has it.
209
+ require 'rubygems/package_task'
210
+ ::Gem::PackageTask.new( This.platform_gemspec ) do
211
+ # nothing
212
+ end
213
+
214
+ #------------------------------------------------------------------------------
215
+ # Release - the steps we go through to do a final release, this is pulled from
216
+ # a compbination of mojombo's rakegem, hoe and hoe-git
217
+ #
218
+ # 1) make sure we are on the main branch
219
+ # 2) make sure there are no uncommitted items
220
+ # 3) check the manifest and make sure all looks good
221
+ # 4) build the gem
222
+ # 5) do an empty commit to have the commit message of the version
223
+ # 6) tag that commit as the version
224
+ # 7) push main
225
+ # 8) push the tag
226
+ # 7) pus the gem
227
+ #------------------------------------------------------------------------------
228
+ task :release_check do
229
+ unless `git branch` =~ /^\* main/
230
+ abort "You must be on the main branch to release!"
231
+ end
232
+ unless `git status` =~ /^nothing to commit/m
233
+ abort "Nope, sorry, you have unfinished business"
234
+ end
235
+ end
236
+
237
+ desc "Create tag v#{This.version}, build and push #{This.platform_gemspec.full_name} to rubygems.org"
238
+ task :release => [ :release_check, 'manifest:check', :gem ] do
239
+ sh "git commit --allow-empty -a -m 'Release #{This.version}'"
240
+ sh "git tag -a -m 'v#{This.version}' v#{This.version}"
241
+ sh "git push origin main"
242
+ sh "git push origin v#{This.version}"
243
+ sh "gem push pkg/#{This.platform_gemspec.full_name}.gem"
244
+ end
@@ -0,0 +1,28 @@
1
+ # To be used if the gem has extensions.
2
+ # If this task set is inclueded then you will need to also have
3
+ #
4
+ # spec.add_development_dependency( 'rake-compiler', '~> 0.8.1' )
5
+ #
6
+ # in your top level rakefile
7
+ begin
8
+ require 'rake/extensiontask'
9
+ Rake::ExtensionTask.new( This.name ) do |ext|
10
+ ext.ext_dir = File.join( 'ext', This.name, "c" )
11
+ ext.lib_dir = File.join( 'lib', This.name )
12
+ ext.gem_spec = This.ruby_gemspec
13
+
14
+ ext.cross_compile = true # enable cross compilation (requires cross compile toolchain)
15
+ ext.cross_platform = %w[
16
+ x86-mingw32
17
+ x64-mingw-ucrt
18
+ x64-mingw32
19
+ ]
20
+ end
21
+
22
+ task :test_requirements => :compile
23
+ rescue LoadError
24
+ This.task_warning( 'extension' )
25
+ end
26
+
27
+ CLOBBER << "lib/**/*.{jar,so,bundle}"
28
+ CLOBBER << "lib/#{This.name}/{1,2,3}.*/"
data/tasks/this.rb ADDED
@@ -0,0 +1,208 @@
1
+ require 'pathname'
2
+
3
+ # Public: A Class containing all the metadata and utilities needed to manage a
4
+ # ruby project.
5
+ class ThisProject
6
+ # The name of this project
7
+ attr_accessor :name
8
+
9
+ # The author's name
10
+ attr_accessor :author
11
+
12
+ # The email address of the author(s)
13
+ attr_accessor :email
14
+
15
+ # The homepage of this project
16
+ attr_accessor :homepage
17
+
18
+ # The regex of files to exclude from the manifest
19
+ attr_accessor :exclude_from_manifest
20
+
21
+ # The hash of Gem::Specifications keyed' by platform
22
+ attr_accessor :gemspecs
23
+
24
+ # Public: Initialize ThisProject
25
+ #
26
+ # Yields self
27
+ def initialize(&block)
28
+ @exclude_from_manifest = Regexp.union(/\.(git|DS_Store|semaphore)/,
29
+ /^(doc|coverage|pkg|tmp|Gemfile(\.lock)?)/,
30
+ /^[^\/]+\.gemspec/,
31
+ /\.(swp|jar|bundle|so|rvmrc|travis.yml|byebug_history|fossa.yml|ruby-version)$/,
32
+ /~$/)
33
+ @gemspecs = Hash.new
34
+ yield self if block_given?
35
+ end
36
+
37
+ # Public: return the version of ThisProject
38
+ #
39
+ # Search the ruby files in the project looking for the one that has the
40
+ # version string in it. This does not eval any code in the project, it parses
41
+ # the source code looking for the string.
42
+ #
43
+ # Returns a String version
44
+ def version
45
+ [ "lib/#{ name }.rb", "lib/#{ name }/version.rb" ].each do |v|
46
+ path = project_path( v )
47
+ line = path.read[/^\s*VERSION\s*=\s*.*/]
48
+ if line then
49
+ return line.match(/.*VERSION\s*=\s*['"](.*)['"]/)[1]
50
+ end
51
+ end
52
+ end
53
+
54
+ # Internal: Return a section of an RDoc file with the given section name
55
+ #
56
+ # path - the relative path in the project of the file to parse
57
+ # section_name - the section out of the file from which to parse data
58
+ #
59
+ # Retuns the text of the section as an array of paragrphs.
60
+ def section_of( file, section_name )
61
+ re = /^[=#]+ (.*)$/
62
+ sectional = project_path( file )
63
+ parts = sectional.read.split( re )[1..-1]
64
+ parts.map! { |p| p.strip }
65
+
66
+ sections = Hash.new
67
+ Hash[*parts].each do |k,v|
68
+ sections[k] = v.split("\n\n")
69
+ end
70
+ return sections[section_name]
71
+ end
72
+
73
+ # Internal: print out a warning about the give task
74
+ def task_warning( task )
75
+ warn "WARNING: '#{task}' tasks are not defined. Please run 'rake develop'"
76
+ end
77
+
78
+ # Internal: Return the full path to the file that is relative to the project
79
+ # root.
80
+ #
81
+ # path - the relative path of the file from the project root
82
+ #
83
+ # Returns the Pathname of the file
84
+ def project_path( *relative_path )
85
+ project_root.join( *relative_path )
86
+ end
87
+
88
+ # Internal: The absolute path of this file
89
+ #
90
+ # Returns the Pathname of this file.
91
+ def this_file_path
92
+ Pathname.new( __FILE__ ).expand_path
93
+ end
94
+
95
+ # Internal: The root directory of this project
96
+ #
97
+ # This is defined as being the directory that is in the path of this project
98
+ # that has the first Rakefile
99
+ #
100
+ # Returns the Pathname of the directory
101
+ def project_root
102
+ this_file_path.ascend do |p|
103
+ rakefile = p.join( 'Rakefile' )
104
+ return p if rakefile.exist?
105
+ end
106
+ end
107
+
108
+ # Internal: Returns the contents of the Manifest.txt file as an array
109
+ #
110
+ # Returns an Array of strings
111
+ def manifest
112
+ manifest_file = project_path( "Manifest.txt" )
113
+ abort "You need a Manifest.txt" unless manifest_file.readable?
114
+ manifest_file.readlines.map { |l| l.strip }
115
+ end
116
+
117
+ # Internal: Return the files that define the extensions
118
+ #
119
+ # Returns an Array
120
+ def extension_conf_files
121
+ manifest.grep( /extconf.rb\Z/ )
122
+ end
123
+
124
+ # Internal: Returns the gemspace associated with the current ruby platform
125
+ def platform_gemspec
126
+ gemspecs.fetch(platform) { This.ruby_gemspec }
127
+ end
128
+
129
+ def core_gemspec
130
+ Gem::Specification.new do |spec|
131
+ spec.name = name
132
+ spec.version = version
133
+ spec.author = author
134
+ spec.email = email
135
+ spec.homepage = homepage
136
+
137
+ spec.summary = summary
138
+ spec.description = summary
139
+ spec.license = license
140
+
141
+ spec.files = manifest
142
+ spec.executables = spec.files.grep(/^bin/) { |f| File.basename(f) }
143
+ spec.test_files = spec.files.grep(/^spec/)
144
+
145
+ spec.extra_rdoc_files += spec.files.grep(/(txt|rdoc|md)$/)
146
+ spec.rdoc_options = [ "--main" , 'README.md',
147
+ "--markup", "tomdoc" ]
148
+
149
+ spec.required_ruby_version = '>= 2.2.2'
150
+ end
151
+ end
152
+
153
+ # Internal: Return the gemspec for the ruby platform
154
+ def ruby_gemspec( core = core_gemspec, &block )
155
+ yielding_gemspec( 'ruby', core, &block )
156
+ end
157
+
158
+ # Internal: Return the gemspec for the jruby platform
159
+ def java_gemspec( core = core_gemspec, &block )
160
+ yielding_gemspec( 'java', core, &block )
161
+ end
162
+
163
+ # Internal: give an initial spec and a key, create a new gemspec based off of
164
+ # it.
165
+ #
166
+ # This will force the new gemspecs 'platform' to be that of the key, since the
167
+ # only reason you would have multiple gemspecs at this point is to deal with
168
+ # different platforms.
169
+ def yielding_gemspec( key, core )
170
+ spec = gemspecs[key] ||= core.dup
171
+ spec.platform = key
172
+ yield spec if block_given?
173
+ return spec
174
+ end
175
+
176
+ # Internal: Return the platform of ThisProject at the current moment in time.
177
+ def platform
178
+ (RUBY_PLATFORM == "java") ? 'java' : Gem::Platform::RUBY
179
+ end
180
+
181
+ # Internal: Return the DESCRIPTION section of the README.rdoc file
182
+ def description_section
183
+ section_of( 'README.md', 'DESCRIPTION')
184
+ end
185
+
186
+ # Internal: Return the summary text from the README
187
+ def summary
188
+ description_section.first.gsub(/\s+/, ' ')
189
+ end
190
+
191
+ # Internal: Return the full description text from the README
192
+ def description
193
+ description_section.join(" ").tr("\n", ' ').gsub(/[{}]/,'').gsub(/\[[^\]]+\]/,'') # strip rdoc
194
+ end
195
+
196
+ def license
197
+ license_file = project_path("LICENSE")
198
+ line = license_file.readlines.first
199
+ line.split(/\s+/).first
200
+ end
201
+
202
+ # Internal: The path to the gemspec file
203
+ def gemspec_file
204
+ project_path( "#{ name }.gemspec" )
205
+ end
206
+ end
207
+
208
+ This = ThisProject.new