]> git.openstreetmap.org Git - chef.git/blob - cookbooks/tile/recipes/default.rb
bfedc3c170ff4ef2bf239fe6269d46f7e89ea3ed
[chef.git] / cookbooks / tile / recipes / default.rb
1 #
2 # Cookbook:: tile
3 # Recipe:: default
4 #
5 # Copyright:: 2013, OpenStreetMap Foundation
6 #
7 # Licensed under the Apache License, Version 2.0 (the "License");
8 # you may not use this file except in compliance with the License.
9 # You may obtain a copy of the License at
10 #
11 #     https://www.apache.org/licenses/LICENSE-2.0
12 #
13 # Unless required by applicable law or agreed to in writing, software
14 # distributed under the License is distributed on an "AS IS" BASIS,
15 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 # See the License for the specific language governing permissions and
17 # limitations under the License.
18 #
19
20 include_recipe "accounts"
21 include_recipe "apache"
22 include_recipe "git"
23 include_recipe "munin"
24 include_recipe "nodejs"
25 include_recipe "postgresql"
26 include_recipe "prometheus"
27 include_recipe "python"
28 include_recipe "ruby"
29 include_recipe "tools"
30
31 blocks = data_bag_item("tile", "blocks")
32 admins = data_bag_item("apache", "admins")
33 web_passwords = data_bag_item("web", "passwords")
34
35 apache_module "alias"
36 apache_module "cgi"
37 apache_module "expires"
38 apache_module "headers"
39 apache_module "remoteip"
40 apache_module "rewrite"
41
42 apache_module "tile" do
43   conf "tile.conf.erb"
44 end
45
46 apache_conf "renderd" do
47   action :disable
48 end
49
50 ssl_certificate node[:fqdn] do
51   domains [node[:fqdn], "tile.openstreetmap.org", "render.openstreetmap.org"]
52   notifies :reload, "service[apache2]"
53 end
54
55 remote_file "#{Chef::Config[:file_cache_path]}/fastly-ip-list.json" do
56   source "https://api.fastly.com/public-ip-list"
57   compile_time true
58   ignore_failure true
59 end
60
61 fastlyips = JSON.parse(IO.read("#{Chef::Config[:file_cache_path]}/fastly-ip-list.json"))
62
63 remote_file "#{Chef::Config[:file_cache_path]}/statuscake-locations.json" do
64   source "https://app.statuscake.com/Workfloor/Locations.php?format=json"
65   compile_time true
66   ignore_failure true
67 end
68
69 statuscakelocations = JSON.parse(IO.read("#{Chef::Config[:file_cache_path]}/statuscake-locations.json"))
70
71 apache_site "default" do
72   action :disable
73 end
74
75 apache_site "tileserver_site" do
76   action :disable
77 end
78
79 apache_site "tile.openstreetmap.org" do
80   template "apache.erb"
81   variables :fastly => fastlyips["addresses"] + fastlyips["ipv6_addresses"],
82             :statuscake => statuscakelocations.flat_map { |_, v| [v["ip"], v["ipv6"]] },
83             :admins => admins["hosts"]
84 end
85
86 template "/etc/logrotate.d/apache2" do
87   source "logrotate.apache.erb"
88   owner "root"
89   group "root"
90   mode "644"
91 end
92
93 directory "/srv/tile.openstreetmap.org" do
94   owner "tile"
95   group "tile"
96   mode "755"
97 end
98
99 tile_directories = node[:tile][:styles].collect do |_, style|
100   style[:tile_directories].collect { |directory| directory[:name] }
101 end.flatten.sort.uniq
102
103 package "renderd"
104
105 systemd_service "renderd" do
106   dropin "chef"
107   after "postgresql.service"
108   wants "postgresql.service"
109   limit_nofile 4096
110   sandbox true
111   restrict_address_families "AF_UNIX"
112   read_write_paths tile_directories
113   restart "on-failure"
114 end
115
116 service "renderd" do
117   action [:enable, :start]
118   subscribes :restart, "systemd_service[renderd]"
119 end
120
121 directory "/srv/tile.openstreetmap.org/tiles" do
122   owner "tile"
123   group "tile"
124   mode "755"
125 end
126
127 template "/etc/renderd.conf" do
128   source "renderd.conf.erb"
129   owner "root"
130   group "root"
131   mode "644"
132   notifies :reload, "service[apache2]"
133   notifies :restart, "service[renderd]"
134 end
135
136 remote_directory "/srv/tile.openstreetmap.org/html" do
137   source "html"
138   owner "tile"
139   group "tile"
140   mode "755"
141   files_owner "tile"
142   files_group "tile"
143   files_mode "644"
144 end
145
146 template "/srv/tile.openstreetmap.org/html/index.html" do
147   source "index.html.erb"
148   owner "tile"
149   group "tile"
150   mode "644"
151 end
152
153 package %w[
154   python3-cairo
155   python3-mapnik
156   python3-pyproj
157   python3-setuptools
158 ]
159
160 python_package "pyotp" do
161   python_version "3"
162 end
163
164 directory "/srv/tile.openstreetmap.org/cgi-bin" do
165   owner "tile"
166   group "tile"
167   mode "755"
168 end
169
170 template "/srv/tile.openstreetmap.org/cgi-bin/export" do
171   source "export.erb"
172   owner "tile"
173   group "tile"
174   mode "755"
175   variables :blocks => blocks, :totp_key => web_passwords["totp_key"]
176 end
177
178 template "/srv/tile.openstreetmap.org/cgi-bin/debug" do
179   source "debug.erb"
180   owner "tile"
181   group "tile"
182   mode "755"
183 end
184
185 systemd_service "export-cleanup" do
186   description "Cleanup stale export temporary files"
187   joins_namespace_of "apache2.service"
188   exec_start "find /tmp -ignore_readdir_race -name 'export??????' -mmin +60 -delete"
189   user "www-data"
190   sandbox true
191 end
192
193 systemd_timer "export-cleanup" do
194   description "Cleanup stale export temporary files"
195   on_boot_sec "60m"
196   on_unit_inactive_sec "60m"
197 end
198
199 service "export-cleanup.timer" do
200   action [:enable, :start]
201 end
202
203 directory "/srv/tile.openstreetmap.org/data" do
204   owner "tile"
205   group "tile"
206   mode "755"
207 end
208
209 package %w[
210   mapnik-utils
211   tar
212   unzip
213 ]
214
215 node[:tile][:data].each_value do |data|
216   url = data[:url]
217   file = "/srv/tile.openstreetmap.org/data/#{File.basename(url)}"
218
219   if data[:directory]
220     directory = "/srv/tile.openstreetmap.org/data/#{data[:directory]}"
221
222     directory directory do
223       owner "tile"
224       group "tile"
225       mode "755"
226     end
227   else
228     directory = "/srv/tile.openstreetmap.org/data"
229   end
230
231   if file =~ /\.tgz$/
232     execute file do
233       action :nothing
234       command "tar -zxf #{file} -C #{directory}"
235       user "tile"
236       group "tile"
237     end
238   elsif file =~ /\.tar\.bz2$/
239     execute file do
240       action :nothing
241       command "tar -jxf #{file} -C #{directory}"
242       user "tile"
243       group "tile"
244     end
245   elsif file =~ /\.zip$/
246     execute file do
247       action :nothing
248       command "unzip -qq -o #{file} -d #{directory}"
249       user "tile"
250       group "tile"
251     end
252   end
253
254   execute "#{file}_shapeindex" do
255     action :nothing
256     command "find #{directory} -type f -iname '*.shp' -print0 | xargs -0 --no-run-if-empty shapeindex --shape_files"
257     user "tile"
258     group "tile"
259     subscribes :run, "execute[#{file}]", :immediately
260   end
261
262   remote_file file do
263     if data[:refresh]
264       action :create
265       use_conditional_get true
266       ignore_failure true
267     else
268       action :create_if_missing
269     end
270
271     source url
272     owner "tile"
273     group "tile"
274     mode "644"
275     backup false
276     notifies :run, "execute[#{file}]", :immediately
277     notifies :restart, "service[renderd]"
278   end
279 end
280
281 nodejs_package "carto"
282
283 systemd_service "update-lowzoom@" do
284   description "Low zoom tile update service for %i layer"
285   user "tile"
286   exec_start_pre "+/bin/systemctl stop render-lowzoom.service"
287   exec_start "/bin/bash /usr/local/bin/update-lowzoom-%i"
288   runtime_directory "update-lowzoom-%i"
289   sandbox true
290   restrict_address_families "AF_UNIX"
291   read_write_paths [
292     "/srv/tile.openstreetmap.org/tiles/%i",
293     "/var/log/tile"
294   ]
295   restart "on-failure"
296 end
297
298 directory "/srv/tile.openstreetmap.org/styles" do
299   owner "tile"
300   group "tile"
301   mode "755"
302 end
303
304 node[:tile][:styles].each do |name, details|
305   style_directory = "/srv/tile.openstreetmap.org/styles/#{name}"
306   tile_directory = "/srv/tile.openstreetmap.org/tiles/#{name}"
307
308   template "/usr/local/bin/update-lowzoom-#{name}" do
309     source "update-lowzoom.erb"
310     owner "root"
311     group "root"
312     mode "755"
313     variables :style => name
314   end
315
316   service "update-lowzoom@#{name}" do
317     action :disable
318     supports :restart => true
319   end
320
321   directory tile_directory do
322     owner "tile"
323     group "tile"
324     mode "755"
325   end
326
327   details[:tile_directories].each do |directory|
328     directory directory[:name] do
329       owner "_renderd"
330       group "_renderd"
331       mode "755"
332     end
333
334     directory[:min_zoom].upto(directory[:max_zoom]) do |zoom|
335       directory "#{directory[:name]}/#{zoom}" do
336         owner "_renderd"
337         group "_renderd"
338         mode "755"
339       end
340
341       link "#{tile_directory}/#{zoom}" do
342         to "#{directory[:name]}/#{zoom}"
343         owner "tile"
344         group "tile"
345       end
346     end
347   end
348
349   file "#{tile_directory}/planet-import-complete" do
350     action :create_if_missing
351     owner "tile"
352     group "tile"
353     mode "444"
354   end
355
356   git style_directory do
357     action :sync
358     repository details[:repository]
359     revision details[:revision]
360     user "tile"
361     group "tile"
362   end
363
364   link "#{style_directory}/data" do
365     to "/srv/tile.openstreetmap.org/data"
366     owner "tile"
367     group "tile"
368   end
369
370   if details[:fonts_script]
371     execute details[:fonts_script] do
372       action :nothing
373       command details[:fonts_script]
374       cwd style_directory
375       user "tile"
376       group "tile"
377       subscribes :run, "git[#{style_directory}]"
378     end
379   end
380
381   execute "#{style_directory}/project.mml" do
382     action :nothing
383     command "carto -a 3.0.22 project.mml > project.xml"
384     cwd style_directory
385     user "tile"
386     group "tile"
387     subscribes :run, "git[#{style_directory}]"
388     notifies :restart, "service[renderd]", :immediately
389     notifies :restart, "service[update-lowzoom@#{name}]"
390   end
391 end
392
393 postgresql_version = node[:tile][:database][:cluster].split("/").first
394 postgis_version = node[:tile][:database][:postgis]
395
396 package "postgresql-#{postgresql_version}-postgis-#{postgis_version}"
397
398 postgresql_user "jburgess" do
399   cluster node[:tile][:database][:cluster]
400   superuser true
401 end
402
403 postgresql_user "tomh" do
404   cluster node[:tile][:database][:cluster]
405   superuser true
406 end
407
408 postgresql_user "pnorman" do
409   cluster node[:tile][:database][:cluster]
410   superuser true
411 end
412
413 postgresql_user "tile" do
414   cluster node[:tile][:database][:cluster]
415 end
416
417 postgresql_user "www-data" do
418   cluster node[:tile][:database][:cluster]
419 end
420
421 postgresql_user "_renderd" do
422   cluster node[:tile][:database][:cluster]
423 end
424
425 postgresql_database "gis" do
426   cluster node[:tile][:database][:cluster]
427   owner "tile"
428 end
429
430 postgresql_extension "postgis" do
431   cluster node[:tile][:database][:cluster]
432   database "gis"
433 end
434
435 postgresql_extension "hstore" do
436   cluster node[:tile][:database][:cluster]
437   database "gis"
438   only_if { node[:tile][:database][:hstore] }
439 end
440
441 %w[geography_columns planet_osm_nodes planet_osm_rels planet_osm_ways raster_columns raster_overviews].each do |table|
442   postgresql_table table do
443     cluster node[:tile][:database][:cluster]
444     database "gis"
445     owner "tile"
446     permissions "tile" => :all
447   end
448 end
449
450 %w[geometry_columns planet_osm_line planet_osm_point planet_osm_polygon planet_osm_roads spatial_ref_sys].each do |table|
451   postgresql_table table do
452     cluster node[:tile][:database][:cluster]
453     database "gis"
454     owner "tile"
455     permissions "tile" => :all, "www-data" => :select, "_renderd" => :select
456   end
457 end
458
459 package %w[
460   gdal-bin
461   python3-yaml
462   python3-psycopg2
463 ]
464
465 if node[:tile][:database][:external_data_script]
466   execute node[:tile][:database][:external_data_script] do
467     command "#{node[:tile][:database][:external_data_script]} -R _renderd"
468     cwd "/srv/tile.openstreetmap.org"
469     user "tile"
470     group "tile"
471     ignore_failure true
472   end
473
474   Array(node[:tile][:database][:external_data_tables]).each do |table|
475     postgresql_table table do
476       cluster node[:tile][:database][:cluster]
477       database "gis"
478       owner "tile"
479       permissions "tile" => :all, "www-data" => :select, "_renderd" => :select
480     end
481   end
482 end
483
484 postgresql_munin "gis" do
485   cluster node[:tile][:database][:cluster]
486   database "gis"
487 end
488
489 directory File.dirname(node[:tile][:database][:node_file]) do
490   owner "root"
491   group "root"
492   mode "755"
493   recursive true
494 end
495
496 file node[:tile][:database][:node_file] do
497   owner "tile"
498   group "_renderd"
499   mode "660"
500 end
501
502 directory "/var/log/tile" do
503   owner "tile"
504   group "tile"
505   mode "755"
506 end
507
508 package %w[
509   osm2pgsql
510   osmium-tool
511   pyosmium
512   python3-pyproj
513 ]
514
515 remote_directory "/usr/local/bin" do
516   source "bin"
517   owner "root"
518   group "root"
519   mode "755"
520   files_owner "root"
521   files_group "root"
522   files_mode "755"
523 end
524
525 directory "/var/lib/replicate" do
526   owner "tile"
527   group "tile"
528   mode "755"
529 end
530
531 if node[:tile][:replication][:engine] == "custom"
532   template "/usr/local/bin/expire-tiles" do
533     source "expire-tiles-custom.erb"
534     owner "root"
535     group "root"
536     mode "755"
537   end
538
539   directory "/var/lib/replicate/expire-queue" do
540     owner "tile"
541     group "_renderd"
542     mode "775"
543   end
544
545   template "/usr/local/bin/replicate" do
546     source "replicate.erb"
547     owner "root"
548     group "root"
549     mode "755"
550     variables :postgresql_version => postgresql_version.to_f
551   end
552
553   systemd_service "expire-tiles" do
554     description "Tile dirtying service"
555     type "simple"
556     user "_renderd"
557     exec_start "/usr/local/bin/expire-tiles"
558     nice 10
559     sandbox true
560     read_write_paths tile_directories + [
561                        "/store/database/nodes",
562                        "/var/lib/replicate/expire-queue",
563                        "/var/log/tile"
564                      ]
565   end
566
567   systemd_path "expire-tiles" do
568     description "Tile dirtying trigger"
569     directory_not_empty "/var/lib/replicate/expire-queue"
570   end
571
572   service "expire-tiles.path" do
573     action [:enable, :start]
574     subscribes :restart, "systemd_path[expire-tiles]"
575   end
576
577   systemd_service "replicate" do
578     description "Rendering database replication service"
579     after "postgresql.service"
580     wants "postgresql.service"
581     user "tile"
582     exec_start "/usr/local/bin/replicate"
583     sandbox :enable_network => true
584     restrict_address_families "AF_UNIX"
585     read_write_paths [
586       "/store/database/nodes",
587       "/var/lib/replicate",
588       "/var/log/tile"
589     ]
590     restart "on-failure"
591   end
592
593   service "replicate" do
594     action [:enable, :start]
595     subscribes :restart, "template[/usr/local/bin/replicate]"
596     subscribes :restart, "systemd_service[replicate]"
597   end
598 elsif node[:tile][:replication][:engine] == "osm2pgsql"
599   service "expire-tiles.path" do
600     action [:disable, :stop]
601   end
602
603   systemd_path "expire-tiles" do
604     action :delete
605   end
606
607   service "expire-tiles" do
608     action :stop
609   end
610
611   systemd_service "expire-tiles" do
612     action :delete
613   end
614
615   template "/usr/local/bin/expire-tiles" do
616     source "expire-tiles-osm2pgsql.erb"
617     owner "root"
618     group "root"
619     mode "755"
620   end
621
622   osm2pgsql_arguments = %w[
623     --number-processes=1
624     --log-progress=false
625     --expire-tiles=13-19
626     --expire-output=/var/lib/replicate/dirty-tiles.txt
627   ]
628
629   osm2pgsql_arguments.append("--multi-geometry") if node[:tile][:database][:multi_geometry]
630   osm2pgsql_arguments.append("--hstore") if node[:tile][:database][:hstore]
631   osm2pgsql_arguments.append("--tag-transform-script=#{node[:tile][:database][:tag_transform_script]}") if node[:tile][:database][:tag_transform_script]
632
633   systemd_service "replicate" do
634     description "Rendering database replication service"
635     after "postgresql.service"
636     wants "postgresql.service"
637     user "tile"
638     exec_start "/bin/osm2pgsql-replication update --database gis --post-processing /usr/local/bin/expire-tiles -- #{osm2pgsql_arguments.join(' ')}"
639     sandbox :enable_network => true
640     restrict_address_families "AF_UNIX"
641     read_write_paths [
642       "/store/database/nodes",
643       "/var/lib/replicate"
644     ]
645     restart "always"
646     restart_sec 30
647   end
648
649   service "replicate" do
650     action [:enable, :start]
651     subscribes :restart, "systemd_service[replicate]"
652   end
653 end
654
655 template "/usr/local/bin/render-lowzoom" do
656   source "render-lowzoom.erb"
657   owner "root"
658   group "root"
659   mode "755"
660 end
661
662 systemd_service "render-lowzoom" do
663   description "Render low zoom tiles"
664   condition_path_exists_glob "!/run/update-lowzoom-*"
665   user "tile"
666   exec_start "/usr/local/bin/render-lowzoom"
667   sandbox true
668   restrict_address_families "AF_UNIX"
669   read_write_paths "/var/log/tile"
670 end
671
672 systemd_timer "render-lowzoom" do
673   description "Render low zoom tiles"
674   on_calendar "Fri *-*-* 23:00:00 UTC"
675 end
676
677 service "render-lowzoom.timer" do
678   action [:enable, :start]
679 end
680
681 package "liblockfile-simple-perl"
682 package "libfilesys-df-perl"
683
684 template "/usr/local/bin/cleanup-tiles" do
685   source "cleanup-tiles.erb"
686   owner "root"
687   group "root"
688   mode "755"
689 end
690
691 systemd_service "cleanup-tiles@" do
692   description "Cleanup old tiles for /%I"
693   exec_start "/usr/local/bin/cleanup-tiles /%I"
694   user "_renderd"
695   io_scheduling_class "idle"
696   sandbox true
697   read_write_paths "/%I"
698 end
699
700 systemd_timer "cleanup-tiles@" do
701   description "Cleanup old tiles for /%I"
702   on_boot_sec "30m"
703   on_unit_inactive_sec "60m"
704   randomized_delay_sec "10m"
705 end
706
707 tile_directories.each do |directory|
708   label = directory[1..].gsub("/", "-")
709
710   service "cleanup-tiles@#{label}.timer" do
711     action [:enable, :start]
712   end
713 end
714
715 munin_plugin "mod_tile_fresh"
716 munin_plugin "mod_tile_latency"
717 munin_plugin "mod_tile_response"
718 munin_plugin "mod_tile_zoom"
719
720 munin_plugin "renderd_processed"
721 munin_plugin "renderd_queue"
722 munin_plugin "renderd_queue_time"
723 munin_plugin "renderd_zoom"
724 munin_plugin "renderd_zoom_time"
725
726 munin_plugin "replication_delay"
727
728 package "ruby-webrick"
729
730 prometheus_exporter "modtile" do
731   port 9494
732 end
733
734 prometheus_exporter "renderd" do
735   port 9393
736 end