load-service.cc 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758
  1. #include <algorithm>
  2. #include <string>
  3. #include <fstream>
  4. #include <locale>
  5. #include <limits>
  6. #include <list>
  7. #include <utility>
  8. #include <iterator>
  9. #include <cstring>
  10. #include <cstdlib>
  11. #include <sys/stat.h>
  12. #include <sys/types.h>
  13. #include <pwd.h>
  14. #include <grp.h>
  15. #include <dirent.h>
  16. #include "proc-service.h"
  17. #include "dinit-log.h"
  18. #include "dinit-util.h"
  19. #include "dinit-utmp.h"
  20. using string = std::string;
  21. using string_iterator = std::string::iterator;
  22. // Perform environment variable substitution on a command line, if specified.
  23. // line - the string storing the command and arguments
  24. // offsets - the [start,end) pair of offsets of the command and each argument within the string
  25. //
  26. static void do_env_subst(const char *setting_name, ha_string &line,
  27. std::list<std::pair<unsigned,unsigned>> &offsets,
  28. environment::env_map const &envmap)
  29. {
  30. using namespace dinit_load;
  31. std::string line_s = std::string(line.c_str(), line.length());
  32. value_var_subst(setting_name, line_s, offsets, resolve_env_var, envmap);
  33. line = line_s;
  34. }
  35. // Process a dependency directory - filenames contained within correspond to service names which
  36. // are loaded and added as a dependency of the given type. Expected use is with a directory
  37. // containing symbolic links to other service descriptions, but this isn't required.
  38. // Failure to read the directory contents, or to find a service listed within, is not considered
  39. // a fatal error.
  40. static void process_dep_dir(dirload_service_set &sset,
  41. const char *servicename,
  42. const string &service_filename,
  43. std::list<prelim_dep> &deplist, const std::string &depdirpath,
  44. dependency_type dep_type,
  45. const service_record *avoid_circular)
  46. {
  47. std::string depdir_fname = combine_paths(parent_path(service_filename), depdirpath.c_str());
  48. DIR *depdir = opendir(depdir_fname.c_str());
  49. if (depdir == nullptr) {
  50. log(loglevel_t::WARN, "Could not open dependency directory '", depdir_fname,
  51. "' for ", servicename, " service.");
  52. return;
  53. }
  54. errno = 0;
  55. dirent * dent = readdir(depdir);
  56. while (dent != nullptr) {
  57. char * name = dent->d_name;
  58. if (name[0] != '.') {
  59. try {
  60. service_record * sr = sset.load_service(name);
  61. deplist.emplace_back(sr, dep_type);
  62. }
  63. catch (service_not_found &) {
  64. log(loglevel_t::WARN, "Ignoring unresolved dependency '", name,
  65. "' in dependency directory '", depdirpath,
  66. "' for ", servicename, " service.");
  67. }
  68. }
  69. errno = 0; // errno may have changed in the meantime
  70. dent = readdir(depdir);
  71. }
  72. if (errno != 0) {
  73. log(loglevel_t::WARN, "Error reading dependency directory '", depdirpath,
  74. "' for ", servicename, " service.");
  75. }
  76. closedir(depdir);
  77. }
  78. service_record * dirload_service_set::load_service(const char * name, const service_record *avoid_circular)
  79. {
  80. return load_reload_service(name, nullptr, avoid_circular);
  81. }
  82. service_record * dirload_service_set::reload_service(service_record * service)
  83. {
  84. return load_reload_service(service->get_name().c_str(), service, service);
  85. }
  86. using service_dep_list = decltype(std::declval<dinit_load::service_settings_wrapper<prelim_dep>>().depends);
  87. // Check for dependency cycles for the specified service (orig) with the given set of dependencies
  88. static void check_cycle(service_dep_list &deps, service_record *orig)
  89. {
  90. linked_uo_set<service_record *> pending;
  91. for (auto &new_dep : deps) {
  92. if (new_dep.to == orig) {
  93. throw service_cyclic_dependency(orig->get_name());
  94. }
  95. pending.add_back(new_dep.to);
  96. }
  97. for (auto i = pending.begin(); i != pending.end(); ++i) {
  98. auto &dep_list = (*i)->get_dependencies();
  99. for (auto &dep : dep_list) {
  100. if (dep.get_to() == orig) {
  101. throw service_cyclic_dependency(orig->get_name());
  102. }
  103. pending.add_back(dep.get_to());
  104. }
  105. }
  106. }
  107. // Update the dependencies of the specified service atomically.
  108. // May fail with bad_alloc, service_cyclic_dependency.
  109. static void update_depenencies(service_record *service,
  110. dinit_load::service_settings_wrapper<prelim_dep> &settings,
  111. std::list<service_dep> &before_deps)
  112. {
  113. check_cycle(settings.depends, service);
  114. std::list<service_dep> &deps = service->get_dependencies();
  115. auto first_preexisting = deps.begin();
  116. auto &depts = service->get_dependents();
  117. auto first_pre_dept = depts.begin();
  118. try {
  119. // Insert all new dependents (from "before" relationships) before the first pre-existing dependent
  120. for (auto new_dept_i = before_deps.begin(); new_dept_i != before_deps.end(); ) {
  121. auto &new_dept = *new_dept_i;
  122. depts.insert(depts.begin(), &new_dept);
  123. // splice the dependency into the dependent:
  124. auto next_dept_i = std::next(new_dept_i);
  125. auto &from_deps = new_dept.get_from()->get_dependencies();
  126. from_deps.splice(from_deps.begin(), before_deps, new_dept_i);
  127. new_dept_i = next_dept_i;
  128. }
  129. // Insert all the new dependencies before the first pre-existing dependency
  130. for (auto &new_dep : settings.depends) {
  131. service->add_dep(new_dep.to, new_dep.dep_type, first_preexisting);
  132. }
  133. }
  134. catch (...) {
  135. // remove "before" dependencies from dependents
  136. for (auto i = depts.begin(); i != first_pre_dept; ) {
  137. auto next_i = std::next(i);
  138. (*i)->get_from()->rm_dep(**i);
  139. i = next_i;
  140. }
  141. // remove the inserted dependencies
  142. for (auto i = deps.begin(); i != first_preexisting; ) {
  143. i = service->rm_dep(i);
  144. }
  145. // re-throw the exception
  146. throw;
  147. }
  148. // Now remove all pre-existing dependencies (no exceptions possible from here).
  149. for( ; first_preexisting != deps.end(); ) {
  150. first_preexisting = service->rm_dep(first_preexisting);
  151. }
  152. // Also remove pre-existing "before" dependents
  153. for( ; first_pre_dept != depts.end(); ) {
  154. auto next_pre_dept = std::next(first_pre_dept);
  155. if ((*first_pre_dept)->dep_type == dependency_type::BEFORE) {
  156. (*first_pre_dept)->get_from()->rm_dep(**first_pre_dept);
  157. }
  158. first_pre_dept = next_pre_dept;
  159. }
  160. }
  161. // Update the command, and dependencies, of the specified service atomically.
  162. // May fail with bad_alloc, service_cyclic_dependency.
  163. static void update_command_and_dependencies(base_process_service *service,
  164. dinit_load::service_settings_wrapper<prelim_dep> &settings,
  165. std::list<service_dep> &before_deps)
  166. {
  167. // Get the current command parts
  168. ha_string orig_cmd; std::vector<const char *> orig_arg_parts;
  169. service->get_command(orig_cmd, orig_arg_parts);
  170. // Separate the new command parts and set
  171. std::vector<const char *> cmd_arg_parts = separate_args(settings.command, settings.command_offsets);
  172. service->set_command(std::move(settings.command), std::move(cmd_arg_parts));
  173. try {
  174. update_depenencies(service, settings, before_deps);
  175. }
  176. catch (...) {
  177. // restore original command
  178. service->set_command(std::move(orig_cmd), std::move(orig_arg_parts));
  179. // re-throw the exception
  180. throw;
  181. }
  182. }
  183. service_record * dirload_service_set::load_reload_service(const char *name, service_record *reload_svc,
  184. const service_record *avoid_circular)
  185. {
  186. // For reload, we have the following problems:
  187. // - ideally want to allow changing service type, at least for stopped services. That implies creating
  188. // a new (replacement) service_record object, at least in cases where the type does change.
  189. // - dependencies may change (including addition of new dependencies which aren't yet loaded). We need
  190. // to prevent cyclic dependencies forming.
  191. // - We want atomicity. If any new settings are not valid/alterable, or if a cyclic dependency is
  192. // created, nothing should change. Ideally this would extend to unloading any dependencies which were
  193. // loaded as part of the reload attempt.
  194. // - We need to either transfer handles referring to the old service (so that they refer to the new
  195. // service), or make them invalid. Or, we alter the original service without creating a new one
  196. // (which we can only do if the type doesn't change).
  197. // Approach:
  198. // - remember the initial service count, so we can remove services loaded as part of the reload
  199. // operation if we want to abort it later (i.e. if service count changed from N to N+X, remove the
  200. // last X services)
  201. // - check that the new settings are valid (if the service is running, check if the settings can be
  202. // altered, though we may just defer some changes until service is restarted)
  203. // - check all dependencies of the newly created service record for cyclic dependencies, via depth-first
  204. // traversal.
  205. // - If changing type:
  206. // - create the service initially just as if loading a new service (but with no dummy placeholder,
  207. // use the original service for that).
  208. // - switch all dependents to depend on the new record. Copy necessary runtime data from the original
  209. // to the new service record. Remove dependencies from the old record, and release any dependency
  210. // services as appropriate (so they stop if no longer needed). Finally, remove the old service
  211. // record and delete it.
  212. // Otherwise:
  213. // - copy the new settings to the existing service
  214. // - fix dependencies
  215. //
  216. // Limitations:
  217. // - caller must check there are no handles (or only a single requesting handle) to the service before
  218. // calling
  219. // - cannot change the type of a non-stopped service
  220. using std::string;
  221. using std::ifstream;
  222. using std::ios;
  223. using std::ios_base;
  224. using std::locale;
  225. using std::isspace;
  226. using std::list;
  227. using std::pair;
  228. using namespace dinit_load;
  229. if (reload_svc == nullptr) {
  230. // First try and find an existing record...
  231. service_record * rval = find_service(string(name));
  232. if (rval != nullptr) {
  233. if (rval == avoid_circular || rval->is_dummy()) {
  234. throw service_cyclic_dependency(name);
  235. }
  236. return rval;
  237. }
  238. }
  239. service_record *rval = nullptr;
  240. service_record *dummy = nullptr;
  241. ifstream service_file;
  242. string service_filename;
  243. int fail_load_errno = 0;
  244. std::string fail_load_path;
  245. // Couldn't find one. Have to load it.
  246. for (auto &service_dir : service_dirs) {
  247. service_filename = service_dir.get_dir();
  248. if (*(service_filename.rbegin()) != '/') {
  249. service_filename += '/';
  250. }
  251. service_filename += name;
  252. service_file.open(service_filename.c_str(), ios::in);
  253. if (service_file) break;
  254. if (errno != ENOENT && fail_load_errno == 0) {
  255. fail_load_errno = errno;
  256. fail_load_path = std::move(service_filename);
  257. }
  258. }
  259. if (!service_file) {
  260. if (fail_load_errno == 0) {
  261. throw service_not_found(string(name));
  262. }
  263. else {
  264. throw service_load_error(name, std::move(fail_load_path), fail_load_errno);
  265. }
  266. }
  267. service_settings_wrapper<prelim_dep> settings;
  268. string line;
  269. // getline can set failbit if it reaches end-of-file, we don't want an exception in that case. There's
  270. // no good way to handle an I/O error however, so we'll have exceptions thrown on badbit:
  271. service_file.exceptions(ios::badbit);
  272. bool create_new_record = true;
  273. try {
  274. if (reload_svc == nullptr) {
  275. // Add a dummy service record now to prevent infinite recursion in case of cyclic dependency.
  276. // We replace this with the real service later (or remove it if we find a configuration error).
  277. dummy = new service_record(this, string(name));
  278. add_service(dummy);
  279. }
  280. process_service_file(name, service_file,
  281. [&](string &line, unsigned line_num, string &setting,
  282. string_iterator &i, string_iterator &end) -> void {
  283. auto process_dep_dir_n = [&](std::list<prelim_dep> &deplist, const std::string &waitsford,
  284. dependency_type dep_type) -> void {
  285. process_dep_dir(*this, name, service_filename, deplist, waitsford, dep_type, reload_svc);
  286. };
  287. auto load_service_n = [&](const string &dep_name) -> service_record * {
  288. try {
  289. return load_service(dep_name.c_str(), reload_svc);
  290. }
  291. catch (service_description_exc &sle) {
  292. log_service_load_failure(sle);
  293. throw service_load_exc(name, "could not load dependency.");
  294. }
  295. catch (service_load_exc &sle) {
  296. log(loglevel_t::ERROR, "Could not load service ", sle.service_name, ": ",
  297. sle.exc_description);
  298. throw service_load_exc(name, "could not load dependency.");
  299. }
  300. };
  301. process_service_line(settings, name, line, line_num, setting, i, end, load_service_n,
  302. process_dep_dir_n);
  303. });
  304. service_file.close();
  305. auto report_err = [&](const char *msg){
  306. throw service_load_exc(name, msg);
  307. };
  308. environment srv_env;
  309. /* fill user vars before reading env file */
  310. if (settings.export_passwd_vars) {
  311. fill_environment_userinfo(settings.run_as_uid, name, srv_env);
  312. }
  313. /* set service name in environment if desired */
  314. if (settings.export_service_name) {
  315. std::string envname = "DINIT_SERVICE=";
  316. envname += name;
  317. srv_env.set_var(std::move(envname));
  318. }
  319. // this mapping is temporary, for load substitutions
  320. // the reason for this is that the environment actually *may* change
  321. // after load, e.g. through dinitctl setenv (either from the outside
  322. // or from within services) and we want this to refresh for each
  323. // process invocation
  324. environment::env_map srv_envmap;
  325. if (!settings.env_file.empty()) {
  326. try {
  327. read_env_file(settings.env_file.data(), false, srv_env);
  328. } catch (const std::system_error &se) {
  329. throw service_load_exc(name, std::string("could not load environment file: ") + se.what());
  330. }
  331. }
  332. srv_envmap = srv_env.build(main_env);
  333. settings.finalise(report_err, srv_envmap);
  334. auto service_type = settings.service_type;
  335. if (reload_svc != nullptr) {
  336. // Make sure settings are able to be changed/are compatible
  337. service_record *service = reload_svc;
  338. if (service->get_state() != service_state_t::STOPPED) {
  339. // Can not change type of a running service.
  340. if (service_type != service->get_type()) {
  341. throw service_load_exc(name, "cannot change type of non-stopped service.");
  342. }
  343. // Can not alter a starting/stopping service, at least for now.
  344. if (service->get_state() != service_state_t::STARTED) {
  345. throw service_load_exc(name,
  346. "cannot alter settings for service which is currently starting/stopping.");
  347. }
  348. // Check validity of dependencies (if started, regular deps must be started)
  349. for (auto &new_dep : settings.depends) {
  350. if (new_dep.dep_type == dependency_type::REGULAR) {
  351. if (new_dep.to->get_state() != service_state_t::STARTED) {
  352. throw service_load_exc(name,
  353. std::string("cannot add non-started dependency '")
  354. + new_dep.to->get_name() + "'.");
  355. }
  356. }
  357. }
  358. // Cannot change certain flags
  359. auto current_flags = service->get_flags();
  360. if (current_flags.starts_on_console != settings.onstart_flags.starts_on_console
  361. || current_flags.shares_console != settings.onstart_flags.shares_console) {
  362. throw service_load_exc(name, "cannot change starts_on_console/"
  363. "shares_console flags for a running service.");
  364. }
  365. // Cannot change pid file
  366. if (service->get_type() == service_type_t::BGPROCESS) {
  367. auto *bgp_service = static_cast<bgproc_service *>(service);
  368. if (bgp_service->get_pid_file() != settings.pid_file) {
  369. throw service_load_exc(name, "cannot change pid_file for running service.");
  370. }
  371. }
  372. // Cannot change inittab_id/inittab_line
  373. #if USE_UTMPX
  374. if (service->get_type() == service_type_t::PROCESS) {
  375. auto *proc_service = static_cast<process_service *>(service);
  376. auto *svc_utmp_id = proc_service->get_utmp_id();
  377. auto *svc_utmp_ln = proc_service->get_utmp_line();
  378. if (strncmp(svc_utmp_id, settings.inittab_id, proc_service->get_utmp_id_size()) != 0
  379. || strncmp(svc_utmp_ln, settings.inittab_line,
  380. proc_service->get_utmp_line_size()) != 0) {
  381. throw service_load_exc(name, "cannot change inittab-id or inittab-line "
  382. "settings for running service.");
  383. }
  384. }
  385. #endif
  386. // Already started; we must replace settings on existing service record
  387. create_new_record = false;
  388. }
  389. else if (service_type != service->get_type()) {
  390. // No need to create a new record if the type hasn't changed
  391. create_new_record = false;
  392. }
  393. }
  394. // Note, we need to be very careful to handle exceptions properly and roll back any changes that
  395. // we've made before the exception occurred.
  396. // if we have "before" constraints, check them now, before we potentially do irreversible changes
  397. // to an existing service.
  398. std::list<service_dep> before_deps;
  399. if (dummy == nullptr) {
  400. for (const std::string &before_ent : settings.before_svcs) {
  401. service_record *before_svc;
  402. try {
  403. before_svc = load_service(before_ent.c_str());
  404. }
  405. catch (service_description_exc &sle) {
  406. log_service_load_failure(sle);
  407. throw service_load_exc(name, "could not load dependency.");
  408. }
  409. catch (service_load_exc &sle) {
  410. log(loglevel_t::ERROR, "Could not load service ", sle.service_name, ": ",
  411. sle.exc_description);
  412. throw service_load_exc(name, "could not load dependency.");
  413. }
  414. before_deps.emplace_back(before_svc, reload_svc, dependency_type::BEFORE);
  415. // (note, we may need to adjust the to-service if we create a new service record object)
  416. check_cycle(settings.depends, before_svc);
  417. if (before_svc == reload_svc) {
  418. throw service_cyclic_dependency(before_svc->get_name());
  419. }
  420. }
  421. }
  422. else {
  423. // If we have a dummy service in place, we can't load "before" services since they
  424. // may depend on *this* service which is currently represented as a dummy, which would
  425. // trigger cycle detection.
  426. // So, we'll do it later in this case. We can also postpone if we'll be creating a
  427. // replacement service record rather than modifying the original.
  428. }
  429. if (service_type == service_type_t::PROCESS) {
  430. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  431. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  432. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  433. process_service *rvalps;
  434. if (create_new_record) {
  435. rvalps = new process_service(this, string(name), std::move(settings.command),
  436. settings.command_offsets, settings.depends);
  437. if (reload_svc != nullptr) {
  438. check_cycle(settings.depends, reload_svc);
  439. }
  440. }
  441. else {
  442. rvalps = static_cast<process_service *>(reload_svc);
  443. update_command_and_dependencies(rvalps, settings, before_deps);
  444. }
  445. rval = rvalps;
  446. // All of the following should be noexcept or must perform rollback on exception
  447. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  448. rvalps->set_working_dir(std::move(settings.working_dir));
  449. rvalps->set_env_file(std::move(settings.env_file));
  450. #if SUPPORT_CGROUPS
  451. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  452. #endif
  453. rvalps->set_rlimits(std::move(settings.rlimits));
  454. rvalps->set_restart_interval(settings.restart_interval, settings.max_restarts);
  455. rvalps->set_restart_delay(settings.restart_delay);
  456. rvalps->set_stop_timeout(settings.stop_timeout);
  457. rvalps->set_start_timeout(settings.start_timeout);
  458. rvalps->set_extra_termination_signal(settings.term_signal);
  459. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  460. rvalps->set_notification_fd(settings.readiness_fd);
  461. rvalps->set_notification_var(std::move(settings.readiness_var));
  462. rvalps->set_log_file(std::move(settings.logfile));
  463. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  464. rvalps->set_log_mode(settings.log_type);
  465. #if USE_UTMPX
  466. rvalps->set_utmp_id(settings.inittab_id);
  467. rvalps->set_utmp_line(settings.inittab_line);
  468. #endif
  469. }
  470. else if (service_type == service_type_t::BGPROCESS) {
  471. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  472. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  473. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  474. bgproc_service *rvalps;
  475. if (create_new_record) {
  476. rvalps = new bgproc_service(this, string(name), std::move(settings.command),
  477. settings.command_offsets, settings.depends);
  478. if (reload_svc != nullptr) {
  479. check_cycle(settings.depends, reload_svc);
  480. }
  481. }
  482. else {
  483. rvalps = static_cast<bgproc_service *>(reload_svc);
  484. update_command_and_dependencies(rvalps, settings, before_deps);
  485. }
  486. rval = rvalps;
  487. // All of the following should be noexcept or must perform rollback on exception
  488. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  489. rvalps->set_working_dir(std::move(settings.working_dir));
  490. rvalps->set_env_file(std::move(settings.env_file));
  491. #if SUPPORT_CGROUPS
  492. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  493. #endif
  494. rvalps->set_rlimits(std::move(settings.rlimits));
  495. rvalps->set_pid_file(std::move(settings.pid_file));
  496. rvalps->set_restart_interval(settings.restart_interval, settings.max_restarts);
  497. rvalps->set_restart_delay(settings.restart_delay);
  498. rvalps->set_stop_timeout(settings.stop_timeout);
  499. rvalps->set_start_timeout(settings.start_timeout);
  500. rvalps->set_extra_termination_signal(settings.term_signal);
  501. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  502. rvalps->set_log_file(std::move(settings.logfile));
  503. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  504. rvalps->set_log_mode(settings.log_type);
  505. settings.onstart_flags.runs_on_console = false;
  506. }
  507. else if (service_type == service_type_t::SCRIPTED) {
  508. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  509. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  510. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  511. scripted_service *rvalps;
  512. if (create_new_record) {
  513. rvalps = new scripted_service(this, string(name), std::move(settings.command),
  514. settings.command_offsets, settings.depends);
  515. if (reload_svc != nullptr) {
  516. check_cycle(settings.depends, reload_svc);
  517. }
  518. }
  519. else {
  520. rvalps = static_cast<scripted_service *>(reload_svc);
  521. update_command_and_dependencies(rvalps, settings, before_deps);
  522. }
  523. rval = rvalps;
  524. // All of the following should be noexcept or must perform rollback on exception
  525. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  526. rvalps->set_working_dir(std::move(settings.working_dir));
  527. rvalps->set_env_file(std::move(settings.env_file));
  528. #if SUPPORT_CGROUPS
  529. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  530. #endif
  531. rvalps->set_rlimits(std::move(settings.rlimits));
  532. rvalps->set_stop_timeout(settings.stop_timeout);
  533. rvalps->set_start_timeout(settings.start_timeout);
  534. rvalps->set_extra_termination_signal(settings.term_signal);
  535. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  536. rvalps->set_log_file(std::move(settings.logfile));
  537. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  538. rvalps->set_log_mode(settings.log_type);
  539. }
  540. else {
  541. if (create_new_record) {
  542. if (service_type == service_type_t::INTERNAL) {
  543. rval = new service_record(this, string(name), service_type, settings.depends);
  544. }
  545. else {
  546. /* TRIGGERED */
  547. rval = new triggered_service(this, string(name), service_type, settings.depends);
  548. }
  549. if (reload_svc != nullptr) {
  550. check_cycle(settings.depends, reload_svc);
  551. }
  552. }
  553. else {
  554. rval = reload_svc;
  555. update_depenencies(rval, settings, before_deps);
  556. }
  557. }
  558. rval->set_auto_restart(settings.auto_restart);
  559. rval->set_smooth_recovery(settings.smooth_recovery);
  560. rval->set_flags(settings.onstart_flags);
  561. rval->set_socket_details(std::move(settings.socket_path), settings.socket_perms,
  562. settings.socket_uid, settings.socket_gid);
  563. rval->set_chain_to(std::move(settings.chain_to_name));
  564. rval->set_environment(std::move(srv_env));
  565. if (create_new_record && reload_svc != nullptr) {
  566. // switch dependencies on old record so that they refer to the new record
  567. auto &dept_list = rval->get_dependents();
  568. for (auto &dept : before_deps) {
  569. dept_list.push_back(&dept);
  570. }
  571. // Add dependent-link for all dependencies. Add to the new service first, so we can rollback
  572. // on failure:
  573. int added_dep_links = 0;
  574. try {
  575. for (auto &dep : rval->get_dependencies()) {
  576. dep.get_to()->get_dependents().push_back(&dep);
  577. added_dep_links++;
  578. }
  579. }
  580. catch (...) {
  581. // exception caught; roll back any added dependencies and re-throw
  582. for (auto &dep : rval->get_dependencies()) {
  583. if (added_dep_links-- == 0) break;
  584. dep.get_to()->get_dependents().pop_back();
  585. }
  586. throw;
  587. }
  588. // --- Point of no return: mustn't fail from here ---
  589. // Remove all "before" dependents from the original service
  590. auto &reload_depts = reload_svc->get_dependents();
  591. for (auto i = reload_depts.begin(); i != reload_depts.end(); ) {
  592. auto next_i = std::next(i);
  593. if ((*i)->dep_type == dependency_type::BEFORE) {
  594. (*i)->get_from()->rm_dep(**i);
  595. }
  596. i = next_i;
  597. }
  598. // Transfer dependents from the original service record to the new record;
  599. // set links in all dependents on the original to point to the new service:
  600. auto first_new_before = dept_list.begin();
  601. dept_list.splice(first_new_before, reload_depts);
  602. for (auto &dept : dept_list) {
  603. dept->set_to(rval);
  604. }
  605. // Remove dependent-link for all dependencies from the original:
  606. reload_svc->prepare_for_unload();
  607. // Splice in the "before" dependencies
  608. auto i = before_deps.begin();
  609. decltype(i) j;
  610. while (i != before_deps.end()) {
  611. j = std::next(i);
  612. i->set_to(rval);
  613. auto &from_deps = i->get_from()->get_dependencies();
  614. from_deps.splice(from_deps.end(), before_deps, i);
  615. i = j;
  616. }
  617. }
  618. if (dummy != nullptr) {
  619. auto iter = std::find(records.begin(), records.end(), dummy);
  620. *iter = rval;
  621. delete dummy;
  622. // process before entries now. We must do it after "installing" the newly loaded service
  623. // in the service set (which we do just above) in order to avoid triggering the cycle
  624. // detection (in case the "before" service depends directly on this one) due to the dummy
  625. // service.
  626. auto ii = std::prev(rval->get_dependents().end());
  627. auto i = settings.before_svcs.begin();
  628. try {
  629. for ( ; i != settings.before_svcs.end(); ++i) {
  630. const std::string &before_ent = *i;
  631. service_record *before_svc = load_service(before_ent.c_str());
  632. before_svc->add_dep(rval, dependency_type::BEFORE);
  633. }
  634. }
  635. catch (...) {
  636. // undo if unsuccessful:
  637. for (auto j = std::next(ii); j != rval->get_dependents().end(); j = std::next(ii)) {
  638. (*j)->get_to()->rm_dep(**j);
  639. }
  640. dummy = nullptr;
  641. rval->prepare_for_unload();
  642. records.erase(std::find(records.begin(), records.end(), rval));
  643. throw;
  644. }
  645. }
  646. return rval;
  647. }
  648. catch (service_description_exc &setting_exc)
  649. {
  650. // Must remove the dummy service record.
  651. if (dummy != nullptr) {
  652. records.erase(std::find(records.begin(), records.end(), dummy));
  653. delete dummy;
  654. }
  655. if (create_new_record) delete rval;
  656. if (setting_exc.service_name.empty()) {
  657. setting_exc.service_name = name;
  658. }
  659. throw;
  660. }
  661. catch (std::system_error &sys_err)
  662. {
  663. if (dummy != nullptr) {
  664. records.erase(std::find(records.begin(), records.end(), dummy));
  665. delete dummy;
  666. }
  667. if (create_new_record) delete rval;
  668. throw service_load_exc(name, sys_err.what());
  669. }
  670. catch (...) // (should only be std::bad_alloc or service_load_exc)
  671. {
  672. if (dummy != nullptr) {
  673. records.erase(std::find(records.begin(), records.end(), dummy));
  674. delete dummy;
  675. }
  676. if (create_new_record) delete rval;
  677. throw;
  678. }
  679. }