3 # The Potlatch module provides helper functions for potlatch and its communication with the server
5 # The AMF class is a set of helper functions for encoding and decoding AMF.
7 # Return two-byte integer
9 s.getbyte * 256 + s.getbyte
12 # Return four-byte long
14 ((s.getbyte * 256 + s.getbyte) * 256 + s.getbyte) * 256 + s.getbyte
17 # Return string with two-byte length
19 len = s.getbyte * 256 + s.getbyte
21 str.force_encoding("UTF-8") if str.respond_to?("force_encoding")
25 # Return eight-byte double-precision float
27 a = s.read(8).unpack('G') # G big-endian, E little-endian
31 # Return numeric array
44 while (key = getstring(s))
45 if (key == '') then break end
46 arr[key] = getvalue(s)
48 s.getbyte # skip the 9 'end of object' value
55 when 0 then return getdouble(s) # number
56 when 1 then return s.getbyte # boolean
57 when 2 then return getstring(s) # string
58 when 3 then return getobject(s) # object/hash
59 when 5 then return nil # null
60 when 6 then return nil # undefined
61 when 8 then s.read(4) # mixedArray
62 return getobject(s) # |
63 when 10 then return getarray(s) # array
64 else; return nil # error
68 # Envelope data into AMF writeable form
69 def self.putdata(index, n)
70 d = encodestring(index + "/onResult")
71 d += encodestring("null")
76 # Pack variables as AMF
77 def self.encodevalue(n)
80 a = 10.chr + encodelong(n.length)
88 a += encodestring(k.to_s) + encodevalue(v)
90 a + 0.chr + 0.chr + 9.chr
92 2.chr + encodestring(n)
93 when 'Bignum', 'Fixnum', 'Float'
94 0.chr + encodedouble(n)
98 0.chr + encodedouble(1)
100 0.chr + encodedouble(0)
102 Rails.logger.error("Unexpected Ruby type for AMF conversion: " + n.class.to_s)
106 # Encode string with two-byte length
107 def self.encodestring(n)
108 n = n.dup.force_encoding("ASCII-8BIT") if n.respond_to?("force_encoding")
109 a, b = n.size.divmod(256)
113 # Encode number as eight-byte double precision float
114 def self.encodedouble(n)
118 # Encode number as four-byte long
119 def self.encodelong(n)
124 # The Dispatcher class handles decoding a series of RPC calls
125 # from the request, dispatching them, and encoding the response
127 def initialize(request, &_block)
128 # Get stream for request data
129 @request = StringIO.new(request + 0.chr)
131 # Skip version indicator and client ID
135 AMF.getint(@request).times do # Read number of headers and loop
136 AMF.getstring(@request) # | skip name
137 req.getbyte # | skip boolean
138 AMF.getvalue(@request) # | skip value
141 # Capture the dispatch routine
146 # Read number of message bodies
147 bodies = AMF.getint(@request)
149 # Output response header
150 a, b = bodies.divmod(256)
151 yield 0.chr + 0.chr + 0.chr + 0.chr + a.chr + b.chr
154 bodies.times do # Read each body
155 name = AMF.getstring(@request) # | get message name
156 index = AMF.getstring(@request) # | get index in response sequence
157 bytes = AMF.getlong(@request) # | get total size in bytes
158 args = AMF.getvalue(@request) # | get response (probably an array)
160 result = @dispatch.call(name, *args)
162 yield AMF.putdata(index, result)
167 # The Potlatch class is a helper for Potlatch
171 # does: reads tag preset menus, colours, and autocomplete config files
172 # out: [0] presets, [1] presetmenus, [2] presetnames,
173 # [3] colours, [4] casing, [5] areas, [6] autotags
176 Rails.logger.info(" Message: getpresets")
180 presetmenus = {}; presetmenus['point'] = []; presetmenus['way'] = []; presetmenus['POI'] = []
181 presetnames = {}; presetnames['point'] = {}; presetnames['way'] = {}; presetnames['POI'] = {}
184 # StringIO.open(txt) do |file|
185 File.open("#{Rails.root}/config/potlatch/presets.txt") do |file|
186 file.each_line do|line|
188 if t =~ /(\w+)\/(\w+)/
191 presetmenus[presettype].push(presetcategory)
192 presetnames[presettype][presetcategory] = ["(no preset)"]
193 elsif t =~ /^([\w\s]+):\s?(.+)$/
195 presetnames[presettype][presetcategory].push(pre)
197 kv.split(',').each do|a|
198 if a =~ /^(.+)=(.*)$/ then presets[pre][$1] = $2 end
204 # Read colours/styling
205 colours = {}; casing = {}; areas = {}
206 File.open("#{Rails.root}/config/potlatch/colours.txt") do |file|
207 file.each_line do|line|
209 if t =~ /(\w+)\s+([^\s]+)\s+([^\s]+)\s+([^\s]+)/
211 if ($2 != '-') then colours[tag] = $2.hex end
212 if ($3 != '-') then casing[tag] = $3.hex end
213 if ($4 != '-') then areas[tag] = $4.hex end
218 # Read relations colours/styling
219 relcolours = {}; relalphas = {}; relwidths = {}
220 File.open("#{Rails.root}/config/potlatch/relation_colours.txt") do |file|
221 file.each_line do|line|
223 if t =~ /(\w+)\s+([^\s]+)\s+([^\s]+)\s+([^\s]+)/
225 if ($2 != '-') then relcolours[tag] = $2.hex end
226 if ($3 != '-') then relalphas[tag] = $3.to_i end
227 if ($4 != '-') then relwidths[tag] = $4.to_i end
233 icon_list = []; icon_tags = {}
234 File.open("#{Rails.root}/config/potlatch/icon_presets.txt") do |file|
235 file.each_line do|line|
236 (icon, tags) = line.chomp.split("\t")
238 icon_tags[icon] = Hash[*tags.scan(/([^;=]+)=([^;=]+)/).flatten]
244 autotags = {}; autotags['point'] = {}; autotags['way'] = {}; autotags['POI'] = {}
245 File.open("#{Rails.root}/config/potlatch/autocomplete.txt") do |file|
246 file.each_line do|line|
248 if t =~ /^([\w:]+)\/(\w+)\s+(.+)$/
249 tag = $1; type = $2; values = $3
250 if values == '-' then autotags[type][tag] = []
251 else autotags[type][tag] = values.split(',').sort.reverse end
256 [presets, presetmenus, presetnames, colours, casing, areas, autotags, relcolours, relalphas, relwidths, icon_list, {}, icon_tags]