nitro 0.28.0 → 0.29.0
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +382 -0
- data/ProjectInfo +4 -4
- data/README +1 -1
- data/doc/AUTHORS +15 -15
- data/doc/MIGRATION +13 -0
- data/doc/RELEASES +102 -0
- data/lib/glue/sweeper.rb +1 -1
- data/lib/nitro.rb +38 -9
- data/lib/nitro/adapter/acgi.rb +1 -3
- data/lib/nitro/adapter/cgi.rb +1 -1
- data/lib/nitro/adapter/fastcgi.rb +1 -3
- data/lib/nitro/adapter/mongrel.rb +8 -6
- data/lib/nitro/adapter/webrick.rb +1 -2
- data/lib/nitro/cgi.rb +1 -1
- data/lib/nitro/compiler.rb +21 -40
- data/lib/nitro/compiler/elements.rb +72 -32
- data/lib/nitro/compiler/errors.rb +92 -42
- data/lib/nitro/compiler/include.rb +47 -17
- data/lib/nitro/compiler/morphing.rb +1 -3
- data/lib/nitro/compiler/script.rb +2 -2
- data/lib/nitro/context.rb +36 -0
- data/lib/nitro/controller.rb +140 -31
- data/lib/nitro/dispatcher.rb +27 -28
- data/lib/nitro/element.rb +52 -15
- data/lib/nitro/flash.rb +44 -0
- data/lib/nitro/helper/buffer.rb +0 -2
- data/lib/nitro/helper/form.rb +2 -2
- data/lib/nitro/helper/form/controls.rb +14 -3
- data/lib/nitro/helper/pager.rb +1 -1
- data/lib/nitro/helper/table.rb +4 -3
- data/lib/nitro/helper/xml.rb +1 -1
- data/lib/nitro/part.rb +20 -0
- data/lib/nitro/render.rb +44 -5
- data/lib/nitro/router.rb +81 -0
- data/lib/nitro/scaffolding.rb +24 -23
- data/lib/nitro/server.rb +12 -1
- data/lib/nitro/server/runner.rb +12 -0
- data/lib/nitro/session.rb +3 -12
- data/lib/nitro/session/drb.rb +2 -5
- data/lib/nitro/session/file.rb +2 -2
- data/lib/nitro/session/memcached.rb +14 -0
- data/lib/nitro/session/memory.rb +3 -26
- data/lib/nitro/session/og.rb +1 -1
- data/lib/nitro/test/assertions.rb +1 -1
- data/lib/nitro/test/context.rb +8 -2
- data/lib/nitro/test/testcase.rb +16 -7
- data/proto/public/error.xhtml +58 -21
- data/proto/public/js/controls.js +60 -15
- data/proto/public/js/dragdrop.js +105 -16
- data/proto/public/js/effects.js +19 -12
- data/proto/public/js/scriptaculous.js +1 -1
- data/proto/public/js/slider.js +2 -2
- data/proto/public/js/unittest.js +29 -20
- data/proto/public/scaffold/edit.xhtml +1 -1
- data/proto/public/scaffold/index.xhtml +2 -2
- data/proto/public/scaffold/list.xhtml +2 -2
- data/proto/public/scaffold/new.xhtml +1 -1
- data/proto/public/scaffold/search.xhtml +1 -1
- data/src/part/admin/controller.rb +5 -5
- data/src/part/admin/template/index.xhtml +2 -2
- data/test/nitro/compiler/tc_compiler.rb +23 -0
- data/test/nitro/helper/tc_table.rb +35 -0
- data/test/nitro/tc_cgi.rb +1 -1
- data/test/nitro/tc_controller.rb +3 -3
- data/test/nitro/tc_controller_aspect.rb +2 -0
- data/test/nitro/tc_dispatcher.rb +10 -1
- data/test/nitro/tc_flash.rb +14 -0
- data/test/nitro/tc_router.rb +58 -0
- data/test/nitro/tc_session.rb +26 -9
- metadata +13 -12
- data/lib/nitro/routing.rb +0 -41
- data/test/nitro/caching/tc_stores.rb +0 -17
- data/test/nitro/tc_table.rb +0 -66
@@ -1,9 +1,11 @@
|
|
1
1
|
require 'rexml/document'
|
2
2
|
require 'rexml/streamlistener'
|
3
|
+
require 'facet/class/by_name'
|
3
4
|
|
4
5
|
require 'nitro/element'
|
5
6
|
require "glue/html"
|
6
7
|
|
8
|
+
|
7
9
|
module Nitro
|
8
10
|
|
9
11
|
# A compiler that handles the processing of Elements
|
@@ -16,8 +18,9 @@ class Elements # :nodoc: all
|
|
16
18
|
attr_accessor :buffer
|
17
19
|
attr_accessor :stack
|
18
20
|
|
19
|
-
def initialize
|
20
|
-
super
|
21
|
+
def initialize(compiler)
|
22
|
+
super()
|
23
|
+
@compiler = compiler
|
21
24
|
@buffer = ''
|
22
25
|
@stack = []
|
23
26
|
end
|
@@ -29,35 +32,20 @@ class Elements # :nodoc: all
|
|
29
32
|
# check if the name starts with the element prefix, or
|
30
33
|
# is capitalized.
|
31
34
|
|
32
|
-
if
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
unless klass.ancestors.include? Nitro::Element
|
39
|
-
if Element.auto_extend
|
40
|
-
klass.send(:include, Nitro::ElementMixin)
|
41
|
-
else
|
42
|
-
Logger.error "Invalid element class '#{name}', does not extend Nitro::Element"
|
43
|
-
end
|
44
|
-
end
|
45
|
-
else
|
46
|
-
Logger.error "The class of this element tag '#{name}' does not exist"
|
47
|
-
end
|
48
|
-
|
49
|
-
if klass and obj = klass.new
|
50
|
-
attributes.each do | k, v |
|
51
|
-
obj.instance_variable_set("@#{k}", v)
|
52
|
-
end
|
35
|
+
if klass = is_element?(name)
|
36
|
+
|
37
|
+
obj = klass.new
|
38
|
+
attributes.each do | k, v |
|
39
|
+
obj.instance_variable_set("@#{k}", v)
|
40
|
+
end
|
53
41
|
|
54
|
-
|
42
|
+
@stack.push [obj, @buffer, @parent]
|
55
43
|
|
56
|
-
|
57
|
-
|
44
|
+
@buffer = obj._text
|
45
|
+
@parent.add_child(obj) if @parent
|
58
46
|
|
59
|
-
|
60
|
-
|
47
|
+
@parent = obj
|
48
|
+
|
61
49
|
else # This is a static element.
|
62
50
|
attrs = []
|
63
51
|
|
@@ -74,14 +62,66 @@ class Elements # :nodoc: all
|
|
74
62
|
def tag_end(name)
|
75
63
|
# check if the name starts with the element prefix, or
|
76
64
|
# is capitalized.
|
77
|
-
if
|
78
|
-
name = name.split(':')[1].camelize if name =~ PREFIX_RE
|
65
|
+
if is_element? name
|
79
66
|
obj, @buffer, @parent = @stack.pop
|
80
67
|
@buffer << obj.render
|
81
68
|
else
|
82
69
|
@buffer << "</#{name}>"
|
83
70
|
end
|
84
71
|
end
|
72
|
+
|
73
|
+
|
74
|
+
# Check if a tag is a Nitro::Element. If found, it also tries to
|
75
|
+
# auto-extend the klass.
|
76
|
+
#
|
77
|
+
# returns the Element class if found
|
78
|
+
def is_element?(name)
|
79
|
+
# Doesn't support modulized classes
|
80
|
+
# name = name.demodulize
|
81
|
+
return false unless name =~ PREFIX_RE or name =~ CAPITALIZED_RE
|
82
|
+
|
83
|
+
name = name.gsub(/#{PREFIX_RE}:/,'').camelize if name =~ PREFIX_RE
|
84
|
+
|
85
|
+
# First try to use Nitro::Element::xxx then ::xxx
|
86
|
+
|
87
|
+
begin
|
88
|
+
klass = Class.by_name("Nitro::Element::#{name}")
|
89
|
+
rescue
|
90
|
+
end
|
91
|
+
|
92
|
+
# Look into the module the controller's module if any
|
93
|
+
|
94
|
+
begin
|
95
|
+
namespace = @compiler.controller.name
|
96
|
+
if /::/ =~ namespace
|
97
|
+
namespace = namespace.gsub( /::[a-zA-Z]+$/, "::#{name}" )
|
98
|
+
pp namespace
|
99
|
+
klass = Class.by_name(namespace)
|
100
|
+
end
|
101
|
+
rescue
|
102
|
+
end unless klass
|
103
|
+
|
104
|
+
# Look in the root module
|
105
|
+
|
106
|
+
begin
|
107
|
+
klass = Class.by_name(name)
|
108
|
+
rescue
|
109
|
+
end unless klass
|
110
|
+
|
111
|
+
return false unless klass.kind_of?( Class )
|
112
|
+
|
113
|
+
# Try to auto-extend
|
114
|
+
|
115
|
+
unless klass.ancestors.include? Nitro::Element
|
116
|
+
if Element.auto_extend
|
117
|
+
klass.send(:include, Nitro::ElementMixin)
|
118
|
+
else
|
119
|
+
return false
|
120
|
+
end
|
121
|
+
end
|
122
|
+
|
123
|
+
return klass
|
124
|
+
end
|
85
125
|
|
86
126
|
def text(str)
|
87
127
|
@buffer << str
|
@@ -92,7 +132,7 @@ class Elements # :nodoc: all
|
|
92
132
|
end
|
93
133
|
|
94
134
|
def comment(c)
|
95
|
-
unless Template.strip_xml_comments
|
135
|
+
unless Glue::Template.strip_xml_comments
|
96
136
|
@buffer << "<!--#{c}-->"
|
97
137
|
end
|
98
138
|
end
|
@@ -116,7 +156,7 @@ class Elements # :nodoc: all
|
|
116
156
|
#++
|
117
157
|
|
118
158
|
def transform(source, compiler)
|
119
|
-
listener = Listener.new
|
159
|
+
listener = Listener.new(compiler)
|
120
160
|
|
121
161
|
REXML::Document.parse_stream(source, listener)
|
122
162
|
|
@@ -1,65 +1,115 @@
|
|
1
1
|
require 'facet/ormsupport'
|
2
2
|
|
3
|
-
|
3
|
+
# This supports source code extraction from a full
|
4
|
+
# backtrace from any exception and also exposes class
|
5
|
+
# methods incase you wish to harness them from elsewhere.
|
6
|
+
#--
|
7
|
+
# TODO: Cache source code? Memory vs Speed - Is it worth
|
8
|
+
# it?
|
9
|
+
# FIXME: Don't change the Exception class, use NitroException
|
10
|
+
# or something.
|
11
|
+
#++
|
4
12
|
|
5
|
-
class
|
6
|
-
SOURCE_CODE_RADIUS = 5
|
13
|
+
class Exception
|
7
14
|
|
8
|
-
|
9
|
-
|
10
|
-
def initialize(source_code, filename, original_exception)
|
11
|
-
@source_code = source_code.split("\n")
|
12
|
-
@filename = filename
|
13
|
-
@original_exception = original_exception
|
14
|
-
end
|
15
|
+
# Radius of the source code to display arround the error line.
|
15
16
|
|
16
|
-
|
17
|
-
|
17
|
+
SOURCE_RADIUS = 5
|
18
|
+
|
19
|
+
class << self
|
18
20
|
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
21
|
+
# Extracts the filename and line from the given line (step)
|
22
|
+
# of the backtrace.
|
23
|
+
|
24
|
+
def file_and_line_no(step)
|
25
|
+
file = no = nil
|
26
|
+
if res = step.match(/^(.+):(\d+)$/)
|
27
|
+
file = res[1]
|
28
|
+
no = res[2]
|
29
|
+
elsif res = step.match(/^(.+):(\d+):in `.+'$/)
|
30
|
+
file = res[1]
|
31
|
+
no = res[2]
|
32
|
+
end
|
33
|
+
if file and no
|
34
|
+
[file,no.to_i]
|
35
|
+
else
|
36
|
+
nil
|
37
|
+
end
|
23
38
|
end
|
24
|
-
|
39
|
+
|
40
|
+
# Extract the source arround the error line.
|
41
|
+
|
42
|
+
def source_extract(step, indent = 0)
|
43
|
+
begin
|
44
|
+
file, no = file_and_line_no(step)
|
45
|
+
source = File.read(file)
|
46
|
+
|
47
|
+
if file =~ /\.xhtml$/
|
48
|
+
source = Compiler.new.transform_template(source)
|
49
|
+
no -= 2
|
50
|
+
end
|
51
|
+
|
52
|
+
source = source.split("\n")
|
53
|
+
start = (no-1) - SOURCE_RADIUS
|
54
|
+
finish = (no-1) + SOURCE_RADIUS
|
55
|
+
start = 0 if start < 0
|
56
|
+
finish = source.size if finish > source.size
|
25
57
|
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
58
|
+
number = start
|
59
|
+
extract = source[start..finish].collect do |line|
|
60
|
+
number += 1
|
61
|
+
line = line.gsub(/; @out << %\^/, ' ?>').gsub(/\^;/, '<?r ')
|
62
|
+
"#{' ' * indent}#{number}: #{line}"
|
63
|
+
end
|
30
64
|
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
line = line.gsub(/; @out << %\^/, ' ?>').gsub(/\^;/, '<?r ')
|
35
|
-
if number == line_number
|
36
|
-
"#{' ' * indent}#{number}: #{line}"
|
37
|
-
else
|
38
|
-
"#{' ' * indent}#{number}: #{line}"
|
65
|
+
return extract.join("\n")
|
66
|
+
rescue => ex
|
67
|
+
''
|
39
68
|
end
|
40
69
|
end
|
70
|
+
|
71
|
+
end
|
41
72
|
|
42
|
-
|
73
|
+
# Identifies the 'hot' index in the backtrace, where the error
|
74
|
+
# actually happened.
|
75
|
+
|
76
|
+
def hot_trace_index
|
77
|
+
index = 0
|
78
|
+
while backtrace[index] =~ /controller\.rb(.*)method_missing/
|
79
|
+
index += 1
|
80
|
+
end
|
81
|
+
return index
|
43
82
|
end
|
44
83
|
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
84
|
+
# Returns the 'hot' backtrace line.
|
85
|
+
|
86
|
+
def hot_trace_step
|
87
|
+
backtrace[hot_trace_index]
|
49
88
|
end
|
50
89
|
|
51
|
-
|
52
|
-
|
90
|
+
# Returns the 'hot' backtrace file.
|
91
|
+
|
92
|
+
def hot_file
|
93
|
+
file_and_line_no(hot_trace_step).first
|
53
94
|
end
|
54
95
|
|
55
|
-
def
|
56
|
-
|
96
|
+
def file_and_line_no(step)
|
97
|
+
self.class.file_and_line_no(step)
|
98
|
+
end
|
99
|
+
|
100
|
+
def source_extract(step = hot_trace_step, indent = 0)
|
101
|
+
self.class.source_extract(step,indent)
|
102
|
+
end
|
103
|
+
|
104
|
+
# Extract source for backtrace.
|
105
|
+
|
106
|
+
def source_for_backtrace(indent = 0)
|
107
|
+
backtrace.inject([]) do |sources, step|
|
108
|
+
sources << source_extract(indent,step)
|
109
|
+
end
|
57
110
|
end
|
58
|
-
end
|
59
111
|
|
60
|
-
class TemplateCompileError < ActionCompileError
|
61
|
-
end
|
62
|
-
|
63
112
|
end
|
64
113
|
|
65
114
|
# * George Moschovitis <gm@navel.gr>
|
115
|
+
# * Rob Pitt <rob@motionpath.co.uk>
|
@@ -7,27 +7,57 @@ module Nitro
|
|
7
7
|
# to use static includes in many many cases.
|
8
8
|
|
9
9
|
class StaticInclude
|
10
|
-
|
11
|
-
#
|
12
|
-
|
10
|
+
# Statically include sub-template files.
|
11
|
+
# The target file is included at compile time.
|
12
|
+
# If the given path is relative, the template_root stack of
|
13
|
+
# the controller is traversed. If an absolute path is provided,
|
14
|
+
# templates are searched only in Template.root
|
15
|
+
#
|
16
|
+
# gmosx: must be xformed before the <?r pi.
|
17
|
+
#
|
18
|
+
# === Example
|
19
|
+
# <?include href="root/myfile.sx" ?>
|
13
20
|
|
14
|
-
def self.transform(text)
|
15
|
-
|
16
|
-
|
17
|
-
#
|
18
|
-
# gmosx: must be xformed before the <?r pi.
|
19
|
-
#
|
20
|
-
# Example:
|
21
|
-
# <?include href="root/myfile.sx" ?>
|
22
|
-
|
21
|
+
def self.transform(text, compiler)
|
22
|
+
controller = compiler.controller
|
23
|
+
|
23
24
|
return text.gsub(/<\?include href=["|'](.*?)["|'](.*)\?>/) do |match|
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
25
|
+
found = false
|
26
|
+
|
27
|
+
if $1[0] == ?/
|
28
|
+
# Absolute path, search only in the application template
|
29
|
+
# root.
|
30
|
+
|
31
|
+
href = $1[1, 999999] # hack!!
|
32
|
+
template_stack = [ Glue::Template.root ]
|
33
|
+
else
|
34
|
+
# Relative path, traverse the template_root stack.
|
35
|
+
|
36
|
+
href = $1
|
37
|
+
template_stack = controller.instance_variable_get(:@template_root)
|
38
|
+
end
|
39
|
+
|
40
|
+
for template_root in template_stack
|
41
|
+
if File.exist?(filename = "#{template_root}/#{href}")
|
42
|
+
found = true
|
43
|
+
break
|
44
|
+
end
|
45
|
+
|
46
|
+
if File.exist?(filename = "#{template_root}/#{href}.xinc")
|
47
|
+
found = true
|
48
|
+
break
|
49
|
+
end
|
50
|
+
|
51
|
+
if File.exist?(filename = "#{template_root}/#{href}.xhtml")
|
52
|
+
found = true
|
53
|
+
break
|
29
54
|
end
|
30
55
|
end
|
56
|
+
|
57
|
+
unless found
|
58
|
+
raise "Cannot statically include '#{href}'"
|
59
|
+
end
|
60
|
+
|
31
61
|
itext = File.read(filename)
|
32
62
|
itext.gsub!(/<\?xml.*\?>/, '')
|
33
63
|
itext.gsub!(/<\/?root(.*?)>/m, ' ');
|
@@ -4,8 +4,6 @@ require 'rexml/streamlistener'
|
|
4
4
|
require 'facet/dictionary'
|
5
5
|
require 'facet/string/blank'
|
6
6
|
|
7
|
-
require 'glue/html'
|
8
|
-
|
9
7
|
module Nitro
|
10
8
|
|
11
9
|
# :section: A collection of standard morphers.
|
@@ -205,7 +203,7 @@ class Morphing
|
|
205
203
|
end
|
206
204
|
|
207
205
|
def comment(c)
|
208
|
-
unless Template.strip_xml_comments
|
206
|
+
unless Glue::Template.strip_xml_comments
|
209
207
|
@buffer << "<!--#{c}-->"
|
210
208
|
end
|
211
209
|
end
|
@@ -87,13 +87,13 @@ module ScriptCompiler
|
|
87
87
|
js_buffer = compiler.shared[:js_buffer]
|
88
88
|
|
89
89
|
if script or js_buffer
|
90
|
-
text.sub!(/<\/
|
90
|
+
text.sub!(/<\/body>/) do |match|
|
91
91
|
%{
|
92
92
|
<script type="text/javascript">
|
93
93
|
#{script}
|
94
94
|
#{js_buffer}
|
95
95
|
</script>
|
96
|
-
</
|
96
|
+
</body>
|
97
97
|
}
|
98
98
|
end
|
99
99
|
end
|
data/lib/nitro/context.rb
CHANGED
@@ -1,3 +1,5 @@
|
|
1
|
+
require 'forwardable'
|
2
|
+
|
1
3
|
require 'facet/kernel/assign_with'
|
2
4
|
|
3
5
|
require 'nitro/cgi'
|
@@ -21,6 +23,10 @@ class Context
|
|
21
23
|
include Response
|
22
24
|
include Render
|
23
25
|
|
26
|
+
# The cache/store used to back global variables.
|
27
|
+
|
28
|
+
setting :global_cache_class, :default => ($NITRO_GLOBAL_CACHE_CLASS || MemoryCache), :doc => 'The store used to back global variables'
|
29
|
+
|
24
30
|
# The configuration parameters.
|
25
31
|
|
26
32
|
attr_accessor :conf
|
@@ -60,6 +66,8 @@ class Context
|
|
60
66
|
if @session.has_key?(:FLASH)
|
61
67
|
@session[:FLASH].clean
|
62
68
|
end
|
69
|
+
|
70
|
+
# INVESTIGATE: is this needed?
|
63
71
|
@session.sync
|
64
72
|
end
|
65
73
|
end
|
@@ -85,6 +93,14 @@ class Context
|
|
85
93
|
@session || @session = Session.lookup(self)
|
86
94
|
end
|
87
95
|
|
96
|
+
# Access global variables. In a distributed server scenario,
|
97
|
+
# these variables can reside outside of the process.
|
98
|
+
|
99
|
+
def global
|
100
|
+
return $global
|
101
|
+
end
|
102
|
+
alias_method :application, :global
|
103
|
+
|
88
104
|
# Lookup the controller for this request.
|
89
105
|
#--
|
90
106
|
# FIXME: improve this! BUGGY
|
@@ -101,6 +117,14 @@ class Context
|
|
101
117
|
#
|
102
118
|
# * name
|
103
119
|
# * force_boolean
|
120
|
+
#
|
121
|
+
# === Example
|
122
|
+
#
|
123
|
+
# request.fill(User.new)
|
124
|
+
#
|
125
|
+
# Prefer to use the following form:
|
126
|
+
#
|
127
|
+
# User.new.assign_with(request)
|
104
128
|
|
105
129
|
def fill(obj, options = {})
|
106
130
|
Property.populate_object(obj, @params, options)
|
@@ -110,6 +134,18 @@ class Context
|
|
110
134
|
|
111
135
|
end
|
112
136
|
|
137
|
+
# Forwards to the context
|
138
|
+
#
|
139
|
+
# use in Nitro::Action
|
140
|
+
|
141
|
+
module ContextHelper
|
142
|
+
extend Forwardable
|
143
|
+
|
144
|
+
attr_accessor :context
|
145
|
+
|
146
|
+
def_delegators :@context, :controller, :request, :response, :session, :out
|
147
|
+
end
|
148
|
+
|
113
149
|
end
|
114
150
|
|
115
151
|
# * George Moschovitis <gm@navel.gr>
|