atlas_engine 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (298) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +123 -0
  3. data/Rakefile +20 -0
  4. data/app/assets/config/atlas_engine_manifest.js +3 -0
  5. data/app/assets/stylesheets/atlas_engine/application.css +15 -0
  6. data/app/concerns/atlas_engine/handles_blob.rb +26 -0
  7. data/app/concerns/atlas_engine/handles_interruption.rb +22 -0
  8. data/app/controllers/atlas_engine/application_controller.rb +7 -0
  9. data/app/controllers/atlas_engine/connectivity_controller.rb +21 -0
  10. data/app/controllers/atlas_engine/country_imports_controller.rb +73 -0
  11. data/app/controllers/atlas_engine/graphql_controller.rb +59 -0
  12. data/app/countries/atlas_engine/ar/country_profile.yml +9 -0
  13. data/app/countries/atlas_engine/at/address_importer/corrections/open_address/city_corrector.rb +23 -0
  14. data/app/countries/atlas_engine/at/country_profile.yml +24 -0
  15. data/app/countries/atlas_engine/at/index_configuration.yml +63 -0
  16. data/app/countries/atlas_engine/at/synonyms.yml +6 -0
  17. data/app/countries/atlas_engine/at/validation_transcriber/address_parser.rb +58 -0
  18. data/app/countries/atlas_engine/au/address_importer/open_address/filter.rb +26 -0
  19. data/app/countries/atlas_engine/au/address_importer/open_address/mapper.rb +41 -0
  20. data/app/countries/atlas_engine/au/country_profile.yml +13 -0
  21. data/app/countries/atlas_engine/au/synonyms.yml +209 -0
  22. data/app/countries/atlas_engine/au/validation_transcriber/address_parser.rb +121 -0
  23. data/app/countries/atlas_engine/be/country_profile.yml +12 -0
  24. data/app/countries/atlas_engine/bm/address_importer/corrections/open_address/city_alias_corrector.rb +38 -0
  25. data/app/countries/atlas_engine/bm/address_importer/open_address/mapper.rb +40 -0
  26. data/app/countries/atlas_engine/bm/country_profile.yml +12 -0
  27. data/app/countries/atlas_engine/br/country_profile.yml +4 -0
  28. data/app/countries/atlas_engine/ca/country_profile.yml +7 -0
  29. data/app/countries/atlas_engine/ca/synonyms.yml +1615 -0
  30. data/app/countries/atlas_engine/ch/address_importer/corrections/open_address/city_corrector.rb +29 -0
  31. data/app/countries/atlas_engine/ch/address_importer/corrections/open_address/locale_corrector.rb +74 -0
  32. data/app/countries/atlas_engine/ch/address_importer/open_address/mapper.rb +40 -0
  33. data/app/countries/atlas_engine/ch/country_profile.yml +15 -0
  34. data/app/countries/atlas_engine/ch/locales/de/country_profile.yml +15 -0
  35. data/app/countries/atlas_engine/ch/locales/de/index_configuration.yml +63 -0
  36. data/app/countries/atlas_engine/ch/locales/de/synonyms.yml +7 -0
  37. data/app/countries/atlas_engine/ch/locales/fr/synonyms.yml +21 -0
  38. data/app/countries/atlas_engine/cz/country_profile.yml +6 -0
  39. data/app/countries/atlas_engine/de/country_profile.yml +19 -0
  40. data/app/countries/atlas_engine/de/index_configuration.yml +64 -0
  41. data/app/countries/atlas_engine/de/synonyms.yml +2 -0
  42. data/app/countries/atlas_engine/de/validation_transcriber/address_parser.rb +19 -0
  43. data/app/countries/atlas_engine/dk/country_profile.yml +6 -0
  44. data/app/countries/atlas_engine/dk/synonyms.yml +3 -0
  45. data/app/countries/atlas_engine/dk/validation_transcriber/address_parser.rb +21 -0
  46. data/app/countries/atlas_engine/fo/country_profile.yml +5 -0
  47. data/app/countries/atlas_engine/fr/address_importer/corrections/open_address/city_corrector.rb +28 -0
  48. data/app/countries/atlas_engine/fr/country_profile.yml +13 -0
  49. data/app/countries/atlas_engine/fr/synonyms.yml +21 -0
  50. data/app/countries/atlas_engine/fr/validation_transcriber/address_parser.rb +34 -0
  51. data/app/countries/atlas_engine/gb/address_validation/es/query_builder.rb +98 -0
  52. data/app/countries/atlas_engine/gb/country_profile.yml +10 -0
  53. data/app/countries/atlas_engine/gb/validation_transcriber/full_address_parser.rb +164 -0
  54. data/app/countries/atlas_engine/gb/validation_transcriber/parsed_address.rb +120 -0
  55. data/app/countries/atlas_engine/gg/address_validation/validators/full_address/restrictions/unsupported_city.rb +39 -0
  56. data/app/countries/atlas_engine/gg/country_profile.yml +7 -0
  57. data/app/countries/atlas_engine/ie/country_profile.yml +3 -0
  58. data/app/countries/atlas_engine/it/address_importer/corrections/open_address/city_corrector.rb +27 -0
  59. data/app/countries/atlas_engine/it/address_importer/corrections/open_address/province_corrector.rb +29 -0
  60. data/app/countries/atlas_engine/it/address_importer/open_address/mapper.rb +42 -0
  61. data/app/countries/atlas_engine/it/country_profile.yml +11 -0
  62. data/app/countries/atlas_engine/jp/address_validation/es/data_mapper.rb +63 -0
  63. data/app/countries/atlas_engine/jp/country_profile.yml +6 -0
  64. data/app/countries/atlas_engine/kr/address_importer/open_address/mapper.rb +41 -0
  65. data/app/countries/atlas_engine/kr/country_profile.yml +11 -0
  66. data/app/countries/atlas_engine/li/address_importer/corrections/open_address/city_corrector.rb +25 -0
  67. data/app/countries/atlas_engine/li/country_profile.yml +21 -0
  68. data/app/countries/atlas_engine/li/index_configuration.yml +63 -0
  69. data/app/countries/atlas_engine/li/synonyms.yml +6 -0
  70. data/app/countries/atlas_engine/lt/country_profile.yml +6 -0
  71. data/app/countries/atlas_engine/lt/synonyms.yml +7 -0
  72. data/app/countries/atlas_engine/lt/validation_transcriber/address_parser.rb +24 -0
  73. data/app/countries/atlas_engine/lu/address_importer/corrections/open_address/locale_corrector.rb +54 -0
  74. data/app/countries/atlas_engine/lu/country_profile.yml +12 -0
  75. data/app/countries/atlas_engine/nl/address_importer/corrections/open_address/city_corrector.rb +25 -0
  76. data/app/countries/atlas_engine/nl/country_profile.yml +18 -0
  77. data/app/countries/atlas_engine/nl/index_configuration.yml +52 -0
  78. data/app/countries/atlas_engine/nl/synonyms.yml +92 -0
  79. data/app/countries/atlas_engine/nl/validation_transcriber/address_parser.rb +85 -0
  80. data/app/countries/atlas_engine/no/country_profile.yml +5 -0
  81. data/app/countries/atlas_engine/nz/country_profile.yml +3 -0
  82. data/app/countries/atlas_engine/pl/country_profile.yml +5 -0
  83. data/app/countries/atlas_engine/pl/validation_transcriber/address_parser.rb +19 -0
  84. data/app/countries/atlas_engine/pt/address_importer/corrections/open_address/city_corrector.rb +32 -0
  85. data/app/countries/atlas_engine/pt/address_importer/open_address/mapper.rb +39 -0
  86. data/app/countries/atlas_engine/pt/country_profile.yml +10 -0
  87. data/app/countries/atlas_engine/pt/synonyms.yml +7 -0
  88. data/app/countries/atlas_engine/sa/country_profile.yml +10 -0
  89. data/app/countries/atlas_engine/se/country_profile.yml +5 -0
  90. data/app/countries/atlas_engine/tt/address_importer/open_address/mapper.rb +38 -0
  91. data/app/countries/atlas_engine/tt/country_profile.yml +7 -0
  92. data/app/countries/atlas_engine/us/country_profile.yml +12 -0
  93. data/app/countries/atlas_engine/us/synonyms.yml +350 -0
  94. data/app/graphql/atlas_engine/errors/locale_unsupported_error.rb +17 -0
  95. data/app/graphql/atlas_engine/schema.graphql +1293 -0
  96. data/app/graphql/atlas_engine/schema.rb +23 -0
  97. data/app/graphql/atlas_engine/types/address_validation/address_input.rb +51 -0
  98. data/app/graphql/atlas_engine/types/address_validation/concern_type.rb +20 -0
  99. data/app/graphql/atlas_engine/types/address_validation/enums/concern_enum.rb +15 -0
  100. data/app/graphql/atlas_engine/types/address_validation/field_type.rb +15 -0
  101. data/app/graphql/atlas_engine/types/address_validation/suggestion_type.rb +21 -0
  102. data/app/graphql/atlas_engine/types/base_argument.rb +9 -0
  103. data/app/graphql/atlas_engine/types/base_enum.rb +9 -0
  104. data/app/graphql/atlas_engine/types/base_field.rb +10 -0
  105. data/app/graphql/atlas_engine/types/base_input_object.rb +9 -0
  106. data/app/graphql/atlas_engine/types/base_interface.rb +10 -0
  107. data/app/graphql/atlas_engine/types/base_object.rb +9 -0
  108. data/app/graphql/atlas_engine/types/base_scalar.rb +9 -0
  109. data/app/graphql/atlas_engine/types/base_union.rb +9 -0
  110. data/app/graphql/atlas_engine/types/matching_strategy_type.rb +12 -0
  111. data/app/graphql/atlas_engine/types/mutation_type.rb +9 -0
  112. data/app/graphql/atlas_engine/types/query_type.rb +61 -0
  113. data/app/graphql/atlas_engine/types/validation_supported_country.rb +12 -0
  114. data/app/graphql/atlas_engine/types/validation_type.rb +22 -0
  115. data/app/helpers/atlas_engine/address_importer/import_log_helper.rb +66 -0
  116. data/app/helpers/atlas_engine/application_helper.rb +7 -0
  117. data/app/helpers/atlas_engine/locale_format_helper.rb +40 -0
  118. data/app/helpers/atlas_engine/log_base.rb +32 -0
  119. data/app/helpers/atlas_engine/log_helper.rb +24 -0
  120. data/app/helpers/atlas_engine/metrics_helper.rb +25 -0
  121. data/app/jobs/atlas_engine/address_importer/clear_records_job.rb +39 -0
  122. data/app/jobs/atlas_engine/address_importer/open_address/geo_json_import_job.rb +212 -0
  123. data/app/jobs/atlas_engine/address_importer/open_address/geo_json_import_launcher_job.rb +67 -0
  124. data/app/jobs/atlas_engine/address_importer/open_address/prepares_geo_json_file.rb +41 -0
  125. data/app/jobs/atlas_engine/address_importer/resumable_import_job.rb +49 -0
  126. data/app/jobs/atlas_engine/address_importer/street_backfill_job.rb +63 -0
  127. data/app/jobs/atlas_engine/application_job.rb +10 -0
  128. data/app/jobs/atlas_engine/concerns/address_importer/handles_errors.rb +43 -0
  129. data/app/lib/atlas_engine/concern_formatter.rb +40 -0
  130. data/app/lib/atlas_engine/restrictions/base.rb +20 -0
  131. data/app/lib/atlas_engine/restrictions/unsupported_script.rb +31 -0
  132. data/app/lib/atlas_engine/validation_transcriber/address_parser_base.rb +201 -0
  133. data/app/lib/atlas_engine/validation_transcriber/address_parser_factory.rb +27 -0
  134. data/app/lib/atlas_engine/validation_transcriber/address_parser_north_america.rb +39 -0
  135. data/app/lib/atlas_engine/validation_transcriber/address_parser_oceanic.rb +17 -0
  136. data/app/lib/atlas_engine/validation_transcriber/address_parser_preprocessor.rb +132 -0
  137. data/app/lib/atlas_engine/validation_transcriber/address_parsing_helper.rb +38 -0
  138. data/app/lib/atlas_engine/validation_transcriber/address_parsings.rb +54 -0
  139. data/app/lib/atlas_engine/validation_transcriber/constants.rb +50 -0
  140. data/app/lib/atlas_engine/validation_transcriber/english_street_parser.rb +59 -0
  141. data/app/lib/atlas_engine/validation_transcriber/formatter.rb +46 -0
  142. data/app/lib/atlas_engine/validation_transcriber/french_street_parser.rb +50 -0
  143. data/app/lib/atlas_engine/validation_transcriber/province_code_normalizer.rb +45 -0
  144. data/app/lib/atlas_engine/validation_transcriber/street_parser.rb +18 -0
  145. data/app/lib/atlas_engine/validation_transcriber/zip_normalizer.rb +23 -0
  146. data/app/mailers/atlas_engine/application_mailer.rb +9 -0
  147. data/app/models/atlas_engine/address_importer/corrections/corrector.rb +33 -0
  148. data/app/models/atlas_engine/address_importer/import_events_notifier/base.rb +35 -0
  149. data/app/models/atlas_engine/address_importer/import_events_notifier/notifier.rb +26 -0
  150. data/app/models/atlas_engine/address_importer/open_address/default_mapper.rb +46 -0
  151. data/app/models/atlas_engine/address_importer/open_address/feature_helper.rb +110 -0
  152. data/app/models/atlas_engine/address_importer/open_address/filter.rb +17 -0
  153. data/app/models/atlas_engine/address_importer/open_address/loader.rb +27 -0
  154. data/app/models/atlas_engine/address_importer/open_address/transformer.rb +39 -0
  155. data/app/models/atlas_engine/address_importer/open_address.rb +10 -0
  156. data/app/models/atlas_engine/address_importer/validation/base_validator.rb +86 -0
  157. data/app/models/atlas_engine/address_importer/validation/default_validator.rb +27 -0
  158. data/app/models/atlas_engine/address_importer/validation/field_validations/city.rb +47 -0
  159. data/app/models/atlas_engine/address_importer/validation/field_validations/interface.rb +29 -0
  160. data/app/models/atlas_engine/address_importer/validation/field_validations/province.rb +73 -0
  161. data/app/models/atlas_engine/address_importer/validation/field_validations/zip.rb +84 -0
  162. data/app/models/atlas_engine/address_importer/validation/validator.rb +17 -0
  163. data/app/models/atlas_engine/address_importer/validation/wrapper.rb +70 -0
  164. data/app/models/atlas_engine/address_number.rb +36 -0
  165. data/app/models/atlas_engine/address_number_range.rb +200 -0
  166. data/app/models/atlas_engine/address_validation/abstract_address.rb +49 -0
  167. data/app/models/atlas_engine/address_validation/address.rb +47 -0
  168. data/app/models/atlas_engine/address_validation/candidate.rb +109 -0
  169. data/app/models/atlas_engine/address_validation/candidate_tuple.rb +15 -0
  170. data/app/models/atlas_engine/address_validation/concern.rb +74 -0
  171. data/app/models/atlas_engine/address_validation/concern_producer.rb +19 -0
  172. data/app/models/atlas_engine/address_validation/concern_queue.rb +20 -0
  173. data/app/models/atlas_engine/address_validation/concern_record.rb +122 -0
  174. data/app/models/atlas_engine/address_validation/datastore_base.rb +27 -0
  175. data/app/models/atlas_engine/address_validation/errors.rb +13 -0
  176. data/app/models/atlas_engine/address_validation/es/candidate_selector.rb +70 -0
  177. data/app/models/atlas_engine/address_validation/es/data_mappers/decompounding_data_mapper.rb +39 -0
  178. data/app/models/atlas_engine/address_validation/es/data_mappers/default_data_mapper.rb +110 -0
  179. data/app/models/atlas_engine/address_validation/es/datastore.rb +229 -0
  180. data/app/models/atlas_engine/address_validation/es/default_query_builder.rb +30 -0
  181. data/app/models/atlas_engine/address_validation/es/query_builder.rb +160 -0
  182. data/app/models/atlas_engine/address_validation/es/term_vectors.rb +78 -0
  183. data/app/models/atlas_engine/address_validation/es/validators/full_address.rb +123 -0
  184. data/app/models/atlas_engine/address_validation/es/validators/full_address_street.rb +18 -0
  185. data/app/models/atlas_engine/address_validation/es/validators/restriction_evaluator.rb +37 -0
  186. data/app/models/atlas_engine/address_validation/field.rb +30 -0
  187. data/app/models/atlas_engine/address_validation/full_address_validator_base.rb +27 -0
  188. data/app/models/atlas_engine/address_validation/log_emitter.rb +66 -0
  189. data/app/models/atlas_engine/address_validation/matching_strategies.rb +16 -0
  190. data/app/models/atlas_engine/address_validation/normalizer.rb +38 -0
  191. data/app/models/atlas_engine/address_validation/predicate_pipeline.rb +80 -0
  192. data/app/models/atlas_engine/address_validation/request.rb +12 -0
  193. data/app/models/atlas_engine/address_validation/result.rb +154 -0
  194. data/app/models/atlas_engine/address_validation/runs_validation.rb +16 -0
  195. data/app/models/atlas_engine/address_validation/session.rb +47 -0
  196. data/app/models/atlas_engine/address_validation/statsd_emitter.rb +72 -0
  197. data/app/models/atlas_engine/address_validation/strategies.rb +10 -0
  198. data/app/models/atlas_engine/address_validation/suggestion.rb +97 -0
  199. data/app/models/atlas_engine/address_validation/token/comparator.rb +44 -0
  200. data/app/models/atlas_engine/address_validation/token/comparison.rb +76 -0
  201. data/app/models/atlas_engine/address_validation/token/sequence/comparator.rb +158 -0
  202. data/app/models/atlas_engine/address_validation/token/sequence/comparison.rb +166 -0
  203. data/app/models/atlas_engine/address_validation/token/sequence.rb +147 -0
  204. data/app/models/atlas_engine/address_validation/token/synonyms.rb +77 -0
  205. data/app/models/atlas_engine/address_validation/token.rb +113 -0
  206. data/app/models/atlas_engine/address_validation/validator.rb +147 -0
  207. data/app/models/atlas_engine/address_validation/validators/full_address/address_comparison.rb +97 -0
  208. data/app/models/atlas_engine/address_validation/validators/full_address/candidate_result.rb +164 -0
  209. data/app/models/atlas_engine/address_validation/validators/full_address/candidate_result_base.rb +46 -0
  210. data/app/models/atlas_engine/address_validation/validators/full_address/comparison_helper.rb +135 -0
  211. data/app/models/atlas_engine/address_validation/validators/full_address/components_to_validate.rb +88 -0
  212. data/app/models/atlas_engine/address_validation/validators/full_address/concern_builder.rb +127 -0
  213. data/app/models/atlas_engine/address_validation/validators/full_address/exclusions/exclusion_base.rb +23 -0
  214. data/app/models/atlas_engine/address_validation/validators/full_address/invalid_zip_concern_builder.rb +42 -0
  215. data/app/models/atlas_engine/address_validation/validators/full_address/invalid_zip_for_country_concern.rb +37 -0
  216. data/app/models/atlas_engine/address_validation/validators/full_address/invalid_zip_for_province_concern.rb +37 -0
  217. data/app/models/atlas_engine/address_validation/validators/full_address/no_candidate_result.rb +26 -0
  218. data/app/models/atlas_engine/address_validation/validators/full_address/number_comparison.rb +31 -0
  219. data/app/models/atlas_engine/address_validation/validators/full_address/postal_code_matcher.rb +60 -0
  220. data/app/models/atlas_engine/address_validation/validators/full_address/result_updater.rb +42 -0
  221. data/app/models/atlas_engine/address_validation/validators/full_address/suggestion_builder.rb +140 -0
  222. data/app/models/atlas_engine/address_validation/validators/full_address/unknown_address_concern.rb +30 -0
  223. data/app/models/atlas_engine/address_validation/validators/full_address/unknown_province_concern.rb +38 -0
  224. data/app/models/atlas_engine/address_validation/validators/full_address/unknown_zip_for_address_concern.rb +32 -0
  225. data/app/models/atlas_engine/address_validation/validators/full_address/unmatched_field_concern.rb +84 -0
  226. data/app/models/atlas_engine/address_validation/validators/full_address/unsupported_script_result.rb +22 -0
  227. data/app/models/atlas_engine/address_validation/validators/predicates/cache.rb +38 -0
  228. data/app/models/atlas_engine/address_validation/validators/predicates/city/present.rb +36 -0
  229. data/app/models/atlas_engine/address_validation/validators/predicates/country/exists.rb +34 -0
  230. data/app/models/atlas_engine/address_validation/validators/predicates/country/valid_for_zip.rb +60 -0
  231. data/app/models/atlas_engine/address_validation/validators/predicates/no_emojis.rb +38 -0
  232. data/app/models/atlas_engine/address_validation/validators/predicates/no_html_tags.rb +39 -0
  233. data/app/models/atlas_engine/address_validation/validators/predicates/no_url.rb +38 -0
  234. data/app/models/atlas_engine/address_validation/validators/predicates/not_exceed_max_length.rb +34 -0
  235. data/app/models/atlas_engine/address_validation/validators/predicates/not_exceed_max_token_count.rb +63 -0
  236. data/app/models/atlas_engine/address_validation/validators/predicates/phone/valid.rb +41 -0
  237. data/app/models/atlas_engine/address_validation/validators/predicates/predicate.rb +37 -0
  238. data/app/models/atlas_engine/address_validation/validators/predicates/province/exists.rb +43 -0
  239. data/app/models/atlas_engine/address_validation/validators/predicates/province/valid_for_country.rb +48 -0
  240. data/app/models/atlas_engine/address_validation/validators/predicates/street/building_number_in_address1.rb +45 -0
  241. data/app/models/atlas_engine/address_validation/validators/predicates/street/building_number_in_address1_or_address2.rb +43 -0
  242. data/app/models/atlas_engine/address_validation/validators/predicates/street/present.rb +35 -0
  243. data/app/models/atlas_engine/address_validation/validators/predicates/zip/present.rb +58 -0
  244. data/app/models/atlas_engine/address_validation/validators/predicates/zip/valid_for_country.rb +45 -0
  245. data/app/models/atlas_engine/address_validation/validators/predicates/zip/valid_for_province.rb +55 -0
  246. data/app/models/atlas_engine/address_validation/validators/predicates/zip/zip_base.rb +25 -0
  247. data/app/models/atlas_engine/address_validation/zip_truncator.rb +32 -0
  248. data/app/models/atlas_engine/application_record.rb +8 -0
  249. data/app/models/atlas_engine/coded_error.rb +18 -0
  250. data/app/models/atlas_engine/coded_errors.rb +17 -0
  251. data/app/models/atlas_engine/country_import.rb +44 -0
  252. data/app/models/atlas_engine/country_profile.rb +270 -0
  253. data/app/models/atlas_engine/country_profile_ingestion_subset.rb +42 -0
  254. data/app/models/atlas_engine/country_profile_subset_base.rb +22 -0
  255. data/app/models/atlas_engine/country_profile_validation_subset.rb +48 -0
  256. data/app/models/atlas_engine/country_repository.rb +110 -0
  257. data/app/models/atlas_engine/elasticsearch/client.rb +116 -0
  258. data/app/models/atlas_engine/elasticsearch/client_interface.rb +89 -0
  259. data/app/models/atlas_engine/elasticsearch/repository.rb +246 -0
  260. data/app/models/atlas_engine/elasticsearch/repository_interface.rb +82 -0
  261. data/app/models/atlas_engine/elasticsearch/response.rb +20 -0
  262. data/app/models/atlas_engine/event.rb +12 -0
  263. data/app/models/atlas_engine/field_decompounder.rb +36 -0
  264. data/app/models/atlas_engine/index_configuration_factory.rb +188 -0
  265. data/app/models/atlas_engine/post_address.rb +114 -0
  266. data/app/models/atlas_engine/post_address_importer.rb +34 -0
  267. data/app/models/atlas_engine/services/service_helper.rb +21 -0
  268. data/app/models/atlas_engine/services/validation.rb +65 -0
  269. data/app/models/atlas_engine/services/validation_eligibility.rb +18 -0
  270. data/app/models/atlas_engine/street.rb +34 -0
  271. data/app/tasks/maintenance/atlas_engine/elasticsearch_index_create_task.rb +106 -0
  272. data/app/tasks/maintenance/atlas_engine/geo_json_import_task.rb +29 -0
  273. data/app/views/atlas_engine/connectivity/index.html.erb +50 -0
  274. data/app/views/atlas_engine/country_imports/index.html.erb +49 -0
  275. data/app/views/atlas_engine/country_imports/show.html.erb +73 -0
  276. data/app/views/layouts/atlas_engine/application.html.erb +15 -0
  277. data/config/initializers/1.ruby_patches.rb +18 -0
  278. data/config/initializers/sorbet.rb +5 -0
  279. data/config/initializers/worldwide.rb +5 -0
  280. data/config/locales/internal/en.yml +14 -0
  281. data/config/routes.rb +17 -0
  282. data/db/data/address_synonyms/index_configurations/default.yml +141 -0
  283. data/db/data/country_profiles/default.yml +23 -0
  284. data/db/data/transcriber.yml +760 -0
  285. data/db/data/validation_pipelines/es.yml +58 -0
  286. data/db/data/validation_pipelines/es_street.yml +58 -0
  287. data/db/data/validation_pipelines/local.yml +60 -0
  288. data/db/migrate/20230919173037_create_atlas_engine_post_addresses.rb +25 -0
  289. data/db/migrate/20231117142735_add_building_and_unit_ranges_column.rb +7 -0
  290. data/db/migrate/20231117143536_create_atlas_engine_country_imports.rb +11 -0
  291. data/db/migrate/20231117145844_create_atlas_engine_events_table.rb +13 -0
  292. data/db/migrate/20231123153554_add_unique_index_to_atlas_engine_post_addresses.rb +14 -0
  293. data/db/migrate/20231123154658_add_index_to_post_addresses_on_source_id_locale_country_code.rb +12 -0
  294. data/lib/atlas_engine/engine.rb +10 -0
  295. data/lib/atlas_engine/version.rb +6 -0
  296. data/lib/atlas_engine.rb +66 -0
  297. data/lib/tasks/atlas_engine/address_importer.rake +20 -0
  298. metadata +553 -0
@@ -0,0 +1,34 @@
1
+ # typed: false
2
+ # frozen_string_literal: true
3
+
4
+ require "csv"
5
+
6
+ module AtlasEngine
7
+ class PostAddressImporter
8
+ def initialize(file)
9
+ @file = file
10
+ end
11
+
12
+ def import
13
+ CSV.foreach(@file, headers: true) do |row|
14
+ PostAddress.create!(
15
+ source_id: row["source_id"],
16
+ locale: row["locale"],
17
+ country_code: row["country_code"],
18
+ province_code: row["province_code"],
19
+ region1: row["region1"],
20
+ region2: row["region2"],
21
+ region3: row["region3"],
22
+ region4: row["region4"],
23
+ city: row["city"],
24
+ suburb: row["suburb"],
25
+ zip: row["zip"],
26
+ street: row["street"],
27
+ building_name: row["building_name"],
28
+ latitude: row["latitude"],
29
+ longitude: row["longitude"],
30
+ )
31
+ end
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,21 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module AtlasEngine
5
+ module Services
6
+ module ServiceHelper
7
+ def handle_metrics(request_type, country_code, using_session_token)
8
+ result = yield
9
+
10
+ ActiveSupport::Notifications.instrument("atlas.service.events", {
11
+ request_type: request_type,
12
+ country_code: country_code,
13
+ using_session_token: using_session_token,
14
+ result: result,
15
+ })
16
+
17
+ result
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,65 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module AtlasEngine
5
+ module Services
6
+ class Validation
7
+ extend T::Sig
8
+ VALIDATE_ADDRESS = "validate_address"
9
+
10
+ class << self
11
+ extend T::Sig
12
+ include ServiceHelper
13
+ T.unsafe(self).include(AtlasEngine.validation_eligibility.constantize)
14
+
15
+ sig { params(request: AddressValidation::Request).returns(AddressValidation::Result) }
16
+ def validate_address(request)
17
+ country_code = request.address.country_code
18
+ handle_metrics(Validation::VALIDATE_ADDRESS, country_code, false) do
19
+ matching_strategy = if validation_enabled?(request.address)
20
+ serialize_strategy(request.matching_strategy, request.address)
21
+ else
22
+ AddressValidation::MatchingStrategies::Local
23
+ end
24
+
25
+ validator = AddressValidation::Validator.new(
26
+ address: request.address,
27
+ locale: request.locale,
28
+ matching_strategy: matching_strategy,
29
+ context: request.address.context.to_h,
30
+ )
31
+ result = validator.run
32
+
33
+ AddressValidation::StatsdEmitter.new(address: request.address, result: result).run
34
+ AddressValidation::LogEmitter.new(address: request.address, result: result).run
35
+
36
+ if Rails.configuration.x.captured_concerns.enabled
37
+ AddressValidation::ConcernProducer.add(result, request.address.context.to_h)
38
+ end
39
+
40
+ result
41
+ end
42
+ end
43
+
44
+ private
45
+
46
+ sig do
47
+ params(
48
+ request_matching_strategy: T.nilable(T.any(String, Symbol)),
49
+ address: AddressValidation::AbstractAddress,
50
+ ).returns(AddressValidation::MatchingStrategies)
51
+ end
52
+ def serialize_strategy(request_matching_strategy, address)
53
+ requested_strategy = if request_matching_strategy.nil?
54
+ CountryProfile.for(T.must(address.country_code)).validation.default_matching_strategy
55
+ else
56
+ request_matching_strategy.to_s.downcase
57
+ end
58
+
59
+ serialized_strategy = AddressValidation::MatchingStrategies.try_deserialize(requested_strategy)
60
+ serialized_strategy.presence || AddressValidation::MatchingStrategies::Local
61
+ end
62
+ end
63
+ end
64
+ end
65
+ end
@@ -0,0 +1,18 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module AtlasEngine
5
+ module Services
6
+ module ValidationEligibility
7
+ extend T::Sig
8
+
9
+ sig { params(address: AddressValidation::AbstractAddress).returns(T::Boolean) }
10
+ def validation_enabled?(address)
11
+ return false if address.country_code.blank?
12
+ return true unless Rails.env.production? || Rails.env.test?
13
+
14
+ CountryProfile.for(T.must(address.country_code)).validation.enabled
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,34 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module AtlasEngine
5
+ class Street
6
+ extend T::Sig
7
+
8
+ attr_reader :street
9
+
10
+ sig { params(street: String).void }
11
+ def initialize(street:)
12
+ @street = street
13
+ end
14
+
15
+ sig { returns(T.nilable(String)) }
16
+ def name
17
+ parsing[:name]
18
+ end
19
+
20
+ sig { returns(String) }
21
+ def with_stripped_name
22
+ return street if name.blank?
23
+
24
+ street.sub(name, T.must(name).gsub(/\s+/, ""))
25
+ end
26
+
27
+ private
28
+
29
+ sig { returns(T::Hash[Symbol, String]) }
30
+ def parsing
31
+ @parsing ||= AtlasEngine::ValidationTranscriber::StreetParser.new.parse(street: street)
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,106 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module Maintenance
5
+ module AtlasEngine
6
+ class ElasticsearchIndexCreateTask < MaintenanceTasks::Task
7
+ include ::AtlasEngine::LogHelper
8
+ extend T::Sig
9
+
10
+ attribute :country_code, :string
11
+ attribute :locale, :string # optional (no locale implies index = country_code)
12
+ attribute :province_codes, :string # optional comma separated list of province codes
13
+ attribute :shard_override, :integer, default: nil
14
+ attribute :replica_override, :integer, default: nil
15
+ attribute :activate_index, :boolean, default: false unless Rails.env.production?
16
+
17
+ validates :country_code, presence: true
18
+
19
+ attr_writer :repository
20
+
21
+ after_complete :switch_index unless Rails.env.production?
22
+
23
+ sig { returns(T.nilable(ActiveRecord::Batches::BatchEnumerator)) }
24
+ def collection
25
+ @batch_number = 0
26
+ batch_size = 2000
27
+
28
+ sanitized_country_code = validate(T.must(country_code))
29
+
30
+ address_conditions = {
31
+ country_code: sanitized_country_code,
32
+ province_code: sanitized_province_codes,
33
+ locale: sanitized_locale,
34
+ }.compact_blank
35
+
36
+ record_count = ::AtlasEngine::PostAddress.where(address_conditions).size
37
+ raise "No records to process for country code: #{country_code}" if record_count.zero?
38
+
39
+ repository.create_next_index(ensure_clean: true)
40
+
41
+ ::AtlasEngine::PostAddress.where(address_conditions).in_batches(of: batch_size)
42
+ end
43
+
44
+ sig { params(batch_of_post_address: ActiveRecord::Relation).void }
45
+ def process(batch_of_post_address)
46
+ log_info("Processing batch #{@batch_number} for repository #{repository.read_alias_name.upcase}.")
47
+ repository.save_records_backfill(batch_of_post_address)
48
+ @batch_number += 1
49
+ end
50
+
51
+ sig { returns(::AtlasEngine::CountryRepository) }
52
+ def repository
53
+ @repository ||= ::AtlasEngine::CountryRepository.new(
54
+ country_code: T.must(country_code),
55
+ repository_class: ::AtlasEngine.elasticsearch_repository.constantize,
56
+ locale: sanitized_locale,
57
+ index_configuration: index_configuration,
58
+ )
59
+ end
60
+
61
+ sig { void }
62
+ def switch_index
63
+ if activate_index
64
+ repository.switch_to_next_index
65
+ log_info("Switched index `#{repository.read_alias_name}` live.")
66
+ end
67
+ end
68
+
69
+ private
70
+
71
+ sig { params(country_code: String).returns(String) }
72
+ def validate(country_code)
73
+ region = Worldwide.region(code: country_code)
74
+ unless region.country?
75
+ raise ArgumentError, "Invalid country code: #{country_code}"
76
+ end
77
+
78
+ region.iso_code
79
+ end
80
+
81
+ sig { returns(T::Array[String]) }
82
+ def sanitized_province_codes
83
+ return [] if province_codes.blank?
84
+
85
+ T.must(province_codes).downcase.split(",").map(&:strip)
86
+ end
87
+
88
+ sig { returns(String) }
89
+ def sanitized_locale
90
+ locale.to_s.downcase
91
+ end
92
+
93
+ sig { returns(::AtlasEngine::IndexConfigurationFactory::IndexConfigurations) }
94
+ def index_configuration
95
+ ::AtlasEngine::IndexConfigurationFactory.new(
96
+ country_code: T.must(country_code),
97
+ locale: sanitized_locale,
98
+ shard_override: shard_override,
99
+ replica_override: replica_override,
100
+ ).index_configuration(
101
+ creating: true,
102
+ )
103
+ end
104
+ end
105
+ end
106
+ end
@@ -0,0 +1,29 @@
1
+ # typed: false
2
+ # frozen_string_literal: true
3
+
4
+ module Maintenance
5
+ module AtlasEngine
6
+ class GeoJsonImportTask < MaintenanceTasks::Task
7
+ include ::AtlasEngine::HandlesBlob
8
+
9
+ no_collection
10
+
11
+ attribute :clear_records, :boolean, default: true
12
+ # ISO3166 two-letter country code.
13
+ attribute :country_code, :string
14
+ validates :country_code, presence: true
15
+ # Filename to import. When running in staging or production, the worker expects to find
16
+ # this file in the relevant GCS bucket, configured in `config/storage/{environment}.yml`
17
+ # It must be placed under `openaddress/` with the same filename.
18
+ attribute :geojson_file_path, :string
19
+ attribute :locale, :string
20
+
21
+ def process = ::AtlasEngine::AddressImporter::OpenAddress::GeoJsonImportLauncherJob.perform_later(
22
+ country_code:,
23
+ geojson_file_path: geojson_file_path.strip,
24
+ clear_records:,
25
+ locale:,
26
+ )
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,50 @@
1
+ <h2>Elasticsearch Indices</h2>
2
+ <div id="elasticsearch_indices">
3
+ <table class="table table-striped table-bordered">
4
+ <thead>
5
+ <tr>
6
+ <th>Name</th>
7
+ <th>Size</th>
8
+ <th>Document Count</th>
9
+ </tr>
10
+ </thead>
11
+
12
+ <tbody>
13
+ <% @indices&.each do |index| %>
14
+ <tr>
15
+ <td><%= index["index"] %></td>
16
+ <td><%= index["store.size"] %></td>
17
+ <td><%= index["docs.count"] %></td>
18
+ </tr>
19
+ <% end %>
20
+ </tbody>
21
+ </table>
22
+ </div>
23
+
24
+ <h2>Post Address Records</h2>
25
+ <div id="post_addresses">
26
+ <table class="table table-striped table-bordered">
27
+ <thead>
28
+ <tr>
29
+ <th>ID</th>
30
+ <th>Street</th>
31
+ <th>City</th>
32
+ <th>Zip</th>
33
+ <th>Province Code</th>
34
+ </tr>
35
+ </thead>
36
+
37
+ <tbody>
38
+ <% @post_addresses&.each do |address| %>
39
+ <tr>
40
+ <td><%= address.id %></td>
41
+ <td><%= address.street %></td>
42
+ <td><%= address.city %></td>
43
+ <td><%= address.zip %></td>
44
+ <td><%= address.province_code %></td>
45
+ </tr>
46
+ <% end %>
47
+ </tbody>
48
+ </table>
49
+ </div>
50
+
@@ -0,0 +1,49 @@
1
+ <h1>Country Imports</h1>
2
+
3
+ <div id="country_imports">
4
+ <table class="table table-striped table-bordered">
5
+ <thead>
6
+ <tr>
7
+ <th>ID</th>
8
+ <th>country</th>
9
+ <th>state</th>
10
+ <th>created_at</th>
11
+ <th>updated_at</th>
12
+ <th>actions</th>
13
+ </tr>
14
+ </thead>
15
+
16
+ <tbody>
17
+ <% @country_imports.each do |import| %>
18
+ <tr>
19
+ <td><%=link_to import.id, country_import_path(import.id) %></td>
20
+ <td><%= import.country_code %></td>
21
+ <td><%= import.state %></td>
22
+ <td><%= import.created_at%></td>
23
+ <td><%= import.updated_at %></td>
24
+ <% if import.in_progress? || import.pending? %>
25
+ <td><%= link_to "Interrupt", interrupt_country_import_path(import.id), class: "btn btn-danger" %></td>
26
+ <% else %>
27
+ <td><p>-</p></td>
28
+ <% end %>
29
+ </tr>
30
+ <% end %>
31
+ </tbody>
32
+ </table>
33
+ </div>
34
+
35
+ <div class="container">
36
+ <div class="span2">
37
+ <% if @page > 0 %>
38
+ <%= link_to "← Previous", country_imports_path(page: @page - 1), type: "button", class: "btn btn-primary btn-block" %>
39
+ <% else %>
40
+ <%= link_to "← Previous", country_imports_path(page: @page - 1), type: "button", class: "btn btn-primary btn-block disabled" %>
41
+ <% end %>
42
+
43
+ <% if AtlasEngine::CountryImport.count > ((@page+1)*AtlasEngine::CountryImport::PAGE_SIZE) %>
44
+ <%= link_to "Next →", country_imports_path(page: @page + 1),type: "button", class:"btn btn-primary btn-block"%>
45
+ <% else %>
46
+ <%= link_to "Next →", country_imports_path(page: @page + 1),type: "button", class:"btn btn-primary btn-block disabled"%>
47
+ <% end %>
48
+ </div>
49
+ </div>
@@ -0,0 +1,73 @@
1
+ <head>
2
+ <meta http-equiv="refresh" content="15">
3
+ </head>
4
+ <h1><%= "#{t('admin.events.country_import.title')} #{@country_import&.id}" %></h1>
5
+ <br>
6
+
7
+ <% if @country_import&.in_progress? || @country_import&.pending? %>
8
+ <%= link_to "Interrupt", interrupt_country_import_path(@country_import.id), class: "btn btn-danger" %>
9
+ <br>
10
+ <br>
11
+ <% end %>
12
+
13
+ <div id="country_imports">
14
+ <table class="table table-striped table-bordered">
15
+ <tr>
16
+ <th><%= t('admin.events.country_import.country') %></th>
17
+ <td>
18
+ <%=@country_import&.country_code %>
19
+ </td>
20
+ </tr>
21
+ <tr>
22
+ <th><%= t('admin.events.country_import.state') %></th>
23
+ <td>
24
+ <%=@country_import&.state %>
25
+ </td>
26
+ </tr>
27
+ <tr>
28
+ <th><%= t('admin.events.country_import.created_at') %></th>
29
+ <td>
30
+ <% unless @country_import.nil? %>
31
+ <%="#{time_ago_in_words(@country_import&.created_at)} ago (#{@country_import&.created_at})"%>
32
+ <% end %>
33
+ </td>
34
+ </tr>
35
+ <tr>
36
+ <th><%= t('admin.events.country_import.updated_at') %></th>
37
+ <td>
38
+ <% unless @country_import.nil? %>
39
+ <%="#{time_ago_in_words(@country_import&.updated_at)} ago (#{@country_import&.updated_at})"%>
40
+ <% end %>
41
+ </td>
42
+ </tr>
43
+ </tbody>
44
+ </table>
45
+ <h1><%= t('admin.events.events.title') %></h1>
46
+ <br>
47
+ <div id="country_imports">
48
+ <table class="table table-striped table-bordered">
49
+ <thead>
50
+ <tr>
51
+ <th><%= t('admin.events.events.created_at') %></th>
52
+ <th><%= t('admin.events.events.message') %></th>
53
+ <th><%= t('admin.events.events.additional_params') %></th>
54
+ </tr>
55
+ </thead>
56
+ <tbody>
57
+ <% @events.each do |event| %>
58
+ <tr>
59
+ <td>
60
+ <%="#{time_ago_in_words(event.created_at)} ago (#{event.created_at})"%>
61
+ </td>
62
+ <td>
63
+ <%=event.message %>
64
+ </td>
65
+ <% if event.additional_params %>
66
+ <td><%= event.additional_params %></td>
67
+ <% else %>
68
+ <td><p>-</p></td>
69
+ <% end %>
70
+ </tr>
71
+ <% end %>
72
+ </tbody>
73
+ </table>
@@ -0,0 +1,15 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Atlas engine</title>
5
+ <%= csrf_meta_tags %>
6
+ <%= csp_meta_tag %>
7
+
8
+ <%= stylesheet_link_tag "atlas_engine/application", media: "all" %>
9
+ </head>
10
+ <body>
11
+
12
+ <%= yield %>
13
+
14
+ </body>
15
+ </html>
@@ -0,0 +1,18 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ module Enumerable
5
+ extend T::Sig
6
+
7
+ raise "Enumerable#stable_sort_by is already defined" if method_defined?(:stable_sort_by)
8
+
9
+ sig do
10
+ params(
11
+ block: T.proc.params(arg0: T.untyped).returns(T.any(Comparable, T::Array[BasicObject])),
12
+ )
13
+ .returns(T::Array[T.untyped])
14
+ end
15
+ def stable_sort_by(&block)
16
+ sort_by.with_index { |x, i| [yield(x), i] }
17
+ end
18
+ end
@@ -0,0 +1,5 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ # Includes `extend T::Sig` in every class
5
+ Module.include(T::Sig)
@@ -0,0 +1,5 @@
1
+ # typed: false
2
+ # frozen_string_literal: true
3
+
4
+ I18n.available_locales = Worldwide::Locales.known
5
+ Worldwide::Config.configure_i18n
@@ -0,0 +1,14 @@
1
+ en:
2
+ admin:
3
+ events:
4
+ country_import:
5
+ title: "Country Import"
6
+ country: "country:"
7
+ state: "state:"
8
+ created_at: "created_at:"
9
+ updated_at: "updated_at:"
10
+ events:
11
+ title: "Events"
12
+ created_at: "created_at"
13
+ message: "message"
14
+ additional_params: "additional params"
data/config/routes.rb ADDED
@@ -0,0 +1,17 @@
1
+ # typed: false
2
+ # frozen_string_literal: true
3
+
4
+ AtlasEngine::Engine.routes.draw do
5
+ mount MaintenanceTasks::Engine => "/maintenance_tasks"
6
+ if Rails.env.local?
7
+ mount GraphiQL::Rails::Engine, at: "/graphiql", graphql_path: "/graphql"
8
+ end
9
+ post "/graphql", to: "graphql#execute"
10
+ get "/connectivity", to: "connectivity#index"
11
+
12
+ resources :country_imports do
13
+ member do
14
+ get "interrupt"
15
+ end
16
+ end
17
+ end