load-service.cc 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864
  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. Report
  88. // any cycles as occurring in _report_svc_name_.
  89. static void check_cycle(service_dep_list &deps, service_record *orig, const std::string &report_svc_name)
  90. {
  91. linked_uo_set<service_record *> pending;
  92. for (auto &new_dep : deps) {
  93. if (new_dep.to == orig) {
  94. throw service_cyclic_dependency(report_svc_name);
  95. }
  96. pending.add_back(new_dep.to);
  97. }
  98. for (auto i = pending.begin(); i != pending.end(); ++i) {
  99. auto &dep_list = (*i)->get_dependencies();
  100. for (auto &dep : dep_list) {
  101. if (dep.get_to() == orig) {
  102. throw service_cyclic_dependency(report_svc_name);
  103. }
  104. pending.add_back(dep.get_to());
  105. }
  106. }
  107. }
  108. // Check for dependency cycles in "before" dependencies, _orig_ is where cycles will be identified.
  109. static void check_cycle(service_dep_list &deps, service_record *orig)
  110. {
  111. check_cycle(deps, orig, orig->get_name());
  112. }
  113. // Update the dependencies of the specified service atomically.
  114. // May fail with bad_alloc, service_cyclic_dependency.
  115. static void update_depenencies(service_record *service,
  116. dinit_load::service_settings_wrapper<prelim_dep> &settings,
  117. std::list<service_dep> &before_deps)
  118. {
  119. check_cycle(settings.depends, service);
  120. std::list<service_dep> &deps = service->get_dependencies();
  121. auto first_preexisting = deps.begin();
  122. auto &depts = service->get_dependents();
  123. auto first_pre_dept = depts.begin();
  124. try {
  125. // Insert all new dependents (from "before" relationships) before the first pre-existing dependent
  126. for (auto new_dept_i = before_deps.begin(); new_dept_i != before_deps.end(); ) {
  127. auto &new_dept = *new_dept_i;
  128. depts.insert(depts.begin(), &new_dept);
  129. // splice the dependency into the dependent:
  130. auto next_dept_i = std::next(new_dept_i);
  131. auto &from_deps = new_dept.get_from()->get_dependencies();
  132. from_deps.splice(from_deps.begin(), before_deps, new_dept_i);
  133. new_dept_i = next_dept_i;
  134. }
  135. // Insert all the new dependencies before the first pre-existing dependency
  136. for (auto i = settings.depends.begin(); i != settings.depends.end(); ) {
  137. auto &new_dep = *i;
  138. service->add_dep(new_dep.to, new_dep.dep_type, first_preexisting);
  139. i = settings.depends.erase(i);
  140. }
  141. }
  142. catch (...) {
  143. // remove "before" dependencies from dependents
  144. for (auto i = depts.begin(); i != first_pre_dept; ) {
  145. auto next_i = std::next(i);
  146. (*i)->get_from()->rm_dep(**i);
  147. i = next_i;
  148. }
  149. // remove the inserted dependencies
  150. for (auto i = deps.begin(); i != first_preexisting; ) {
  151. i = service->rm_dep(i);
  152. }
  153. // re-throw the exception
  154. throw;
  155. }
  156. // Now remove all pre-existing dependencies, except for "before" dependencies (which come from the
  157. // linked service and so must be retained)
  158. for( ; first_preexisting != deps.end(); ) {
  159. if (first_preexisting->dep_type != dependency_type::BEFORE) {
  160. first_preexisting = service->rm_dep(first_preexisting);
  161. }
  162. else {
  163. ++first_preexisting;
  164. }
  165. }
  166. // Also remove pre-existing "before" dependents (because they come from this service)
  167. for( ; first_pre_dept != depts.end(); ) {
  168. auto next_pre_dept = std::next(first_pre_dept);
  169. if ((*first_pre_dept)->dep_type == dependency_type::BEFORE) {
  170. (*first_pre_dept)->get_from()->rm_dep(**first_pre_dept);
  171. }
  172. first_pre_dept = next_pre_dept;
  173. }
  174. }
  175. // Update the command, and dependencies, of the specified service atomically.
  176. // May fail with bad_alloc, service_cyclic_dependency.
  177. static void update_command_and_dependencies(base_process_service *service,
  178. dinit_load::service_settings_wrapper<prelim_dep> &settings,
  179. std::list<service_dep> &before_deps)
  180. {
  181. // Get the current command parts
  182. ha_string orig_cmd; std::vector<const char *> orig_arg_parts;
  183. service->get_command(orig_cmd, orig_arg_parts);
  184. // Separate the new command parts and set
  185. std::vector<const char *> cmd_arg_parts = separate_args(settings.command, settings.command_offsets);
  186. service->set_command(std::move(settings.command), std::move(cmd_arg_parts));
  187. try {
  188. update_depenencies(service, settings, before_deps);
  189. }
  190. catch (...) {
  191. // restore original command
  192. service->set_command(std::move(orig_cmd), std::move(orig_arg_parts));
  193. // re-throw the exception
  194. throw;
  195. }
  196. }
  197. // Check that the provided settings are compatible / ok to be applied to the specified (already loaded)
  198. // service.
  199. // Returns: true if a new service record must be created, false otherwise
  200. static bool check_settings_for_reload(service_record *service,
  201. dinit_load::service_settings_wrapper<prelim_dep> &settings)
  202. {
  203. using namespace dinit_load;
  204. const std::string &name = service->get_name();
  205. // To begin, assume a new record is needed
  206. bool create_new_record = true;
  207. if (service->get_state() != service_state_t::STOPPED) {
  208. // Can not change type of a running service.
  209. if (settings.service_type != service->get_type()) {
  210. throw service_load_exc(name, "cannot change type of non-stopped service.");
  211. }
  212. // Can not alter a starting/stopping service, at least for now.
  213. if (service->get_state() != service_state_t::STARTED) {
  214. throw service_load_exc(name,
  215. "cannot alter settings for service which is currently starting/stopping.");
  216. }
  217. // Check validity of dependencies (if started, regular deps must be started)
  218. for (auto &new_dep : settings.depends) {
  219. if (new_dep.dep_type == dependency_type::REGULAR) {
  220. if (new_dep.to->get_state() != service_state_t::STARTED) {
  221. throw service_load_exc(name,
  222. std::string("cannot add non-started dependency '")
  223. + new_dep.to->get_name() + "'.");
  224. }
  225. }
  226. }
  227. // Cannot change certain flags
  228. auto current_flags = service->get_flags();
  229. if (current_flags.starts_on_console != settings.onstart_flags.starts_on_console
  230. || current_flags.shares_console != settings.onstart_flags.shares_console) {
  231. throw service_load_exc(name, "cannot change starts_on_console/"
  232. "shares_console flags for a running service.");
  233. }
  234. // Cannot change pid file
  235. if (service->get_type() == service_type_t::BGPROCESS) {
  236. auto *bgp_service = static_cast<bgproc_service *>(service);
  237. if (bgp_service->get_pid_file() != settings.pid_file) {
  238. throw service_load_exc(name, "cannot change pid_file for running service.");
  239. }
  240. }
  241. // Cannot change inittab_id/inittab_line
  242. #if USE_UTMPX
  243. if (service->get_type() == service_type_t::PROCESS) {
  244. auto *proc_service = static_cast<process_service *>(service);
  245. auto *svc_utmp_id = proc_service->get_utmp_id();
  246. auto *svc_utmp_ln = proc_service->get_utmp_line();
  247. if (strncmp(svc_utmp_id, settings.inittab_id, proc_service->get_utmp_id_size()) != 0
  248. || strncmp(svc_utmp_ln, settings.inittab_line,
  249. proc_service->get_utmp_line_size()) != 0) {
  250. throw service_load_exc(name, "cannot change inittab-id or inittab-line "
  251. "settings for running service.");
  252. }
  253. }
  254. #endif
  255. // Cannot change log type
  256. if (value(service->get_type()).is_in(service_type_t::PROCESS, service_type_t::BGPROCESS,
  257. service_type_t::SCRIPTED)) {
  258. base_process_service *bps = static_cast<base_process_service *>(service);
  259. if (bps->get_log_mode() != settings.log_type) {
  260. throw service_load_exc(name, "cannot change log-type for running service.");
  261. }
  262. }
  263. // Already started; we must replace settings on existing service record
  264. create_new_record = false;
  265. }
  266. else if (settings.service_type == service->get_type()) {
  267. // No need to create a new record if the type hasn't changed
  268. create_new_record = false;
  269. }
  270. return create_new_record;
  271. }
  272. service_record * dirload_service_set::load_reload_service(const char *name, service_record *reload_svc,
  273. const service_record *avoid_circular)
  274. {
  275. // Load a new service, or reload an already-loaded service.
  276. // For reload, we have the following problems:
  277. // - ideally want to allow changing service type, at least for stopped services. That implies creating
  278. // a new (replacement) service_record object, at least in cases where the type does change.
  279. // - dependencies may change (including addition of new dependencies which aren't yet loaded). We need
  280. // to prevent cyclic dependencies forming.
  281. // - We want atomicity. If any new settings are not valid/alterable, or if a cyclic dependency is
  282. // created, nothing should change.
  283. // - We need to either transfer handles referring to the old service (so that they refer to the new
  284. // service), or make them invalid. Or, we alter the original service without creating a new one
  285. // (which we can only do if the type doesn't change).
  286. // Approach:
  287. // - determine whether we need a new service record or can alter the existing one
  288. // (loading a new service always creates a new record; reload only creates a new record if the service
  289. // type changes, and otherwise just changes the existing record in-place).
  290. // - if loading a new service, a dummy record is created to enable easy cyclic dependency detection.
  291. // (In other cases cycles must be checked by walking the service graph).
  292. // The dummy is replaced with the real service once loading is complete (or is removed if it fails).
  293. // - process settings from the service file (into a service_settings_wrapper).
  294. // - check that the new settings are valid (for reload, if the service is running, check if the settings
  295. // can be altered).
  296. // - create the new record and install the new settings in it (or the existing record if not creating a
  297. // new record). If doing a reload, check for cycles at this point (there is no dummy record in this
  298. // case, so the quick cycle detection is not active).
  299. // - (if doing a reload, with a new record) move the dependents on the original record to the new record.
  300. //
  301. // "Before" dependencies require special handling, as a "before = " specified in a service actually creates
  302. // a dependency in the specified service on this service. Hence they always require explicit cycle checks
  303. // (the quick cycle detection method using a dummy service cannot be used). For reloads this is done early,
  304. // for new services it is done late (after the dummy has been removed).
  305. //
  306. // This is all an intricate dance. If failure occurs at any stage, we must restore the previous state.
  307. // Limitations:
  308. // - caller must check there are no handles (or only a single requesting handle) to the service before
  309. // calling
  310. // - cannot change the type of a non-stopped service
  311. using std::string;
  312. using std::ifstream;
  313. using std::ios;
  314. using std::ios_base;
  315. using std::locale;
  316. using std::isspace;
  317. using std::list;
  318. using std::pair;
  319. using namespace dinit_load;
  320. if (reload_svc == nullptr) {
  321. // First try and find an existing record...
  322. service_record *existing = find_service(string(name), true);
  323. if (existing != nullptr) {
  324. if (existing == avoid_circular || existing->check_is_loading()) {
  325. throw service_cyclic_dependency(name);
  326. }
  327. if (existing->get_type() != service_type_t::PLACEHOLDER) {
  328. return existing;
  329. }
  330. // If we found a placeholder, we proceed as for a reload:
  331. reload_svc = existing;
  332. }
  333. }
  334. service_record *rval = nullptr;
  335. service_record *dummy = nullptr;
  336. ifstream service_file;
  337. string service_filename;
  338. int fail_load_errno = 0;
  339. std::string fail_load_path;
  340. // Couldn't find one. Have to load it.
  341. for (auto &service_dir : service_dirs) {
  342. service_filename = service_dir.get_dir();
  343. if (*(service_filename.rbegin()) != '/') {
  344. service_filename += '/';
  345. }
  346. service_filename += name;
  347. service_file.open(service_filename.c_str(), ios::in);
  348. if (service_file) break;
  349. if (errno != ENOENT && fail_load_errno == 0) {
  350. fail_load_errno = errno;
  351. fail_load_path = std::move(service_filename);
  352. }
  353. }
  354. if (!service_file) {
  355. if (fail_load_errno == 0) {
  356. throw service_not_found(string(name));
  357. }
  358. else {
  359. throw service_load_error(name, std::move(fail_load_path), fail_load_errno);
  360. }
  361. }
  362. service_settings_wrapper<prelim_dep> settings;
  363. string line;
  364. // getline can set failbit if it reaches end-of-file, we don't want an exception in that case. There's
  365. // no good way to handle an I/O error however, so we'll have exceptions thrown on badbit:
  366. service_file.exceptions(ios::badbit);
  367. bool create_new_record = true;
  368. // any "before" "dependencies" that were loaded
  369. std::list<service_dep> before_deps;
  370. auto exception_cleanup = [&]() {
  371. // Must remove the dummy service record.
  372. if (dummy != nullptr) {
  373. remove_service(dummy);
  374. delete dummy;
  375. }
  376. if (create_new_record && rval != nullptr) {
  377. rval->prepare_for_unload();
  378. delete rval;
  379. }
  380. for (service_dep &before_dep : before_deps) {
  381. service_record *before_svc = before_dep.get_from();
  382. if (before_svc->get_type() == service_type_t::PLACEHOLDER) {
  383. if (before_svc->is_unrefd()) {
  384. remove_service(before_svc);
  385. delete before_svc;
  386. }
  387. }
  388. }
  389. // Remove any "after" placeholders that were created while loading but not successfully added as
  390. // dependencies on the new service (rval). (This requires that settings.depends has been cleared
  391. // of any dependencies that were successfully added).
  392. for (prelim_dep &dep : settings.depends) {
  393. if (dep.dep_type == dependency_type::AFTER && dep.to->is_unrefd()) {
  394. remove_service(dep.to);
  395. }
  396. }
  397. };
  398. if (reload_svc == nullptr) {
  399. // Add a placeholder record now to prevent infinite recursion in case of cyclic dependency.
  400. // We replace this with the real service later (or remove it if we find a configuration error).
  401. try {
  402. dummy = new service_record(this, string(name), service_record::LOADING_TAG);
  403. add_service(dummy);
  404. }
  405. catch (...) {
  406. delete dummy; // (no effect if dummy is null)
  407. dummy = nullptr;
  408. throw;
  409. }
  410. }
  411. try {
  412. process_service_file(name, service_file,
  413. [&](string &line, unsigned line_num, string &setting,
  414. string_iterator &i, string_iterator &end) -> void {
  415. auto process_dep_dir_n = [&](std::list<prelim_dep> &deplist, const std::string &waitsford,
  416. dependency_type dep_type) -> void {
  417. process_dep_dir(*this, name, service_filename, deplist, waitsford, dep_type, reload_svc);
  418. };
  419. auto load_service_n = [&](const string &dep_name) -> service_record * {
  420. try {
  421. return load_service(dep_name.c_str(), reload_svc);
  422. }
  423. catch (service_description_exc &sle) {
  424. log_service_load_failure(sle);
  425. throw service_load_exc(name, "could not load dependency.");
  426. }
  427. catch (service_load_exc &sle) {
  428. log(loglevel_t::ERROR, "Could not load service ", sle.service_name, ": ",
  429. sle.exc_description);
  430. throw service_load_exc(name, "could not load dependency.");
  431. }
  432. };
  433. process_service_line(settings, name, line, line_num, setting, i, end, load_service_n,
  434. process_dep_dir_n);
  435. });
  436. service_file.close();
  437. auto report_err = [&](const char *msg){
  438. throw service_load_exc(name, msg);
  439. };
  440. environment srv_env;
  441. /* fill user vars before reading env file */
  442. if (settings.export_passwd_vars) {
  443. fill_environment_userinfo(settings.run_as_uid, name, srv_env);
  444. }
  445. /* set service name in environment if desired */
  446. if (settings.export_service_name) {
  447. std::string envname = "DINIT_SERVICE=";
  448. envname += name;
  449. srv_env.set_var(std::move(envname));
  450. }
  451. // this mapping is temporary, for load substitutions
  452. // the reason for this is that the environment actually *may* change
  453. // after load, e.g. through dinitctl setenv (either from the outside
  454. // or from within services) and we want this to refresh for each
  455. // process invocation
  456. environment::env_map srv_envmap;
  457. if (!settings.env_file.empty()) {
  458. try {
  459. read_env_file(settings.env_file.data(), false, srv_env);
  460. } catch (const std::system_error &se) {
  461. throw service_load_exc(name, std::string("could not load environment file: ") + se.what());
  462. }
  463. }
  464. srv_envmap = srv_env.build(main_env);
  465. settings.finalise(report_err, srv_envmap);
  466. auto service_type = settings.service_type;
  467. if (reload_svc != nullptr) {
  468. // Make sure settings are able to be changed/are compatible
  469. create_new_record = check_settings_for_reload(reload_svc, settings);
  470. }
  471. // Note, we need to be very careful to handle exceptions properly and roll back any changes that
  472. // we've made before the exception occurred.
  473. // If we have "after" constraints, load them now and treat them as regular dependencies. We need
  474. // to do this now, after the other dependents are loaded, because we might create a placeholder
  475. // instead (and we don't want to create a placeholder, have it added to the list of dependencies,
  476. // then load the same service as a real dependency shortly afterwards, which would replace the
  477. // placeholder but leave a dangling pointer to it in the list).
  478. for (const std::string &after_ent : settings.after_svcs) {
  479. service_record *after_svc;
  480. if (after_ent == name) throw service_cyclic_dependency(name);
  481. after_svc = find_service(after_ent.c_str(), true);
  482. if (after_svc != nullptr) {
  483. if (after_svc->check_is_loading()) {
  484. throw service_cyclic_dependency(name);
  485. }
  486. }
  487. if (after_svc == nullptr) {
  488. after_svc = new service_record(this, after_ent);
  489. try {
  490. add_service(after_svc);
  491. }
  492. catch (...) {
  493. delete after_svc;
  494. throw;
  495. }
  496. }
  497. try {
  498. settings.depends.emplace_back(after_svc, dependency_type::AFTER);
  499. }
  500. catch (...) {
  501. if (after_svc->is_unrefd()) {
  502. remove_service(after_svc);
  503. delete after_svc;
  504. throw;
  505. }
  506. }
  507. }
  508. // if we have "before" constraints, check them now.
  509. for (const std::string &before_ent : settings.before_svcs) {
  510. service_record *before_svc;
  511. if (before_ent == name) throw service_cyclic_dependency(name);
  512. before_svc = find_service(before_ent.c_str(), true);
  513. if (before_svc != nullptr) {
  514. check_cycle(settings.depends, before_svc, name);
  515. }
  516. else {
  517. bool before_svc_added = false;
  518. try {
  519. before_svc = new service_record(this, before_ent);
  520. add_service(before_svc);
  521. before_svc_added = true;
  522. }
  523. catch (...) {
  524. if (before_svc_added) remove_service(before_svc);
  525. delete before_svc;
  526. throw;
  527. }
  528. before_deps.emplace_back(before_svc, reload_svc, dependency_type::BEFORE);
  529. // (note, we may need to adjust the to-service if we create a new service record object
  530. // - this will be done later)
  531. }
  532. }
  533. if (service_type == service_type_t::PROCESS) {
  534. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  535. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  536. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  537. process_service *rvalps;
  538. if (create_new_record) {
  539. rvalps = new process_service(this, string(name), std::move(settings.command),
  540. settings.command_offsets, settings.depends);
  541. settings.depends.clear();
  542. if (reload_svc != nullptr) {
  543. check_cycle(settings.depends, reload_svc);
  544. }
  545. }
  546. else {
  547. rvalps = static_cast<process_service *>(reload_svc);
  548. update_command_and_dependencies(rvalps, settings, before_deps);
  549. }
  550. rval = rvalps;
  551. // All of the following should be noexcept or must perform rollback on exception
  552. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  553. rvalps->set_working_dir(std::move(settings.working_dir));
  554. rvalps->set_env_file(std::move(settings.env_file));
  555. #if SUPPORT_CGROUPS
  556. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  557. #endif
  558. rvalps->set_rlimits(std::move(settings.rlimits));
  559. rvalps->set_restart_interval(settings.restart_interval, settings.max_restarts);
  560. rvalps->set_restart_delay(settings.restart_delay);
  561. rvalps->set_stop_timeout(settings.stop_timeout);
  562. rvalps->set_start_timeout(settings.start_timeout);
  563. rvalps->set_extra_termination_signal(settings.term_signal);
  564. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  565. rvalps->set_notification_fd(settings.readiness_fd);
  566. rvalps->set_notification_var(std::move(settings.readiness_var));
  567. rvalps->set_log_file(std::move(settings.logfile));
  568. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  569. rvalps->set_log_mode(settings.log_type);
  570. #if USE_UTMPX
  571. rvalps->set_utmp_id(settings.inittab_id);
  572. rvalps->set_utmp_line(settings.inittab_line);
  573. #endif
  574. }
  575. else if (service_type == service_type_t::BGPROCESS) {
  576. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  577. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  578. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  579. bgproc_service *rvalps;
  580. if (create_new_record) {
  581. rvalps = new bgproc_service(this, string(name), std::move(settings.command),
  582. settings.command_offsets, settings.depends);
  583. settings.depends.clear();
  584. if (reload_svc != nullptr) {
  585. check_cycle(settings.depends, reload_svc);
  586. }
  587. }
  588. else {
  589. rvalps = static_cast<bgproc_service *>(reload_svc);
  590. update_command_and_dependencies(rvalps, settings, before_deps);
  591. }
  592. rval = rvalps;
  593. // All of the following should be noexcept or must perform rollback on exception
  594. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  595. rvalps->set_working_dir(std::move(settings.working_dir));
  596. rvalps->set_env_file(std::move(settings.env_file));
  597. #if SUPPORT_CGROUPS
  598. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  599. #endif
  600. rvalps->set_rlimits(std::move(settings.rlimits));
  601. rvalps->set_pid_file(std::move(settings.pid_file));
  602. rvalps->set_restart_interval(settings.restart_interval, settings.max_restarts);
  603. rvalps->set_restart_delay(settings.restart_delay);
  604. rvalps->set_stop_timeout(settings.stop_timeout);
  605. rvalps->set_start_timeout(settings.start_timeout);
  606. rvalps->set_extra_termination_signal(settings.term_signal);
  607. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  608. rvalps->set_log_file(std::move(settings.logfile));
  609. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  610. rvalps->set_log_mode(settings.log_type);
  611. settings.onstart_flags.runs_on_console = false;
  612. }
  613. else if (service_type == service_type_t::SCRIPTED) {
  614. do_env_subst("command", settings.command, settings.command_offsets, srv_envmap);
  615. do_env_subst("stop-command", settings.stop_command, settings.stop_command_offsets, srv_envmap);
  616. std::vector<const char *> stop_arg_parts = separate_args(settings.stop_command, settings.stop_command_offsets);
  617. scripted_service *rvalps;
  618. if (create_new_record) {
  619. rvalps = new scripted_service(this, string(name), std::move(settings.command),
  620. settings.command_offsets, settings.depends);
  621. settings.depends.clear();
  622. if (reload_svc != nullptr) {
  623. check_cycle(settings.depends, reload_svc);
  624. }
  625. }
  626. else {
  627. rvalps = static_cast<scripted_service *>(reload_svc);
  628. update_command_and_dependencies(rvalps, settings, before_deps);
  629. }
  630. rval = rvalps;
  631. // All of the following should be noexcept or must perform rollback on exception
  632. rvalps->set_stop_command(std::move(settings.stop_command), std::move(stop_arg_parts));
  633. rvalps->set_working_dir(std::move(settings.working_dir));
  634. rvalps->set_env_file(std::move(settings.env_file));
  635. #if SUPPORT_CGROUPS
  636. rvalps->set_cgroup(std::move(settings.run_in_cgroup));
  637. #endif
  638. rvalps->set_rlimits(std::move(settings.rlimits));
  639. rvalps->set_stop_timeout(settings.stop_timeout);
  640. rvalps->set_start_timeout(settings.start_timeout);
  641. rvalps->set_extra_termination_signal(settings.term_signal);
  642. rvalps->set_run_as_uid_gid(settings.run_as_uid, settings.run_as_gid);
  643. rvalps->set_log_file(std::move(settings.logfile));
  644. rvalps->set_log_buf_max(settings.max_log_buffer_sz);
  645. rvalps->set_log_mode(settings.log_type);
  646. }
  647. else {
  648. if (create_new_record) {
  649. if (service_type == service_type_t::INTERNAL) {
  650. rval = new service_record(this, string(name), service_type, settings.depends);
  651. }
  652. else {
  653. /* TRIGGERED */
  654. rval = new triggered_service(this, string(name), service_type, settings.depends);
  655. }
  656. settings.depends.clear();
  657. if (reload_svc != nullptr) {
  658. check_cycle(settings.depends, reload_svc);
  659. }
  660. }
  661. else {
  662. rval = reload_svc;
  663. update_depenencies(rval, settings, before_deps);
  664. }
  665. }
  666. rval->set_auto_restart(settings.auto_restart);
  667. rval->set_smooth_recovery(settings.smooth_recovery);
  668. rval->set_flags(settings.onstart_flags);
  669. rval->set_socket_details(std::move(settings.socket_path), settings.socket_perms,
  670. settings.socket_uid, settings.socket_gid);
  671. rval->set_chain_to(std::move(settings.chain_to_name));
  672. rval->set_environment(std::move(srv_env));
  673. if (create_new_record) {
  674. // switch dependencies on old record so that they refer to the new record
  675. // first link in all the (new) "before" dependents (one way at this stage):
  676. auto &dept_list = rval->get_dependents();
  677. unsigned added_dependents = 0;
  678. try {
  679. for (auto &dept : before_deps) {
  680. dept_list.push_back(&dept);
  681. ++added_dependents;
  682. }
  683. }
  684. catch (...) {
  685. // Undo, since the invalid state will cause issues when the new service is disposed of
  686. while (added_dependents > 0) {
  687. dept_list.pop_back();
  688. --added_dependents;
  689. }
  690. throw;
  691. }
  692. // --- Point of no return: mustn't fail from here ---
  693. // Splice in the new "before" dependencies
  694. auto i = before_deps.begin();
  695. decltype(i) j;
  696. while (i != before_deps.end()) {
  697. j = std::next(i);
  698. i->set_to(rval);
  699. auto &from_deps = i->get_from()->get_dependencies();
  700. from_deps.splice(from_deps.end(), before_deps, i);
  701. i = j;
  702. }
  703. if (reload_svc != nullptr) {
  704. // Complete dependency/dependent transfers.
  705. // Remove all "before" dependents from the original service (these were created by the
  706. // original service itself)
  707. auto &reload_depts = reload_svc->get_dependents();
  708. for (auto i = reload_depts.begin(); i != reload_depts.end(); ) {
  709. auto next_i = std::next(i);
  710. if ((*i)->dep_type == dependency_type::BEFORE) {
  711. service_record *before_svc = (*i)->get_from();
  712. before_svc->rm_dep(**i);
  713. if (before_svc->get_type() == service_type_t::PLACEHOLDER && before_svc->is_unrefd()) {
  714. remove_service(before_svc);
  715. delete before_svc;
  716. }
  717. }
  718. i = next_i;
  719. }
  720. // Transfer dependents from the original service record to the new record;
  721. // set links in all dependents on the original to point to the new service:
  722. auto first_new_before = dept_list.begin();
  723. dept_list.splice(first_new_before, reload_depts);
  724. for (auto &dept : dept_list) {
  725. dept->set_to(rval);
  726. }
  727. // Transfer all "before" dependencies (which are actually created by other services) to the
  728. // new service
  729. auto &dep_list_prev = reload_svc->get_dependencies();
  730. auto &dep_list = rval->get_dependencies();
  731. for (auto i = dep_list_prev.begin(); i != dep_list_prev.end(); ++i) {
  732. if (i->dep_type == dependency_type::BEFORE) {
  733. i->set_from(rval);
  734. dep_list.splice(dep_list.end(), dep_list_prev, i++);
  735. continue;
  736. }
  737. }
  738. // Remove dependent-link for all dependencies from the original:
  739. reload_svc->prepare_for_unload();
  740. }
  741. // Finally, replace the old service with the new one:
  742. service_record *old_service = (reload_svc != nullptr) ? reload_svc : dummy;
  743. auto iter = std::find(records.begin(), records.end(), old_service);
  744. *iter = rval;
  745. delete old_service;
  746. }
  747. return rval;
  748. }
  749. catch (service_description_exc &setting_exc)
  750. {
  751. exception_cleanup();
  752. if (setting_exc.service_name.empty()) {
  753. setting_exc.service_name = name;
  754. }
  755. throw;
  756. }
  757. catch (std::system_error &sys_err)
  758. {
  759. exception_cleanup();
  760. throw service_load_exc(name, sys_err.what());
  761. }
  762. catch (...) // (should only be std::bad_alloc or service_load_exc)
  763. {
  764. exception_cleanup();
  765. throw;
  766. }
  767. }