Add route relation support for multi ref's
This commit is contained in:
		
							parent
							
								
									9a660e3c18
								
							
						
					
					
						commit
						5fd77aebb5
					
				| @ -8,6 +8,7 @@ Handlers = require("lib/way_handlers") | ||||
| Relations = require("lib/relations") | ||||
| find_access_tag = require("lib/access").find_access_tag | ||||
| limit = require("lib/maxspeed").limit | ||||
| Utils = require("lib/utils") | ||||
| 
 | ||||
| function setup() | ||||
|   return { | ||||
| @ -398,22 +399,55 @@ function process_way(profile, way, result, relations) | ||||
|   WayHandlers.run(profile, way, result, data, handlers, relations) | ||||
| 
 | ||||
|   -- now process relations data | ||||
|   local data = Relations.Merge(relations) | ||||
|   local direction = data["route_direction"] | ||||
|   if direction then | ||||
|     if result.ref then | ||||
|       if result.ref:len() > 0 then | ||||
|         result.ref = result.ref .. ' ' .. direction | ||||
|   local matched_refs = nil; | ||||
|   if result.ref then | ||||
|     matched_refs = Relations.MatchToRef(relations, result.ref) | ||||
| 
 | ||||
|     local ref = '' | ||||
|     for k ,v in pairs(matched_refs) do | ||||
|       if ref ~= '' then | ||||
|         ref = ref .. '; ' | ||||
|       end | ||||
| 
 | ||||
|       if v then | ||||
|         ref = ref .. k .. ' $' .. v | ||||
|       else | ||||
|         result.ref = direction | ||||
|         print(ref) | ||||
|         ref = ref .. k | ||||
|       end | ||||
|     end | ||||
| 
 | ||||
|     result.ref = ref | ||||
|     -- count = 0 | ||||
|     -- for k, v in pairs(matched_refs) do | ||||
|     --   count = count + 1 | ||||
|     -- end | ||||
|     -- if count > 1 then | ||||
|     --   print('---', way:id()) | ||||
|     --   for k, v in pairs(matched_refs) do | ||||
|     --     print(k, v) | ||||
|     --   end | ||||
|     -- end | ||||
|   end | ||||
| 
 | ||||
| 
 | ||||
| end | ||||
| 
 | ||||
| function process_relation(profile, relation, result) | ||||
|   local t = relation:get_value_by_key("type") | ||||
| 
 | ||||
|   function add_extra_data(m) | ||||
|     local name = relation:get_value_by_key("name") | ||||
|     if name then | ||||
|       result[m]["route_name"] = name | ||||
|     end | ||||
| 
 | ||||
|     local ref = relation:get_value_by_key("ref") | ||||
|     if ref then | ||||
|       result[m]["route_ref"] = ref | ||||
|     end | ||||
|   end | ||||
| 
 | ||||
|   if t == "route" then | ||||
|     local route = relation:get_value_by_key("route") | ||||
|     if route == "road" then | ||||
| @ -422,6 +456,7 @@ function process_relation(profile, relation, result) | ||||
|         local role = string.lower(m:role()) | ||||
|         if role == "north" or role == "south" or role == "west" or role == "east" then | ||||
|           result[m]["route_direction"] = role | ||||
|           add_extra_data(m) | ||||
|         end | ||||
|       end | ||||
|     end | ||||
| @ -433,6 +468,7 @@ function process_relation(profile, relation, result) | ||||
|         for _, m in ipairs(relation:members()) do | ||||
|           if m:role() == "forward" then | ||||
|             result[m]["route_direction"] = direction | ||||
|             add_extra_data(m) | ||||
|           end | ||||
|         end | ||||
|       end | ||||
|  | ||||
| @ -3,6 +3,8 @@ | ||||
| -- You can run a selection you find useful in your profile, | ||||
| -- or do you own processing if/when required. | ||||
| 
 | ||||
| Utils = require('lib/utils') | ||||
| 
 | ||||
| Relations = {} | ||||
| 
 | ||||
| function Relations.Merge(relations) | ||||
| @ -17,4 +19,82 @@ function Relations.Merge(relations) | ||||
|   return result | ||||
| end | ||||
| 
 | ||||
| -- match ref values to relations data | ||||
| function Relations.MatchToRef(relations, ref) | ||||
| 
 | ||||
|   function calculate_scores(refs, tag_value) | ||||
|     local tag_tokens = Set(Utils.tokenize_common(tag_value)) | ||||
|     local result = {} | ||||
|     for i, r in ipairs(refs) do | ||||
|       local ref_tokens = Utils.tokenize_common(r) | ||||
|       local score = 0 | ||||
| 
 | ||||
|       for _, t in ipairs(ref_tokens) do | ||||
|         if tag_tokens[t] then | ||||
|           if Utils.is_number(t) then | ||||
|             score = score + 2 | ||||
|           else | ||||
|             score = score + 1 | ||||
|           end | ||||
|         end | ||||
|       end | ||||
| 
 | ||||
|       result[r] = score | ||||
|     end | ||||
| 
 | ||||
|     return result | ||||
|   end | ||||
| 
 | ||||
|   local references = Utils.string_list_tokens(ref) | ||||
|   local result_match = {} | ||||
| 
 | ||||
|   for _, r in ipairs(references) do | ||||
|     result_match[r] = nil | ||||
|   end | ||||
| 
 | ||||
|   for _, rel in ipairs(relations) do | ||||
|     local name_scores = nil | ||||
|     local name_tokens = {} | ||||
|     local route_name = rel["route_name"] | ||||
|     if route_name then | ||||
|       name_scores = calculate_scores(references, route_name) | ||||
|     end | ||||
| 
 | ||||
|     local ref_scores = nil | ||||
|     local ref_tokens = {} | ||||
|     local route_ref = rel["route_ref"] | ||||
|     if route_ref then | ||||
|       ref_scores = calculate_scores(references, route_ref) | ||||
|     end | ||||
| 
 | ||||
|     -- merge scores | ||||
|     local direction = rel["route_direction"] | ||||
|     if direction then | ||||
|       local best_score = -1 | ||||
|       local best_ref = nil | ||||
|        | ||||
|       function find_best(scores) | ||||
|         if scores then | ||||
|           for k ,v in pairs(scores) do | ||||
|             if v > best_score then | ||||
|               best_ref = k | ||||
|               best_score = v | ||||
|             end | ||||
|           end | ||||
|         end | ||||
|       end | ||||
| 
 | ||||
|       find_best(name_scores) | ||||
|       find_best(ref_scores) | ||||
|        | ||||
|       if best_ref then | ||||
|         result_match[best_ref] = direction | ||||
|       end | ||||
|     end | ||||
| 
 | ||||
|   end | ||||
| 
 | ||||
|   return result_match | ||||
| end | ||||
| 
 | ||||
| return Relations | ||||
							
								
								
									
										43
									
								
								profiles/lib/utils.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								profiles/lib/utils.lua
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | ||||
| -- Profile functions to implement common algorithms of data processing | ||||
| -- | ||||
| -- You can run a selection you find useful in your profile, | ||||
| -- or do you own processing if/when required. | ||||
| 
 | ||||
| Utils = {} | ||||
| 
 | ||||
| -- split string 'a; b; c' to table with values ['a', 'b', 'c'] | ||||
| -- so it use just one separator ';' | ||||
| function Utils.string_list_tokens(str) | ||||
|   result = {} | ||||
|   local idx = 0 | ||||
|   for s in str.gmatch(str, "([^;]*)") do | ||||
|     if s ~= nil and s ~= '' then | ||||
|       idx = idx + 1 | ||||
|       table.insert(result, idx, (s:gsub("^%s*(.-)%s*$", "%1"))) | ||||
|     end | ||||
|   end | ||||
| 
 | ||||
|   return result | ||||
| end | ||||
| 
 | ||||
| -- same as Utils.StringListTokens, but with many possible separators: | ||||
| -- ',' | ';' | ' '| '(' | ')' | ||||
| function Utils.tokenize_common(str) | ||||
|   result = {} | ||||
|   local idx = 0 | ||||
|   for s in str.gmatch(str, "%S+") do | ||||
|     if s ~= nil and s ~= '' then | ||||
|       idx = idx + 1 | ||||
|       table.insert(result, idx, (s:gsub("^%s*(.-)%s*$", "%1"))) | ||||
|     end | ||||
|   end | ||||
| 
 | ||||
|   return result | ||||
| end | ||||
| 
 | ||||
| -- returns true, if string contains a number | ||||
| function Utils.is_number(str) | ||||
|   return (tonumber(str) ~= nil) | ||||
| end | ||||
| 
 | ||||
| return Utils | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user