Skip to content

Commit

Permalink
cyr_withlock_run: run an external command with a lock held
Browse files Browse the repository at this point in the history
This can run with either a single user locked OR with the
global exclusive lock held.
  • Loading branch information
brong committed Dec 13, 2023
1 parent 9e684c9 commit 1b848e4
Show file tree
Hide file tree
Showing 4 changed files with 234 additions and 0 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,7 @@ imap/cyr_sphinxmgr
imap/cyr_synclog
imap/cyr_userseen
imap/cyr_virusscan
imap/cyr_withlock_run
imap/cyrdump
imap/dav_reconstruct
imap/deliver
Expand Down
4 changes: 4 additions & 0 deletions Makefile.am
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,7 @@ sbin_PROGRAMS += \
imap/cyr_df \
imap/cyr_ls \
imap/cyr_pwd \
imap/cyr_withlock_run \
imap/cyrdump \
imap/cyr_dbtool \
imap/cyr_deny \
Expand Down Expand Up @@ -923,6 +924,9 @@ imap_cyr_ls_LDADD = $(LD_UTILITY_ADD)
imap_cyr_pwd_SOURCES = imap/cli_fatal.c imap/cyr_pwd.c imap/mutex_fake.c
imap_cyr_pwd_LDADD = $(LD_UTILITY_ADD)

imap_cyr_withlock_run_SOURCES = imap/cli_fatal.c imap/cyr_withlock_run.c imap/mutex_fake.c
imap_cyr_withlock_run_LDADD = $(LD_UTILITY_ADD)

imap_cyr_expire_SOURCES = imap/cli_fatal.c imap/cyr_expire.c imap/mutex_fake.c
imap_cyr_expire_LDADD = $(LD_UTILITY_ADD)

Expand Down
85 changes: 85 additions & 0 deletions docsrc/imap/reference/manpages/systemcommands/cyr_withlock_run.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
.. cyrusman:: cyr_withlock_run(8)

.. author: Bron Gondwana
.. _imap-reference-manpages-systemcommands-cyr_withlock_run:

====================
**cyr_withlock_run**
====================

is used to run system command with a lock held

.. warning::

This command runs as the Cyrus user, so if you need to call something
as root, the cyrus user may need sudo privileges.

Synopsis
========

.. parsed-literal::
**cyr_withlock_run** [ **-C** *config-file* ] [ **-u** *userid* ] cmd args...
Description
===========

**cyr_withlock_run** will run the command and arguments provided on the
command line, either locking the entire server, or a particular user, so
no changes can be made to it while this command is running.

WARNING: since this takes an exclusive lock, it will deadlock if the command tries
to connect to Cyrus via IMAP/JMAP/etc and run commands that take exclusive locks.
You can run cyrus commandline tools so long as the environment is passed through.

It is most useful for running an external filesystem snapshot command, which can
be run safely, knowing that files aren't in a partial state.


**cyr_withlock_run** |default-conf-text|

Options
=======

.. program:: cyr_withlock_run

.. option:: -C config-file

|cli-dash-c-text|

.. option:: -u, --user <userid>

Lock just the specified userid rather than the global lock.

NOTE: if you run for the global lock, then your server must have the
`global_lock` config option set, or this command will fail with an
error as it can't get a guaranteed lock.

Examples
========

.. parsed-literal::
**cyr_withlock_run** sleep 300
..
Pause all writes on the server for 300 seconds (stops any writes from getting locks)

Files
=====

/etc/imapd.conf


Environment
===========

Sets the `CYRUS_HAVELOCK_GLOBAL` or `CYRUS_HAVELOCK_USER` environment variables,
to tell any called cyrus command that this lock is already held.

See Also
========

:cyrusman:`imapd.conf(5)`
144 changes: 144 additions & 0 deletions imap/cyr_withlock_run.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
/* cyr_withlock_run.c -- run a command with the global lock or a user lock held
*
* Copyright (c) 1994-2023 Carnegie Mellon University. All rights reserved.
*
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* 3. The name "Carnegie Mellon University" must not be used to
* endorse or promote products derived from this software without
* prior written permission. For permission or any legal
* details, please contact
* Carnegie Mellon University
* Center for Technology Transfer and Enterprise Creation
* 4615 Forbes Avenue
* Suite 302
* Pittsburgh, PA 15213
* (412) 268-7393, fax: (412) 268-7395
* [email protected]
*
* 4. Redistributions of any form whatsoever must retain the following
* acknowledgment:
* "This product includes software developed by Computing Services
* at Carnegie Mellon University (http://www.cmu.edu/computing/)."
*
* CARNEGIE MELLON UNIVERSITY DISCLAIMS ALL WARRANTIES WITH REGARD TO
* THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
* AND FITNESS, IN NO EVENT SHALL CARNEGIE MELLON UNIVERSITY BE LIABLE
* FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
* AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/

#include <config.h>

#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif

#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include "global.h"
#include "mboxname.h"
#include "command.h"
#include "strarray.h"
#include "user.h"

/* generated headers are not necessarily in current directory */
#include "imap/imap_err.h"

/* current namespace */
static struct namespace cyr_runlock_namespace;

static int usage(const char *error)
{
fprintf(stderr, "usage: cyr_runlock [-C <alt_config>] cmd args\n");
fprintf(stderr, "\n");
if (error) {
fprintf(stderr, "\n");
fprintf(stderr, "ERROR: %s", error);
}
exit(-1);
}

int runcmd(void *rock)
{
return run_command_strarray((const strarray_t *)rock);
}

int main(int argc, char **argv)
{
int r;
int opt;
char *alt_config = NULL;
char *userid = NULL;

/* keep this in alphabetical order */
static const char short_options[] = "C:u:";

static const struct option long_options[] = {
/* n.b. no long option for -C */
{ "user", required_argument, NULL, 'u' },
{ 0, 0, 0, 0 },
};

while (-1 != (opt = getopt_long(argc, argv,
short_options, long_options, NULL)))
{
switch(opt) {
case 'C': /* alt config file */
alt_config = optarg;
break;

case 'u':
userid = optarg;
break;

default:
usage(NULL);
}
}

cyrus_init(alt_config, "cyr_runlock", 0, 0);

r = mboxname_init_namespace(&cyr_runlock_namespace, NAMESPACE_OPTION_ADMIN);
if (r) {
fatal(error_message(r), -1);
}

strarray_t args = STRARRAY_INITIALIZER;
int i;
for (i = optind; i < argc; i++)
strarray_append(&args, argv[i]);
if (userid) {
static char env_havelock[100];
snprintf(env_havelock, sizeof(env_havelock), "CYRUS_HAVELOCK_GLOBAL=1");
putenv(env_havelock);
r = mboxname_run_with_lock(runcmd, &args);
}
else {
static char env_userlock[MAX_MAILBOX_NAME+30];
snprintf(env_userlock, sizeof(env_userlock), "CYRUS_HAVELOCK_USER=%s", userid);
putenv(env_userlock);
r = user_run_with_lock(userid, runcmd, &args);
}
strarray_fini(&args);

cyrus_done();

exit(r ? EXIT_FAILURE : EXIT_SUCCESS);
}

0 comments on commit 1b848e4

Please sign in to comment.