% Licensed under the Apache License, Version 2.0 (the "License"); you may not % use this file except in compliance with the License. You may obtain a copy of % the License at % % http://www.apache.org/licenses/LICENSE-2.0 % % Unless required by applicable law or agreed to in writing, software % distributed under the License is distributed on an "AS IS" BASIS, WITHOUT % WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the % License for the specific language governing permissions and limitations under % the License. -module(couch_server_sup). -behaviour(supervisor). -export([start_link/1, couch_config_start_link_wrapper/2, restart_core_server/0, config_change/2]). -include("couch_db.hrl"). %% supervisor callbacks -export([init/1]). start_link(IniFiles) -> case whereis(couch_server_sup) of undefined -> start_server(IniFiles); _Else -> {error, already_started} end. restart_core_server() -> init:restart(). couch_config_start_link_wrapper(IniFiles, FirstConfigPid) -> case is_process_alive(FirstConfigPid) of true -> link(FirstConfigPid), {ok, FirstConfigPid}; false -> couch_config:start_link(IniFiles) end. start_server(IniFiles) -> case init:get_argument(pidfile) of {ok, [PidFile]} -> case file:write_file(PidFile, os:getpid()) of ok -> ok; Error -> io:format("Failed to write PID file ~s, error: ~p", [PidFile, Error]) end; _ -> ok end, {ok, ConfigPid} = couch_config:start_link(IniFiles), LogLevel = couch_config:get("log", "level", "info"), % announce startup io:format("Apache CouchDB ~s (LogLevel=~s) is starting.~n", [ couch:version(), LogLevel ]), case LogLevel of "debug" -> io:format("Configuration Settings ~p:~n", [IniFiles]), [io:format(" [~s] ~s=~p~n", [Module, Variable, Value]) || {{Module, Variable}, Value} <- couch_config:all()]; _ -> ok end, BaseChildSpecs = {{one_for_all, 10, 60}, [{couch_config, {couch_server_sup, couch_config_start_link_wrapper, [IniFiles, ConfigPid]}, permanent, brutal_kill, worker, [couch_config]}, {couch_config_event, {couch_config_event, start_link, []}, permanent, 1000, worker, dynamic}, {couch_primary_services, {couch_primary_sup, start_link, []}, permanent, infinity, supervisor, [couch_primary_sup]}, {couch_secondary_services, {couch_secondary_sup, start_link, []}, permanent, infinity, supervisor, [couch_secondary_sup]} ]}, {ok, Pid} = supervisor:start_link( {local, couch_server_sup}, couch_server_sup, BaseChildSpecs), % just restart if one of the config settings change. couch_config:register(fun ?MODULE:config_change/2, Pid), unlink(ConfigPid), Ip = couch_config:get("httpd", "bind_address"), io:format("Apache CouchDB has started. Time to relax.~n"), Uris = [get_uri(Name, Ip) || Name <- [couch_httpd, https]], [begin case Uri of undefined -> ok; Uri -> ?LOG_INFO("Apache CouchDB has started on ~s", [Uri]) end end || Uri <- Uris], case couch_config:get("couchdb", "uri_file", null) of null -> ok; UriFile -> Lines = [begin case Uri of undefined -> []; Uri -> io_lib:format("~s~n", [Uri]) end end || Uri <- Uris], ok = file:write_file(UriFile, Lines) end, {ok, Pid}. config_change("daemons", _) -> exit(whereis(couch_server_sup), shutdown); config_change("couchdb", "util_driver_dir") -> [Pid] = [P || {collation_driver,P,_,_} <- supervisor:which_children(couch_primary_services)], Pid ! reload_driver. init(ChildSpecs) -> {ok, ChildSpecs}. get_uri(Name, Ip) -> case get_port(Name) of undefined -> undefined; Port -> io_lib:format("~s://~s:~w/", [get_scheme(Name), Ip, Port]) end. get_scheme(couch_httpd) -> "http"; get_scheme(https) -> "https". get_port(Name) -> try mochiweb_socket_server:get(Name, port) catch exit:{noproc, _}-> undefined end.