2015-08-05 11:02:46 +10:00
|
|
|
#include <stdio.h>
|
2015-08-05 11:30:40 +10:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <wlc/wlc.h>
|
2015-08-13 17:44:56 +10:00
|
|
|
#include <sys/wait.h>
|
2015-09-18 21:16:20 +10:00
|
|
|
#include <sys/types.h>
|
2016-12-02 11:27:35 +11:00
|
|
|
#include <sys/stat.h>
|
2015-10-19 04:53:56 +11:00
|
|
|
#include <sys/un.h>
|
2015-08-13 17:44:56 +10:00
|
|
|
#include <signal.h>
|
2015-12-13 05:01:00 +11:00
|
|
|
#include <unistd.h>
|
2015-08-20 22:37:09 +10:00
|
|
|
#include <getopt.h>
|
2016-12-03 10:37:01 +11:00
|
|
|
#include <sys/capability.h>
|
2016-09-01 22:18:37 +10:00
|
|
|
#include "sway/extensions.h"
|
|
|
|
#include "sway/layout.h"
|
|
|
|
#include "sway/config.h"
|
2016-12-03 00:42:26 +11:00
|
|
|
#include "sway/security.h"
|
2016-09-01 22:18:37 +10:00
|
|
|
#include "sway/handlers.h"
|
|
|
|
#include "sway/input.h"
|
|
|
|
#include "sway/ipc-server.h"
|
2015-12-13 05:01:00 +11:00
|
|
|
#include "ipc-client.h"
|
2016-09-01 22:18:37 +10:00
|
|
|
#include "readline.h"
|
|
|
|
#include "stringop.h"
|
2015-08-20 23:12:34 +10:00
|
|
|
#include "sway.h"
|
2016-09-01 22:18:37 +10:00
|
|
|
#include "log.h"
|
2015-08-20 23:12:34 +10:00
|
|
|
|
|
|
|
static bool terminate_request = false;
|
2016-02-26 19:08:05 +11:00
|
|
|
static int exit_value = 0;
|
2015-08-20 23:12:34 +10:00
|
|
|
|
2016-02-26 19:08:05 +11:00
|
|
|
void sway_terminate(int exit_code) {
|
2015-08-20 23:12:34 +10:00
|
|
|
terminate_request = true;
|
2016-02-26 19:08:05 +11:00
|
|
|
exit_value = exit_code;
|
2015-08-20 23:12:34 +10:00
|
|
|
wlc_terminate();
|
|
|
|
}
|
2015-08-05 11:02:46 +10:00
|
|
|
|
2015-12-29 23:00:35 +11:00
|
|
|
void sig_handler(int signal) {
|
|
|
|
close_views(&root_container);
|
2016-02-26 19:08:05 +11:00
|
|
|
sway_terminate(EXIT_SUCCESS);
|
2015-12-29 23:00:35 +11:00
|
|
|
}
|
|
|
|
|
2015-08-24 03:08:04 +10:00
|
|
|
static void wlc_log_handler(enum wlc_log_type type, const char *str) {
|
|
|
|
if (type == WLC_LOG_ERROR) {
|
2015-08-24 03:31:16 +10:00
|
|
|
sway_log(L_ERROR, "[wlc] %s", str);
|
2015-08-24 03:08:04 +10:00
|
|
|
} else if (type == WLC_LOG_WARN) {
|
2015-08-24 03:31:16 +10:00
|
|
|
sway_log(L_INFO, "[wlc] %s", str);
|
2015-08-24 03:08:04 +10:00
|
|
|
} else {
|
2015-08-24 03:31:16 +10:00
|
|
|
sway_log(L_DEBUG, "[wlc] %s", str);
|
2015-08-24 03:08:04 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-15 03:13:44 +11:00
|
|
|
void detect_proprietary() {
|
2015-09-03 01:46:21 +10:00
|
|
|
FILE *f = fopen("/proc/modules", "r");
|
|
|
|
if (!f) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (!feof(f)) {
|
|
|
|
char *line = read_line(f);
|
|
|
|
if (strstr(line, "nvidia")) {
|
|
|
|
fprintf(stderr, "\x1B[1;31mWarning: Proprietary nvidia drivers do NOT support Wayland. Use nouveau.\x1B[0m\n");
|
2016-03-25 07:48:42 +11:00
|
|
|
fprintf(stderr, "\x1B[1;31mYes, they STILL don't work with the newly announced wayland \"support\".\x1B[0m\n");
|
2015-09-03 01:46:21 +10:00
|
|
|
free(line);
|
2015-09-03 01:47:15 +10:00
|
|
|
break;
|
2015-09-03 01:46:21 +10:00
|
|
|
}
|
2015-12-15 03:13:44 +11:00
|
|
|
if (strstr(line, "fglrx")) {
|
|
|
|
fprintf(stderr, "\x1B[1;31mWarning: Proprietary AMD drivers do NOT support Wayland. Use radeon.\x1B[0m\n");
|
|
|
|
free(line);
|
|
|
|
break;
|
|
|
|
}
|
2015-09-03 01:46:21 +10:00
|
|
|
free(line);
|
|
|
|
}
|
2015-09-03 01:47:15 +10:00
|
|
|
fclose(f);
|
2015-09-03 01:46:21 +10:00
|
|
|
}
|
|
|
|
|
2015-12-13 05:01:00 +11:00
|
|
|
void run_as_ipc_client(char *command, char *socket_path) {
|
|
|
|
int socketfd = ipc_open_socket(socket_path);
|
|
|
|
uint32_t len = strlen(command);
|
|
|
|
char *resp = ipc_single_command(socketfd, IPC_COMMAND, command, &len);
|
|
|
|
printf("%s\n", resp);
|
|
|
|
close(socketfd);
|
|
|
|
}
|
|
|
|
|
2016-10-28 01:37:16 +11:00
|
|
|
static void log_env() {
|
|
|
|
const char *log_vars[] = {
|
|
|
|
"PATH",
|
|
|
|
"LD_LOAD_PATH",
|
|
|
|
"LD_PRELOAD_PATH",
|
2016-10-28 02:05:04 +11:00
|
|
|
"LD_LIBRARY_PATH",
|
2016-10-28 01:37:16 +11:00
|
|
|
"SWAY_CURSOR_THEME",
|
|
|
|
"SWAY_CURSOR_SIZE",
|
|
|
|
"SWAYSOCK",
|
|
|
|
"WLC_DRM_DEVICE",
|
|
|
|
"WLC_SHM",
|
|
|
|
"WLC_OUTPUTS",
|
|
|
|
"WLC_XWAYLAND",
|
|
|
|
"WLC_LIBINPUT",
|
|
|
|
"WLC_REPEAT_DELAY",
|
|
|
|
"WLC_REPEAT_RATE",
|
|
|
|
"XKB_DEFAULT_RULES",
|
|
|
|
"XKB_DEFAULT_MODEL",
|
|
|
|
"XKB_DEFAULT_LAYOUT",
|
|
|
|
"XKB_DEFAULT_VARIANT",
|
|
|
|
"XKB_DEFAULT_OPTIONS",
|
|
|
|
};
|
|
|
|
for (size_t i = 0; i < sizeof(log_vars) / sizeof(char *); ++i) {
|
|
|
|
sway_log(L_INFO, "%s=%s", log_vars[i], getenv(log_vars[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-28 01:48:46 +11:00
|
|
|
static void log_distro() {
|
|
|
|
const char *paths[] = {
|
|
|
|
"/etc/lsb-release",
|
|
|
|
"/etc/os-release",
|
|
|
|
"/etc/debian_version",
|
|
|
|
"/etc/redhat-release",
|
|
|
|
"/etc/gentoo-release",
|
|
|
|
};
|
|
|
|
for (size_t i = 0; i < sizeof(paths) / sizeof(char *); ++i) {
|
|
|
|
FILE *f = fopen(paths[i], "r");
|
|
|
|
if (f) {
|
|
|
|
sway_log(L_INFO, "Contents of %s:", paths[i]);
|
|
|
|
while (!feof(f)) {
|
|
|
|
char *line = read_line(f);
|
|
|
|
if (*line) {
|
|
|
|
sway_log(L_INFO, "%s", line);
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-28 01:50:22 +11:00
|
|
|
static void log_kernel() {
|
|
|
|
FILE *f = popen("uname -a", "r");
|
|
|
|
if (!f) {
|
|
|
|
sway_log(L_INFO, "Unable to determine kernel version");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (!feof(f)) {
|
|
|
|
char *line = read_line(f);
|
|
|
|
if (*line) {
|
|
|
|
sway_log(L_INFO, "%s", line);
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
2016-12-02 11:27:35 +11:00
|
|
|
static void security_sanity_check() {
|
|
|
|
// TODO: Notify users visually if this has issues
|
2016-12-02 13:58:38 +11:00
|
|
|
struct stat s;
|
2016-12-02 11:27:35 +11:00
|
|
|
if (stat("/proc", &s)) {
|
|
|
|
sway_log(L_ERROR,
|
|
|
|
"!! DANGER !! /proc is not available - sway CANNOT enforce security rules!");
|
|
|
|
}
|
2016-12-08 23:34:08 +11:00
|
|
|
#ifdef __linux__
|
2016-12-03 10:37:01 +11:00
|
|
|
cap_flag_value_t v;
|
|
|
|
cap_t cap = cap_get_proc();
|
|
|
|
if (!cap || cap_get_flag(cap, CAP_SYS_PTRACE, CAP_PERMITTED, &v) != 0 || v != CAP_SET) {
|
|
|
|
sway_log(L_ERROR,
|
|
|
|
"!! DANGER !! Sway does not have CAP_SYS_PTRACE and cannot enforce security rules for processes running as other users.");
|
|
|
|
}
|
|
|
|
if (cap) {
|
|
|
|
cap_free(cap);
|
|
|
|
}
|
2016-12-08 23:34:08 +11:00
|
|
|
#endif
|
2016-12-02 11:27:35 +11:00
|
|
|
if (!stat(SYSCONFDIR "/sway", &s)) {
|
2016-12-03 00:42:26 +11:00
|
|
|
if (s.st_uid != 0 || s.st_gid != 0
|
|
|
|
|| (s.st_mode & S_IWGRP) || (s.st_mode & S_IWOTH)) {
|
2016-12-02 11:27:35 +11:00
|
|
|
sway_log(L_ERROR,
|
2016-12-03 00:42:26 +11:00
|
|
|
"!! DANGER !! " SYSCONFDIR "/sway is not secure! It should be owned by root and set to 0755 at the minimum");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
struct {
|
|
|
|
char *command;
|
|
|
|
enum command_context context;
|
|
|
|
bool checked;
|
|
|
|
} expected[] = {
|
|
|
|
{ "reload", CONTEXT_BINDING, false },
|
|
|
|
{ "permit", CONTEXT_CONFIG, false },
|
|
|
|
{ "reject", CONTEXT_CONFIG, false },
|
|
|
|
{ "ipc", CONTEXT_CONFIG, false },
|
|
|
|
};
|
2016-12-05 02:55:11 +11:00
|
|
|
int expected_len = 4;
|
2016-12-03 00:42:26 +11:00
|
|
|
for (int i = 0; i < config->command_policies->length; ++i) {
|
|
|
|
struct command_policy *policy = config->command_policies->items[i];
|
|
|
|
for (int j = 0; j < expected_len; ++j) {
|
|
|
|
if (strcmp(expected[j].command, policy->command) == 0) {
|
|
|
|
expected[j].checked = true;
|
|
|
|
if (expected[j].context != policy->context) {
|
|
|
|
sway_log(L_ERROR,
|
|
|
|
"!! DANGER !! Command security policy for %s should be set to %s",
|
|
|
|
expected[j].command, command_policy_str(expected[j].context));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int j = 0; j < expected_len; ++j) {
|
|
|
|
if (!expected[j].checked) {
|
|
|
|
sway_log(L_ERROR,
|
|
|
|
"!! DANGER !! Command security policy for %s should be set to %s",
|
|
|
|
expected[j].command, command_policy_str(expected[j].context));
|
2016-12-02 11:27:35 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 11:30:40 +10:00
|
|
|
int main(int argc, char **argv) {
|
2015-08-20 22:37:09 +10:00
|
|
|
static int verbose = 0, debug = 0, validate = 0;
|
|
|
|
|
|
|
|
static struct option long_options[] = {
|
2015-11-27 06:03:21 +11:00
|
|
|
{"help", no_argument, NULL, 'h'},
|
2015-08-20 22:37:09 +10:00
|
|
|
{"config", required_argument, NULL, 'c'},
|
2015-11-29 01:35:44 +11:00
|
|
|
{"validate", no_argument, NULL, 'C'},
|
|
|
|
{"debug", no_argument, NULL, 'd'},
|
2015-08-20 22:37:09 +10:00
|
|
|
{"version", no_argument, NULL, 'v'},
|
2015-11-29 01:35:44 +11:00
|
|
|
{"verbose", no_argument, NULL, 'V'},
|
2015-08-20 22:37:09 +10:00
|
|
|
{"get-socketpath", no_argument, NULL, 'p'},
|
2015-08-27 04:01:26 +10:00
|
|
|
{0, 0, 0, 0}
|
2015-08-20 22:37:09 +10:00
|
|
|
};
|
|
|
|
|
|
|
|
char *config_path = NULL;
|
2015-11-27 06:01:37 +11:00
|
|
|
|
|
|
|
const char* usage =
|
|
|
|
"Usage: sway [options] [command]\n"
|
|
|
|
"\n"
|
2015-11-27 06:03:21 +11:00
|
|
|
" -h, --help Show help message and quit.\n"
|
2015-11-27 06:01:37 +11:00
|
|
|
" -c, --config <config> Specify a config file.\n"
|
|
|
|
" -C, --validate Check the validity of the config file, then exit.\n"
|
|
|
|
" -d, --debug Enables full logging, including debug information.\n"
|
|
|
|
" -v, --version Show the version number and quit.\n"
|
|
|
|
" -V, --verbose Enables more verbose logging.\n"
|
|
|
|
" --get-socketpath Gets the IPC socket path and prints it, then exits.\n"
|
|
|
|
"\n";
|
|
|
|
|
2016-12-03 02:23:30 +11:00
|
|
|
// Security:
|
|
|
|
unsetenv("LD_PRELOAD");
|
|
|
|
setenv("LD_LIBRARY_PATH", _LD_LIBRARY_PATH, 1);
|
2016-12-03 00:47:03 +11:00
|
|
|
|
2015-08-20 22:37:09 +10:00
|
|
|
int c;
|
|
|
|
while (1) {
|
|
|
|
int option_index = 0;
|
2016-02-26 07:49:53 +11:00
|
|
|
c = getopt_long(argc, argv, "hCdvVc:", long_options, &option_index);
|
2015-08-20 22:37:09 +10:00
|
|
|
if (c == -1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (c) {
|
2015-11-27 06:03:21 +11:00
|
|
|
case 'h': // help
|
|
|
|
fprintf(stdout, "%s", usage);
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
break;
|
2015-08-20 22:37:09 +10:00
|
|
|
case 'c': // config
|
|
|
|
config_path = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 'C': // validate
|
|
|
|
validate = 1;
|
|
|
|
break;
|
|
|
|
case 'd': // debug
|
|
|
|
debug = 1;
|
|
|
|
break;
|
|
|
|
case 'v': // version
|
2015-08-27 12:13:53 +10:00
|
|
|
#if defined SWAY_GIT_VERSION && defined SWAY_GIT_BRANCH && defined SWAY_VERSION_DATE
|
|
|
|
fprintf(stdout, "sway version %s (%s, branch \"%s\")\n", SWAY_GIT_VERSION, SWAY_VERSION_DATE, SWAY_GIT_BRANCH);
|
2015-08-26 13:04:57 +10:00
|
|
|
#else
|
|
|
|
fprintf(stdout, "version not detected\n");
|
|
|
|
#endif
|
2015-11-29 00:47:44 +11:00
|
|
|
exit(EXIT_SUCCESS);
|
2015-08-20 22:37:09 +10:00
|
|
|
break;
|
|
|
|
case 'V': // verbose
|
|
|
|
verbose = 1;
|
|
|
|
break;
|
2015-10-19 04:53:56 +11:00
|
|
|
case 'p': ; // --get-socketpath
|
2015-11-14 03:53:46 +11:00
|
|
|
if (getenv("SWAYSOCK")) {
|
|
|
|
fprintf(stdout, "%s\n", getenv("SWAYSOCK"));
|
2015-11-29 00:47:44 +11:00
|
|
|
exit(EXIT_SUCCESS);
|
2015-11-14 03:53:46 +11:00
|
|
|
} else {
|
|
|
|
fprintf(stderr, "sway socket not detected.\n");
|
2015-11-29 00:47:44 +11:00
|
|
|
exit(EXIT_FAILURE);
|
2015-11-14 03:53:46 +11:00
|
|
|
}
|
2015-08-20 22:37:09 +10:00
|
|
|
break;
|
2015-11-27 06:01:37 +11:00
|
|
|
default:
|
|
|
|
fprintf(stderr, "%s", usage);
|
|
|
|
exit(EXIT_FAILURE);
|
2015-08-20 22:37:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-13 05:01:00 +11:00
|
|
|
if (optind < argc) { // Behave as IPC client
|
2016-02-26 08:19:33 +11:00
|
|
|
if(optind != 1) {
|
2016-02-26 19:08:05 +11:00
|
|
|
sway_log(L_ERROR, "Don't use options with the IPC client");
|
|
|
|
exit(EXIT_FAILURE);
|
2016-02-26 08:19:33 +11:00
|
|
|
}
|
2015-12-13 05:01:00 +11:00
|
|
|
if (getuid() != geteuid() || getgid() != getegid()) {
|
2016-09-20 23:49:16 +10:00
|
|
|
if (setgid(getgid()) != 0) {
|
2016-02-26 19:08:05 +11:00
|
|
|
sway_log(L_ERROR, "Unable to drop root");
|
|
|
|
exit(EXIT_FAILURE);
|
2015-12-13 05:01:00 +11:00
|
|
|
}
|
2016-09-20 23:49:16 +10:00
|
|
|
if (setuid(getuid()) != 0) {
|
|
|
|
sway_log(L_ERROR, "Unable to drop root");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (setuid(0) != -1) {
|
|
|
|
sway_log(L_ERROR, "Root privileges can be restored.");
|
|
|
|
exit(EXIT_FAILURE);
|
2015-12-13 05:01:00 +11:00
|
|
|
}
|
|
|
|
char *socket_path = getenv("SWAYSOCK");
|
|
|
|
if (!socket_path) {
|
2016-02-26 19:08:05 +11:00
|
|
|
sway_log(L_ERROR, "Unable to retrieve socket path");
|
|
|
|
exit(EXIT_FAILURE);
|
2015-12-13 05:01:00 +11:00
|
|
|
}
|
|
|
|
char *command = join_args(argv + optind, argc - optind);
|
|
|
|
run_as_ipc_client(command, socket_path);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-11-30 00:56:33 +11:00
|
|
|
// we need to setup logging before wlc_init in case it fails.
|
|
|
|
if (debug) {
|
|
|
|
init_log(L_DEBUG);
|
|
|
|
} else if (verbose || validate) {
|
|
|
|
init_log(L_INFO);
|
|
|
|
} else {
|
|
|
|
init_log(L_ERROR);
|
|
|
|
}
|
2015-11-29 05:52:28 +11:00
|
|
|
wlc_log_set_handler(wlc_log_handler);
|
2015-12-15 03:13:44 +11:00
|
|
|
detect_proprietary();
|
2015-11-29 05:52:28 +11:00
|
|
|
|
2016-01-17 21:53:37 +11:00
|
|
|
input_devices = create_list();
|
|
|
|
|
2015-11-29 05:52:28 +11:00
|
|
|
/* Changing code earlier than this point requires detailed review */
|
|
|
|
/* (That code runs as root on systems without logind, and wlc_init drops to
|
|
|
|
* another user.) */
|
2016-03-25 05:08:53 +11:00
|
|
|
register_wlc_handlers();
|
2016-04-17 00:22:50 +10:00
|
|
|
if (!wlc_init()) {
|
2015-11-29 05:52:28 +11:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
register_extensions();
|
|
|
|
|
2015-12-29 23:00:35 +11:00
|
|
|
// handle SIGTERM signals
|
|
|
|
signal(SIGTERM, sig_handler);
|
|
|
|
|
2016-01-22 12:29:18 +11:00
|
|
|
// prevent ipc from crashing sway
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
2015-09-02 23:42:27 +10:00
|
|
|
#if defined SWAY_GIT_VERSION && defined SWAY_GIT_BRANCH && defined SWAY_VERSION_DATE
|
|
|
|
sway_log(L_INFO, "Starting sway version %s (%s, branch \"%s\")\n", SWAY_GIT_VERSION, SWAY_VERSION_DATE, SWAY_GIT_BRANCH);
|
|
|
|
#endif
|
2016-10-28 01:50:22 +11:00
|
|
|
log_kernel();
|
2016-10-28 01:48:46 +11:00
|
|
|
log_distro();
|
2016-10-28 01:50:22 +11:00
|
|
|
log_env();
|
2015-09-02 23:42:27 +10:00
|
|
|
|
2016-01-06 05:16:46 +11:00
|
|
|
init_layout();
|
|
|
|
|
2016-10-03 09:29:40 +11:00
|
|
|
ipc_init();
|
|
|
|
|
2015-08-20 22:37:09 +10:00
|
|
|
if (validate) {
|
2016-03-26 22:31:53 +11:00
|
|
|
bool valid = load_main_config(config_path, false);
|
2015-08-20 22:37:09 +10:00
|
|
|
return valid ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
2016-03-26 22:31:53 +11:00
|
|
|
if (!load_main_config(config_path, false)) {
|
2016-03-25 08:13:42 +11:00
|
|
|
sway_terminate(EXIT_FAILURE);
|
2015-08-18 08:15:05 +10:00
|
|
|
}
|
2016-03-25 08:13:42 +11:00
|
|
|
|
2015-08-20 22:37:09 +10:00
|
|
|
if (config_path) {
|
|
|
|
free(config_path);
|
|
|
|
}
|
2015-08-18 08:15:05 +10:00
|
|
|
|
2016-12-03 00:42:26 +11:00
|
|
|
security_sanity_check();
|
|
|
|
|
2015-08-20 23:12:34 +10:00
|
|
|
if (!terminate_request) {
|
|
|
|
wlc_run();
|
|
|
|
}
|
|
|
|
|
2016-05-06 08:30:28 +10:00
|
|
|
list_free(input_devices);
|
2016-01-17 21:53:37 +11:00
|
|
|
|
2015-08-20 23:12:34 +10:00
|
|
|
ipc_terminate();
|
2015-08-19 09:52:46 +10:00
|
|
|
|
2016-02-25 04:53:09 +11:00
|
|
|
if (config) {
|
|
|
|
free_config(config);
|
|
|
|
}
|
|
|
|
|
2016-02-26 19:08:05 +11:00
|
|
|
return exit_value;
|
2015-08-05 11:02:46 +10:00
|
|
|
}
|
2015-10-08 21:24:35 +11:00
|
|
|
|