2023-08-09 20:39:32 -07:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.shb.authelia;
|
|
|
|
|
2024-01-11 23:22:46 -08:00
|
|
|
contracts = pkgs.callPackage ../contracts {};
|
2024-02-09 20:56:26 -08:00
|
|
|
shblib = pkgs.callPackage ../../lib {};
|
2024-01-11 23:22:46 -08:00
|
|
|
|
2023-08-09 20:39:32 -07:00
|
|
|
fqdn = "${cfg.subdomain}.${cfg.domain}";
|
2024-01-21 23:38:57 -08:00
|
|
|
fqdnWithPort = if isNull cfg.port then fqdn else "${fqdn}:${toString cfg.port}";
|
2023-08-09 20:39:32 -07:00
|
|
|
|
|
|
|
autheliaCfg = config.services.authelia.instances.${fqdn};
|
|
|
|
in
|
|
|
|
{
|
|
|
|
options.shb.authelia = {
|
|
|
|
enable = lib.mkEnableOption "selfhostblocks.authelia";
|
|
|
|
|
|
|
|
subdomain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Subdomain under which Authelia will be served.";
|
2023-11-30 10:38:35 -08:00
|
|
|
example = "auth";
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
domain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "domain under which Authelia will be served.";
|
|
|
|
example = "mydomain.com";
|
|
|
|
};
|
|
|
|
|
2024-01-21 23:38:57 -08:00
|
|
|
port = lib.mkOption {
|
|
|
|
description = "If given, adds a port to the `<subdomain>.<domain>` endpoint.";
|
|
|
|
type = lib.types.nullOr lib.types.port;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
2024-01-11 23:22:46 -08:00
|
|
|
ssl = lib.mkOption {
|
|
|
|
description = "Path to SSL files";
|
|
|
|
type = lib.types.nullOr contracts.ssl.certs;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
2023-08-09 20:39:32 -07:00
|
|
|
ldapEndpoint = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Endpoint for LDAP authentication backend.";
|
|
|
|
example = "ldap.example.com";
|
|
|
|
};
|
|
|
|
|
|
|
|
dcdomain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "dc domain for ldap.";
|
|
|
|
example = "dc=mydomain,dc=com";
|
|
|
|
};
|
|
|
|
|
2023-11-07 00:35:27 -08:00
|
|
|
autheliaUser = lib.mkOption {
|
2023-11-30 22:49:34 -08:00
|
|
|
type = lib.types.str;
|
|
|
|
description = "System user for this Authelia instance.";
|
|
|
|
default = "authelia";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
secrets = lib.mkOption {
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "Secrets needed by Authelia";
|
2023-11-07 00:35:27 -08:00
|
|
|
type = lib.types.submodule {
|
|
|
|
options = {
|
|
|
|
jwtSecretFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "File containing the JWT secret.";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
ldapAdminPasswordFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "File containing the LDAP admin user password.";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
sessionSecretFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "File containing the session secret.";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
storageEncryptionKeyFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "File containing the storage encryption key.";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
identityProvidersOIDCHMACSecretFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2023-12-04 00:33:16 -08:00
|
|
|
description = "File containing the identity provider OIDC HMAC secret.";
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
identityProvidersOIDCIssuerPrivateKeyFile = lib.mkOption {
|
2023-12-10 21:56:31 -08:00
|
|
|
type = lib.types.path;
|
2024-01-21 23:38:57 -08:00
|
|
|
description = ''
|
|
|
|
File containing the identity provider OIDC issuer private key.
|
|
|
|
|
|
|
|
Generate one with `nix run nixpkgs#openssl -- genrsa -out keypair.pem 2048`
|
|
|
|
'';
|
2023-11-07 00:35:27 -08:00
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
oidcClients = lib.mkOption {
|
|
|
|
description = "OIDC clients";
|
2024-05-24 15:00:31 -07:00
|
|
|
default = [
|
|
|
|
{
|
|
|
|
id = "dummy_client";
|
|
|
|
description = "Dummy Client so Authelia can start";
|
|
|
|
secret.source = pkgs.writeText "dummy.secret" "dummy_client_secret";
|
|
|
|
public = false;
|
|
|
|
authorization_policy = "one_factor";
|
|
|
|
redirect_uris = [];
|
|
|
|
}
|
|
|
|
];
|
2024-02-29 15:34:53 -08:00
|
|
|
type = lib.types.listOf (lib.types.submodule {
|
|
|
|
freeformType = lib.types.attrsOf lib.types.anything;
|
|
|
|
|
|
|
|
options = {
|
|
|
|
id = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Unique identifier of the OIDC client.";
|
|
|
|
};
|
|
|
|
|
|
|
|
description = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
|
|
|
description = "Human readable description of the OIDC client.";
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
secret = lib.mkOption {
|
|
|
|
type = shblib.secretFileType;
|
|
|
|
description = "File containing the shared secret with the OIDC client.";
|
|
|
|
};
|
|
|
|
|
|
|
|
public = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
|
|
|
description = "If the OIDC client is public or not.";
|
|
|
|
default = false;
|
|
|
|
apply = v: if v then "true" else "false";
|
|
|
|
};
|
|
|
|
|
|
|
|
authorization_policy = lib.mkOption {
|
|
|
|
type = lib.types.enum [ "one_factor" "two_factor" ];
|
|
|
|
description = "Require one factor (password) or two factor (device) authentication.";
|
|
|
|
default = "one_factor";
|
|
|
|
};
|
|
|
|
|
|
|
|
redirect_uris = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
description = "List of uris that are allowed to be redirected to.";
|
|
|
|
};
|
|
|
|
|
|
|
|
scopes = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
description = "Scopes to ask for";
|
|
|
|
example = [ "openid" "profile" "email" "groups" ];
|
|
|
|
default = [];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
2023-12-10 21:56:31 -08:00
|
|
|
smtp = lib.mkOption {
|
2024-01-21 23:38:57 -08:00
|
|
|
description = ''
|
|
|
|
If a string is given, writes notifications to the given path.Otherwise, send notifications
|
|
|
|
by smtp.
|
|
|
|
|
|
|
|
https://www.authelia.com/configuration/notifications/introduction/
|
|
|
|
'';
|
|
|
|
default = "/tmp/authelia-notifications";
|
|
|
|
type = lib.types.oneOf [
|
|
|
|
lib.types.str
|
|
|
|
(lib.types.nullOr (lib.types.submodule {
|
|
|
|
options = {
|
|
|
|
from_address = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "SMTP address from which the emails originate.";
|
|
|
|
example = "authelia@mydomain.com";
|
|
|
|
};
|
|
|
|
from_name = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "SMTP name from which the emails originate.";
|
|
|
|
default = "Authelia";
|
|
|
|
};
|
|
|
|
host = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "SMTP host to send the emails to.";
|
|
|
|
};
|
|
|
|
port = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
|
|
|
description = "SMTP port to send the emails to.";
|
|
|
|
default = 25;
|
|
|
|
};
|
|
|
|
username = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Username to connect to the SMTP host.";
|
|
|
|
};
|
|
|
|
passwordFile = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "File containing the password to connect to the SMTP host.";
|
|
|
|
};
|
2023-12-10 21:56:31 -08:00
|
|
|
};
|
2024-01-21 23:38:57 -08:00
|
|
|
}))
|
|
|
|
];
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
2023-08-06 22:36:31 -07:00
|
|
|
|
|
|
|
rules = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.anything;
|
|
|
|
description = "Rule based clients";
|
|
|
|
default = [];
|
|
|
|
};
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
config = lib.mkIf cfg.enable {
|
2023-12-10 21:56:31 -08:00
|
|
|
assertions = [
|
|
|
|
{
|
|
|
|
assertion = builtins.length cfg.oidcClients > 0;
|
|
|
|
message = "Must have at least one oidc client otherwise Authelia refuses to start.";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
2023-08-09 20:39:32 -07:00
|
|
|
# Overriding the user name so we don't allow any weird characters anywhere. For example, postgres users do not accept the '.'.
|
|
|
|
users = {
|
|
|
|
groups.${autheliaCfg.user} = {};
|
|
|
|
users.${autheliaCfg.user} = {
|
|
|
|
isSystemUser = true;
|
|
|
|
group = autheliaCfg.user;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
services.authelia.instances.${fqdn} = {
|
|
|
|
enable = true;
|
2023-11-30 22:49:34 -08:00
|
|
|
user = cfg.autheliaUser;
|
2023-08-09 20:39:32 -07:00
|
|
|
|
|
|
|
secrets = {
|
2023-11-07 00:35:27 -08:00
|
|
|
inherit (cfg.secrets) jwtSecretFile storageEncryptionKeyFile;
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
# See https://www.authelia.com/configuration/methods/secrets/
|
|
|
|
environmentVariables = {
|
2023-12-10 21:56:31 -08:00
|
|
|
AUTHELIA_AUTHENTICATION_BACKEND_LDAP_PASSWORD_FILE = toString cfg.secrets.ldapAdminPasswordFile;
|
|
|
|
AUTHELIA_SESSION_SECRET_FILE = toString cfg.secrets.sessionSecretFile;
|
2023-08-09 20:39:32 -07:00
|
|
|
# Not needed since we use peer auth.
|
|
|
|
# AUTHELIA_STORAGE_POSTGRES_PASSWORD_FILE = "/run/secrets/authelia/postgres_password";
|
2023-12-10 21:56:31 -08:00
|
|
|
AUTHELIA_STORAGE_ENCRYPTION_KEY_FILE = toString cfg.secrets.storageEncryptionKeyFile;
|
|
|
|
AUTHELIA_IDENTITY_PROVIDERS_OIDC_HMAC_SECRET_FILE = toString cfg.secrets.identityProvidersOIDCHMACSecretFile;
|
|
|
|
AUTHELIA_IDENTITY_PROVIDERS_OIDC_ISSUER_PRIVATE_KEY_FILE = toString cfg.secrets.identityProvidersOIDCIssuerPrivateKeyFile;
|
|
|
|
|
2024-01-21 23:38:57 -08:00
|
|
|
AUTHELIA_NOTIFIER_SMTP_PASSWORD_FILE = lib.mkIf (!(builtins.isString cfg.smtp)) (toString cfg.smtp.passwordFile);
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
settings = {
|
|
|
|
server.host = "127.0.0.1";
|
|
|
|
server.port = 9091;
|
|
|
|
|
|
|
|
# Inspired from https://github.com/lldap/lldap/blob/7d1f5abc137821c500de99c94f7579761fc949d8/example_configs/authelia_config.yml
|
|
|
|
authentication_backend = {
|
|
|
|
refresh_interval = "5m";
|
|
|
|
password_reset = {
|
|
|
|
disable = "false";
|
|
|
|
};
|
|
|
|
ldap = {
|
|
|
|
implementation = "custom";
|
|
|
|
url = cfg.ldapEndpoint;
|
|
|
|
timeout = "5s";
|
|
|
|
start_tls = "false";
|
|
|
|
base_dn = cfg.dcdomain;
|
|
|
|
username_attribute = "uid";
|
|
|
|
additional_users_dn = "ou=people";
|
|
|
|
# Sign in with username or email.
|
|
|
|
users_filter = "(&(|({username_attribute}={input})({mail_attribute}={input}))(objectClass=person))";
|
|
|
|
additional_groups_dn = "ou=groups";
|
|
|
|
groups_filter = "(member={dn})";
|
|
|
|
group_name_attribute = "cn";
|
|
|
|
mail_attribute = "mail";
|
|
|
|
display_name_attribute = "displayName";
|
|
|
|
user = "uid=admin,ou=people,${cfg.dcdomain}";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
totp = {
|
|
|
|
disable = "false";
|
2024-01-21 23:38:57 -08:00
|
|
|
issuer = fqdnWithPort;
|
2023-08-09 20:39:32 -07:00
|
|
|
algorithm = "sha1";
|
|
|
|
digits = "6";
|
|
|
|
period = "30";
|
|
|
|
skew = "1";
|
|
|
|
secret_size = "32";
|
|
|
|
};
|
|
|
|
# Inspired from https://www.authelia.com/configuration/session/introduction/ and https://www.authelia.com/configuration/session/redis
|
|
|
|
session = {
|
|
|
|
name = "authelia_session";
|
2024-01-21 23:38:57 -08:00
|
|
|
domain = if isNull cfg.port then cfg.domain else "${cfg.domain}:${toString cfg.port}";
|
2023-08-09 20:39:32 -07:00
|
|
|
same_site = "lax";
|
|
|
|
expiration = "1h";
|
|
|
|
inactivity = "5m";
|
|
|
|
remember_me_duration = "1M";
|
|
|
|
redis = {
|
|
|
|
host = config.services.redis.servers.authelia.unixSocket;
|
|
|
|
port = 0;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
storage = {
|
|
|
|
postgres = {
|
|
|
|
host = "/run/postgresql";
|
|
|
|
username = autheliaCfg.user;
|
|
|
|
database = autheliaCfg.user;
|
|
|
|
port = config.services.postgresql.port;
|
|
|
|
# Uses peer auth for local users, so we don't need a password.
|
|
|
|
password = "test";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
notifier = {
|
2024-01-21 23:38:57 -08:00
|
|
|
filesystem = lib.mkIf (builtins.isString cfg.smtp) {
|
|
|
|
filename = cfg.smtp;
|
2023-12-10 21:56:31 -08:00
|
|
|
};
|
2024-01-21 23:38:57 -08:00
|
|
|
smtp = lib.mkIf (!(builtins.isString cfg.smtp)) {
|
2023-12-10 21:56:31 -08:00
|
|
|
host = cfg.smtp.host;
|
|
|
|
port = cfg.smtp.port;
|
|
|
|
username = cfg.smtp.username;
|
|
|
|
sender = "${cfg.smtp.from_name} <${cfg.smtp.from_address}>";
|
2023-08-09 20:39:32 -07:00
|
|
|
subject = "[Authelia] {title}";
|
|
|
|
startup_check_address = "test@authelia.com";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
access_control = {
|
|
|
|
default_policy = "deny";
|
|
|
|
networks = [
|
|
|
|
{
|
|
|
|
name = "internal";
|
|
|
|
networks = [ "10.0.0.0/8" "172.16.0.0/12" "192.168.0.0/18" ];
|
|
|
|
}
|
|
|
|
];
|
|
|
|
rules = [
|
|
|
|
{
|
2024-01-21 23:38:57 -08:00
|
|
|
domain = fqdnWithPort;
|
2023-08-09 00:09:56 -07:00
|
|
|
policy = "bypass";
|
|
|
|
resources = [
|
|
|
|
"^/api/.*"
|
|
|
|
];
|
2023-08-09 20:39:32 -07:00
|
|
|
}
|
2023-08-09 00:09:56 -07:00
|
|
|
] ++ cfg.rules;
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
telemetry = {
|
|
|
|
metrics = {
|
|
|
|
enabled = true;
|
|
|
|
address = "tcp://127.0.0.1:9959";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2023-12-10 21:56:31 -08:00
|
|
|
|
2024-01-21 08:17:25 +01:00
|
|
|
settingsFiles = [ "/var/lib/authelia-${fqdn}/oidc_clients.yaml" ];
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
2023-12-10 21:56:31 -08:00
|
|
|
systemd.services."authelia-${fqdn}".preStart =
|
|
|
|
let
|
2024-01-21 08:17:25 +01:00
|
|
|
mkCfg = clients:
|
2024-02-29 15:34:53 -08:00
|
|
|
shblib.replaceSecrets {
|
|
|
|
userConfig = {
|
|
|
|
identity_providers.oidc.clients = clients;
|
|
|
|
};
|
|
|
|
resultPath = "/var/lib/authelia-${fqdn}/oidc_clients.yaml";
|
2024-05-23 14:28:08 -07:00
|
|
|
generator = shblib.replaceSecretsGeneratorAdapter (lib.generators.toYAML {});
|
2024-02-29 15:34:53 -08:00
|
|
|
};
|
2023-12-10 21:56:31 -08:00
|
|
|
in
|
2024-01-21 08:17:25 +01:00
|
|
|
lib.mkBefore (mkCfg cfg.oidcClients);
|
2023-12-10 21:56:31 -08:00
|
|
|
|
2023-08-09 20:39:32 -07:00
|
|
|
services.nginx.virtualHosts.${fqdn} = {
|
2024-01-11 23:22:46 -08:00
|
|
|
forceSSL = !(isNull cfg.ssl);
|
|
|
|
sslCertificate = lib.mkIf (!(isNull cfg.ssl)) cfg.ssl.paths.cert;
|
|
|
|
sslCertificateKey = lib.mkIf (!(isNull cfg.ssl)) cfg.ssl.paths.key;
|
2023-09-02 15:00:41 -07:00
|
|
|
# Taken from https://github.com/authelia/authelia/issues/178
|
|
|
|
# TODO: merge with config from https://matwick.ca/authelia-nginx-sso/
|
|
|
|
locations."/".extraConfig = ''
|
|
|
|
add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always;
|
|
|
|
add_header X-Content-Type-Options nosniff;
|
|
|
|
add_header X-Frame-Options "SAMEORIGIN";
|
|
|
|
add_header X-XSS-Protection "1; mode=block";
|
|
|
|
add_header X-Robots-Tag "noindex, nofollow, nosnippet, noarchive";
|
|
|
|
add_header X-Download-Options noopen;
|
|
|
|
add_header X-Permitted-Cross-Domain-Policies none;
|
2023-08-09 20:39:32 -07:00
|
|
|
|
2023-09-02 15:00:41 -07:00
|
|
|
proxy_set_header Host $host;
|
|
|
|
proxy_set_header X-Real-IP $remote_addr;
|
|
|
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
|
|
|
proxy_set_header X-Forwarded-Proto $scheme;
|
|
|
|
proxy_http_version 1.1;
|
|
|
|
proxy_set_header Upgrade $http_upgrade;
|
|
|
|
proxy_set_header Connection "upgrade";
|
|
|
|
proxy_cache_bypass $http_upgrade;
|
|
|
|
|
|
|
|
proxy_pass http://127.0.0.1:${toString autheliaCfg.settings.server.port};
|
|
|
|
proxy_intercept_errors on;
|
|
|
|
if ($request_method !~ ^(POST)$){
|
|
|
|
error_page 401 = /error/401;
|
|
|
|
error_page 403 = /error/403;
|
|
|
|
error_page 404 = /error/404;
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
|
|
|
|
locations."/api/verify".extraConfig = ''
|
|
|
|
add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always;
|
|
|
|
add_header X-Content-Type-Options nosniff;
|
|
|
|
add_header X-Frame-Options "SAMEORIGIN";
|
|
|
|
add_header X-XSS-Protection "1; mode=block";
|
|
|
|
add_header X-Robots-Tag "noindex, nofollow, nosnippet, noarchive";
|
|
|
|
add_header X-Download-Options noopen;
|
|
|
|
add_header X-Permitted-Cross-Domain-Policies none;
|
|
|
|
|
|
|
|
proxy_set_header Host $http_x_forwarded_host;
|
|
|
|
proxy_pass http://127.0.0.1:${toString autheliaCfg.settings.server.port};
|
|
|
|
'';
|
2023-08-09 20:39:32 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
services.redis.servers.authelia = {
|
|
|
|
enable = true;
|
|
|
|
user = autheliaCfg.user;
|
|
|
|
};
|
|
|
|
|
2023-11-23 01:03:33 -08:00
|
|
|
shb.postgresql.ensures = [
|
2023-11-05 16:37:50 -08:00
|
|
|
{
|
|
|
|
username = autheliaCfg.user;
|
|
|
|
database = autheliaCfg.user;
|
|
|
|
}
|
|
|
|
];
|
2023-08-09 20:39:32 -07:00
|
|
|
|
|
|
|
services.prometheus.scrapeConfigs = [
|
|
|
|
{
|
|
|
|
job_name = "authelia";
|
|
|
|
static_configs = [
|
|
|
|
{
|
|
|
|
targets = ["127.0.0.1:9959"];
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|
|
|
|
];
|
|
|
|
};
|
|
|
|
}
|