]> git.openstreetmap.org Git - chef.git/blob - cookbooks/tile/recipes/default.rb
Add support for BindPaths and BindReadOnlyPaths to systemd_service
[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 template "/usr/local/bin/expire-tiles" do
526   source "expire-tiles.erb"
527   owner "root"
528   group "root"
529   mode "755"
530 end
531
532 directory "/var/lib/replicate" do
533   owner "tile"
534   group "tile"
535   mode "755"
536 end
537
538 directory "/var/lib/replicate/expire-queue" do
539   owner "tile"
540   group "_renderd"
541   mode "775"
542 end
543
544 template "/usr/local/bin/replicate" do
545   source "replicate.erb"
546   owner "root"
547   group "root"
548   mode "755"
549   variables :postgresql_version => postgresql_version.to_f
550 end
551
552 systemd_service "expire-tiles" do
553   description "Tile dirtying service"
554   type "simple"
555   user "_renderd"
556   exec_start "/usr/local/bin/expire-tiles"
557   nice 10
558   sandbox true
559   read_write_paths tile_directories + [
560     "/store/database/nodes",
561     "/var/lib/replicate/expire-queue",
562     "/var/log/tile"
563   ]
564 end
565
566 systemd_path "expire-tiles" do
567   description "Tile dirtying trigger"
568   directory_not_empty "/var/lib/replicate/expire-queue"
569 end
570
571 service "expire-tiles.path" do
572   action [:enable, :start]
573   subscribes :restart, "systemd_path[expire-tiles]"
574 end
575
576 systemd_service "replicate" do
577   description "Rendering database replication service"
578   after "postgresql.service"
579   wants "postgresql.service"
580   user "tile"
581   exec_start "/usr/local/bin/replicate"
582   sandbox :enable_network => true
583   restrict_address_families "AF_UNIX"
584   read_write_paths [
585     "/store/database/nodes",
586     "/var/lib/replicate",
587     "/var/log/tile"
588   ]
589   restart "on-failure"
590 end
591
592 service "replicate" do
593   action [:enable, :start]
594   subscribes :restart, "template[/usr/local/bin/replicate]"
595   subscribes :restart, "systemd_service[replicate]"
596 end
597
598 template "/usr/local/bin/render-lowzoom" do
599   source "render-lowzoom.erb"
600   owner "root"
601   group "root"
602   mode "755"
603 end
604
605 systemd_service "render-lowzoom" do
606   description "Render low zoom tiles"
607   condition_path_exists_glob "!/run/update-lowzoom-*"
608   user "tile"
609   exec_start "/usr/local/bin/render-lowzoom"
610   sandbox true
611   restrict_address_families "AF_UNIX"
612   read_write_paths "/var/log/tile"
613 end
614
615 systemd_timer "render-lowzoom" do
616   description "Render low zoom tiles"
617   on_calendar "Fri *-*-* 23:00:00 UTC"
618 end
619
620 service "render-lowzoom.timer" do
621   action [:enable, :start]
622 end
623
624 package "liblockfile-simple-perl"
625 package "libfilesys-df-perl"
626
627 template "/usr/local/bin/cleanup-tiles" do
628   source "cleanup-tiles.erb"
629   owner "root"
630   group "root"
631   mode "755"
632 end
633
634 systemd_service "cleanup-tiles@" do
635   description "Cleanup old tiles for /%I"
636   exec_start "/usr/local/bin/cleanup-tiles /%I"
637   user "_renderd"
638   io_scheduling_class "idle"
639   sandbox true
640   read_write_paths "/%I"
641 end
642
643 systemd_timer "cleanup-tiles@" do
644   description "Cleanup old tiles for /%I"
645   on_boot_sec "30m"
646   on_unit_inactive_sec "60m"
647   randomized_delay_sec "10m"
648 end
649
650 tile_directories.each do |directory|
651   label = directory[1..].gsub("/", "-")
652
653   service "cleanup-tiles@#{label}.timer" do
654     action [:enable, :start]
655   end
656 end
657
658 munin_plugin "mod_tile_fresh"
659 munin_plugin "mod_tile_latency"
660 munin_plugin "mod_tile_response"
661 munin_plugin "mod_tile_zoom"
662
663 munin_plugin "renderd_processed"
664 munin_plugin "renderd_queue"
665 munin_plugin "renderd_queue_time"
666 munin_plugin "renderd_zoom"
667 munin_plugin "renderd_zoom_time"
668
669 munin_plugin "replication_delay"
670
671 package "ruby-webrick"
672
673 prometheus_exporter "modtile" do
674   port 9494
675 end
676
677 prometheus_exporter "renderd" do
678   port 9393
679 end