Skip to content

Commit

Permalink
Manage unit running under systemd --user instance
Browse files Browse the repository at this point in the history
New defined type `systemd::user_service` has two roles:

It can globally enable a user unit for all users. e.g:

```puppet
systemd::user_service { 'systemd-tmpfiles-clean.service':
  enable => true,
  global => true,
}
````

If can start/stop/enable/disable a service running for a particular
user. e.g

systemd::user_service { 'ssh-agent.socket':
  ensure => 'running',
  enable => true,
  user   => 'steve',
}

The type instance can also be notified to reload the unit running under
a `systemd --user` instance.

```puppet
file{ '/home/steve/.gpg.conf':
  ensure  => file,
  content => "custom',
  notify  => Systemd::User_service['steve-gpg-agent.socket']
}

systemd::user_service { 'steve-gpg-agent.socket':
  ensure => true,
  enable => true,
  user   => 'steve',
}
```

Note - This merge request does not attempt to notify a
`systemd::daemon-reload{'user': user => 'steve'} instance at
the appropriate points. Solve that with real world experience.
  • Loading branch information
traylenator committed Apr 7, 2024
1 parent fdf1274 commit f9b414b
Show file tree
Hide file tree
Showing 4 changed files with 388 additions and 1 deletion.
98 changes: 98 additions & 0 deletions REFERENCE.md
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
* [`systemd::tmpfile`](#systemd--tmpfile): Creates a systemd tmpfile
* [`systemd::udev::rule`](#systemd--udev--rule): Adds a custom udev rule
* [`systemd::unit_file`](#systemd--unit_file): Creates a systemd unit file
* [`systemd::user_service`](#systemd--user_service): Manage a user service running under systemd --user

### Resource types

Expand Down Expand Up @@ -2251,6 +2252,103 @@ restart (notify) the service when unit file changed

Default value: `true`

### <a name="systemd--user_service"></a>`systemd::user_service`

Manage a user service running under systemd --user

#### Examples

##### Enable a service for all users

```puppet
systemd::user_service { 'systemd-tmpfiles-clean.timer':
enable => true,
global => true,
}
```

##### Enable a particular user's service

```puppet
systemd::user_service { 'podman-auto-update.timer':
ensure => true,
enable => true,
user => 'steve',
}
```

##### Notify a user's service to restart it

```puppet
file{ '/home/steve/.config/systemd/user/podman-auto-update.timer':
ensure => file,
content => ...,
notify => Systemd::User_service['steve-podman-auto-update.timer']
}
systemd::user_service { 'steve-podman-auto-update.timer':
enable => true,
unit => 'podman-auto-update.timer',
user => 'steve',
}
@param unit Unit name to work on
@param ensure Should the unit be started or stopped. Can only be true if user is specified.
@param enable Should the unit be enabled or disabled
@param user User name of user whose unit should be acted upon. Mutually exclusive with
@param global Act globally for all users. Mutually exclusibe with `user`.
```

#### Parameters

The following parameters are available in the `systemd::user_service` defined type:

* [`unit`](#-systemd--user_service--unit)
* [`ensure`](#-systemd--user_service--ensure)
* [`enable`](#-systemd--user_service--enable)
* [`global`](#-systemd--user_service--global)
* [`user`](#-systemd--user_service--user)

##### <a name="-systemd--user_service--unit"></a>`unit`

Data type: `Systemd::Unit`



Default value: `$title`

##### <a name="-systemd--user_service--ensure"></a>`ensure`

Data type: `Variant[Boolean,Enum['stopped','running']]`



Default value: `false`

##### <a name="-systemd--user_service--enable"></a>`enable`

Data type: `Boolean`



Default value: `false`

##### <a name="-systemd--user_service--global"></a>`global`

Data type: `Boolean`



Default value: `false`

##### <a name="-systemd--user_service--user"></a>`user`

Data type: `Optional[String[1]]`



Default value: `undef`

## Resource types

### <a name="loginctl_user"></a>`loginctl_user`
Expand Down
135 changes: 135 additions & 0 deletions manifests/user_service.pp
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
# @summary Manage a user service running under systemd --user
#
# @example Enable a service for all users
# systemd::user_service { 'systemd-tmpfiles-clean.timer':
# enable => true,
# global => true,
# }
#
# @example Enable a particular user's service
# systemd::user_service { 'podman-auto-update.timer':
# ensure => true,
# enable => true,
# user => 'steve',
# }
#
# @example Notify a user's service to restart it
# file{ '/home/steve/.config/systemd/user/podman-auto-update.timer':
# ensure => file,
# content => ...,
# notify => Systemd::User_service['steve-podman-auto-update.timer']
# }
#
# systemd::user_service { 'steve-podman-auto-update.timer':
# enable => true,
# unit => 'podman-auto-update.timer',
# user => 'steve',
# }
#
# @param unit Unit name to work on
# @param ensure Should the unit be started or stopped. Can only be true if user is specified.
# @param enable Should the unit be enabled or disabled
# @param user User name of user whose unit should be acted upon. Mutually exclusive with
# @param global Act globally for all users. Mutually exclusibe with `user`.
#
define systemd::user_service (
Systemd::Unit $unit = $title,
Variant[Boolean,Enum['stopped','running']] $ensure = false,
Boolean $enable = false,
Boolean $global = false,
Optional[String[1]] $user = undef,
) {
$_ensure = $ensure ? {
'stopped' => false,
'running' => true,
default => $ensure
}

if ( $global and $user ) or ( ! $global and ! $user) {
fail('Exactly one of the "user" or "global" parameters must be defined')
}

if $global and $_ensure {
fail('Cannot ensure a service is running for all users globally')
}

if $global {
if $enable {
$_title = "Enable user service ${unit} globally"
$_command = ['systemctl', '--global', 'enable', $unit]
$_unless = [['systemctl', '--global', 'is-enabled', $unit]]
$_onlyif = undef
} else {
$_title = "Disable user service ${unit} globally"
$_command = ['systemctl', '--global', 'disable', $unit]
$_unless = undef
$_onlyif = [['systemctl', '--global', 'is-enabled', $unit]]
}
exec { $_title:
command => $_command,
unless => $_unless,
onlyif => $_onlyif,
path => $facts['path'],
}
} else { # per user services

$_systemctl_user = [
'systemd-run', '--pipe', '--wait', '--user', '--machine', "${user}@.host",
'systemctl', '--user',
]

# To accept notifies of this type.
exec { "try-reload-or-restart-${user}-${unit}":
command => $_systemctl_user + ['try-reload-or-restart', $unit],
refreshonly => true,
path => $facts['path'],
}

if $_ensure {
$_ensure_title = "Start user service ${unit} for user ${user}"
$_ensure_command = $_systemctl_user + ['start', $unit]
$_ensure_unless = [$_systemctl_user + ['is-active', $unit]]
$_ensure_onlyif = undef

# Don't reload just after starting
Exec["try-reload-or-restart-${user}-${unit}"] -> Exec[$_ensure_title]
} else {
$_ensure_title = "Stop user service ${unit} for user ${user}"
$_ensure_command = $_systemctl_user + ['stop', $unit]
$_ensure_unless = undef
$_ensure_onlyif = [$_systemctl_user + ['is-active', $unit]]
}

exec { $_ensure_title:
command => $_ensure_command,
unless => $_ensure_unless,
onlyif => $_ensure_onlyif,
path => $facts['path'],
}

if $enable {
$_enable_title = "Enable user service ${unit} for user ${user}"
$_enable_command = $_systemctl_user + ['enable', $unit]
$_enable_unless = [$_systemctl_user + ['is-enabled', $unit]]
$_enable_onlyif = undef

# Puppet does this for services so lets copy that logic
# don't enable if you can't start.
if $_ensure {
Exec["Start user service ${unit} for user ${user}"] -> Exec[$_enable_title]
}
} else {
$_enable_title = "Disable user service ${unit} for user ${user}"
$_enable_command = $_systemctl_user + ['disable', $unit]
$_enable_unless = undef
$_enable_onlyif = [$_systemctl_user + ['is-enabled', $unit]]
}

exec { $_enable_title:
command => $_enable_command,
unless => $_enable_unless,
onlyif => $_enable_onlyif,
path => $facts['path'],
}
}
}
2 changes: 1 addition & 1 deletion metadata.json
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@
"requirements": [
{
"name": "puppet",
"version_requirement": ">= 7.0.0 < 9.0.0"
"version_requirement": ">= 7.9.0 < 9.0.0"
}
]
}
Loading

0 comments on commit f9b414b

Please sign in to comment.