mirror of
https://github.com/octoleo/lsyncd.git
synced 2025-01-23 23:28:25 +00:00
streamlined log output, fix a waiting issue
This commit is contained in:
parent
ac5357ea5e
commit
9ffc4ed7bf
@ -2,8 +2,9 @@
|
|||||||
bin_PROGRAMS = lsyncd
|
bin_PROGRAMS = lsyncd
|
||||||
lsyncd_SOURCES = lsyncd.c
|
lsyncd_SOURCES = lsyncd.c
|
||||||
TESTS = tests/help.sh \
|
TESTS = tests/help.sh \
|
||||||
tests/directorymv.sh \
|
|
||||||
tests/directorycpr.sh \
|
tests/directorycpr.sh \
|
||||||
|
tests/directorycpr0.sh \
|
||||||
|
tests/directorymv.sh \
|
||||||
tests/pidfile.sh \
|
tests/pidfile.sh \
|
||||||
tests/version.sh \
|
tests/version.sh \
|
||||||
tests/wrong-logfile.sh \
|
tests/wrong-logfile.sh \
|
||||||
|
103
lsyncd.c
103
lsyncd.c
@ -973,7 +973,6 @@ append_delay(const struct log *log,
|
|||||||
{
|
{
|
||||||
printlogf(log, DEBUG, "append_delay(%s, %d)", watch->dirname, alarm);
|
printlogf(log, DEBUG, "append_delay(%s, %d)", watch->dirname, alarm);
|
||||||
if (watch->delayed) {
|
if (watch->delayed) {
|
||||||
printlogf(log, DEBUG, "ignored since already delayed.");
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
watch->delayed = true;
|
watch->delayed = true;
|
||||||
@ -1105,6 +1104,9 @@ action(const struct global_options *opts,
|
|||||||
// makes a copy of all call parameters
|
// makes a copy of all call parameters
|
||||||
// step 1 binary itself
|
// step 1 binary itself
|
||||||
argv[argc++] = s_strdup(log, dir_conf->binary ? dir_conf->binary : opts->default_binary, "argv");
|
argv[argc++] = s_strdup(log, dir_conf->binary ? dir_conf->binary : opts->default_binary, "argv");
|
||||||
|
|
||||||
|
printlogf(log, NORMAL, " %s %s --> %s%s", argv[0], src, dest, recursive ? " (recursive)" : "");
|
||||||
|
|
||||||
// now all other parameters
|
// now all other parameters
|
||||||
for(; optp->kind != CO_EOL; optp++) {
|
for(; optp->kind != CO_EOL; optp++) {
|
||||||
switch (optp->kind) {
|
switch (optp->kind) {
|
||||||
@ -1361,7 +1363,8 @@ buildpath(const struct log *log,
|
|||||||
*/
|
*/
|
||||||
bool
|
bool
|
||||||
rsync_dir(const struct global_options *opts,
|
rsync_dir(const struct global_options *opts,
|
||||||
struct watch *watch)
|
struct watch *watch,
|
||||||
|
const char *event_text)
|
||||||
{
|
{
|
||||||
char pathname[PATH_MAX+1];
|
char pathname[PATH_MAX+1];
|
||||||
char destname[PATH_MAX+1];
|
char destname[PATH_MAX+1];
|
||||||
@ -1373,16 +1376,16 @@ rsync_dir(const struct global_options *opts,
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
printlogf(log, NORMAL, "%s: acting for %s.", event_text, pathname);
|
||||||
|
|
||||||
for (target = watch->dir_conf->targets; *target; target++) {
|
for (target = watch->dir_conf->targets; *target; target++) {
|
||||||
if (!buildpath(log, destname, sizeof(destname), watch, NULL, *target)) {
|
if (!buildpath(log, destname, sizeof(destname), watch, NULL, *target)) {
|
||||||
status = false;
|
status = false;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
printlogf(log, NORMAL, "rsyncing %s --> %s", pathname, destname);
|
// call the action to propagate changes in the directory
|
||||||
|
|
||||||
// call rsync to propagate changes in the directory
|
|
||||||
if (!action(opts, watch->dir_conf, pathname, destname, false)) {
|
if (!action(opts, watch->dir_conf, pathname, destname, false)) {
|
||||||
printlogf(log, ERROR, "Rsync from %s to %s failed", pathname, destname);
|
printlogf(log, ERROR, "Action %s --> %s has failed.", pathname, destname);
|
||||||
status = false;
|
status = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1397,28 +1400,35 @@ rsync_dir(const struct global_options *opts,
|
|||||||
* @param delay if true will put it on the delay, if false act immediately
|
* @param delay if true will put it on the delay, if false act immediately
|
||||||
* @param watch the watches of the delayed directory.
|
* @param watch the watches of the delayed directory.
|
||||||
* @param alarm times() when the directory handling should be fired.
|
* @param alarm times() when the directory handling should be fired.
|
||||||
|
* @param event_text event name for logging output
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
delay_or_act_dir(const struct global_options *opts,
|
delay_or_act_dir(const struct global_options *opts,
|
||||||
struct delay_vector *delays,
|
struct delay_vector *delays,
|
||||||
bool delay,
|
bool delay,
|
||||||
struct watch *watch,
|
struct watch *watch,
|
||||||
clock_t alarm)
|
clock_t alarm,
|
||||||
|
const char *event_text,
|
||||||
|
const char *event_name)
|
||||||
{
|
{
|
||||||
const struct log *log = &opts->log;
|
const struct log *log = &opts->log;
|
||||||
char pathname[PATH_MAX+1];
|
char pathname[PATH_MAX+1];
|
||||||
|
|
||||||
if (!delay) {
|
if (!delay) {
|
||||||
rsync_dir(opts, watch);
|
rsync_dir(opts, watch, event_text);
|
||||||
} else {
|
} else {
|
||||||
|
bool ret;
|
||||||
if (!buildpath(log, pathname, sizeof(pathname), watch, NULL, NULL)) {
|
if (!buildpath(log, pathname, sizeof(pathname), watch, NULL, NULL)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (append_delay(log, delays, watch, alarm)) {
|
ret = append_delay(log, delays, watch, alarm);
|
||||||
printlogf(log, NORMAL, "Putted %s on a delay", pathname);
|
if (event_name) {
|
||||||
|
printlogf(log, NORMAL, "%s %s in %s%s - delayed.",
|
||||||
|
event_text, event_name, pathname, ret ? "" : " already");
|
||||||
} else {
|
} else {
|
||||||
printlogf(log, NORMAL, "Not acted on %s already on delay", pathname);
|
printlogf(log, NORMAL, "%s in %si -%s delayed.",
|
||||||
|
event_text, pathname, ret ? "" : " already");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1460,6 +1470,7 @@ event_text_to_mask(char * text)
|
|||||||
* @param parent If not NULL, the watches to the parent directory already watched.
|
* @param parent If not NULL, the watches to the parent directory already watched.
|
||||||
* Must have absolute path if parent == NULL.
|
* Must have absolute path if parent == NULL.
|
||||||
* @param dir_conf applicateable configuration
|
* @param dir_conf applicateable configuration
|
||||||
|
* @param act if true delay or act on the new directory (dont on startup)
|
||||||
*
|
*
|
||||||
* @returns the watches of the directory or NULL on fail.
|
* @returns the watches of the directory or NULL on fail.
|
||||||
*/
|
*/
|
||||||
@ -1471,17 +1482,14 @@ add_dirwatch(const struct global_options *opts,
|
|||||||
int inotify_fd,
|
int inotify_fd,
|
||||||
char const *dirname,
|
char const *dirname,
|
||||||
struct watch *parent,
|
struct watch *parent,
|
||||||
struct dir_conf *dir_conf)
|
struct dir_conf *dir_conf,
|
||||||
|
bool act)
|
||||||
{
|
{
|
||||||
const struct log *log = &opts->log;
|
const struct log *log = &opts->log;
|
||||||
DIR *d;
|
DIR *d;
|
||||||
struct watch *w;
|
struct watch *w;
|
||||||
char pathname[PATH_MAX+1];
|
char pathname[PATH_MAX+1];
|
||||||
|
|
||||||
printlogf(log, DEBUG, "add_dirwatch(%s, p->dirname:%s, ...)",
|
|
||||||
dirname,
|
|
||||||
parent ? parent->dirname : "NULL");
|
|
||||||
|
|
||||||
if (!buildpath(log, pathname, sizeof(pathname), parent, dirname, NULL)) {
|
if (!buildpath(log, pathname, sizeof(pathname), parent, dirname, NULL)) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@ -1490,10 +1498,9 @@ add_dirwatch(const struct global_options *opts,
|
|||||||
int i;
|
int i;
|
||||||
for (i = 0; i < excludes->len; i++) {
|
for (i = 0; i < excludes->len; i++) {
|
||||||
if (!strcmp(pathname, excludes->data[i])) {
|
if (!strcmp(pathname, excludes->data[i])) {
|
||||||
printlogf(log, NORMAL, "Excluded %s", pathname);
|
printlogf(log, NORMAL, "ignored %s because of exclusion.", pathname);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
printlogf(log, DEBUG, "comparing %s with %s not an exclude so far.", pathname, excludes->data[i]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1503,8 +1510,13 @@ add_dirwatch(const struct global_options *opts,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// put this directory on list to be synced ASAP.
|
// if acting put this directory on list to be synced.
|
||||||
delay_or_act_dir(opts, delays, true, w, times(NULL));
|
// time is now so it as soon as possible, but it will be on
|
||||||
|
// top of the delay FIFO, so the current directory is
|
||||||
|
// guaranteed to be synced first.
|
||||||
|
if (act) {
|
||||||
|
delay_or_act_dir(opts, delays, true, w, times(NULL), "new subdirectory", "");
|
||||||
|
}
|
||||||
|
|
||||||
if (strlen(pathname) + strlen(dirname) + 2 > sizeof(pathname)) {
|
if (strlen(pathname) + strlen(dirname) + 2 > sizeof(pathname)) {
|
||||||
printlogf(log, ERROR, "pathname too long %s//%s", pathname, dirname);
|
printlogf(log, ERROR, "pathname too long %s//%s", pathname, dirname);
|
||||||
@ -1543,10 +1555,10 @@ add_dirwatch(const struct global_options *opts,
|
|||||||
// add watches if its a directory and not . or ..
|
// add watches if its a directory and not . or ..
|
||||||
if (isdir && strcmp(de->d_name, "..") && strcmp(de->d_name, ".")) {
|
if (isdir && strcmp(de->d_name, "..") && strcmp(de->d_name, ".")) {
|
||||||
// recurse into subdirectories
|
// recurse into subdirectories
|
||||||
struct watch *nw = add_dirwatch(opts, watches, delays, excludes, inotify_fd, de->d_name, w, dir_conf);
|
printlogf(log, NORMAL, "%s %s%s",
|
||||||
printlogf(log, NORMAL,
|
act ? "encountered" : "watching",
|
||||||
"found new directory: %s in %s -- %s",
|
pathname, de->d_name);
|
||||||
de->d_name, dirname, nw ? "will be synced" : "ignored it");
|
add_dirwatch(opts, watches, delays, excludes, inotify_fd, de->d_name, w, dir_conf, act);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1716,7 +1728,7 @@ handle_event(const struct global_options *opts,
|
|||||||
if (!watch) {
|
if (!watch) {
|
||||||
// this can happen in case of data moving faster than lsyncd can monitor it.
|
// this can happen in case of data moving faster than lsyncd can monitor it.
|
||||||
printlogf(log, NORMAL,
|
printlogf(log, NORMAL,
|
||||||
"received an inotify event that doesnt match any watched directory.");
|
"event %s:%s is not from a watched directory (which presumably was recently deleted)", masktext, event->name ? event->name : "");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1724,15 +1736,14 @@ handle_event(const struct global_options *opts,
|
|||||||
if ((IN_ATTRIB | IN_CREATE | IN_CLOSE_WRITE | IN_DELETE |
|
if ((IN_ATTRIB | IN_CREATE | IN_CLOSE_WRITE | IN_DELETE |
|
||||||
IN_MOVED_TO | IN_MOVED_FROM) & event->mask
|
IN_MOVED_TO | IN_MOVED_FROM) & event->mask
|
||||||
) {
|
) {
|
||||||
printlogf(log, NORMAL, "received event %s:%s for %d.", masktext, event->name, alarm);
|
delay_or_act_dir(opts, delays, opts->delay > 0, watch, alarm, masktext, event->name);
|
||||||
delay_or_act_dir(opts, delays, opts->delay > 0, watch, alarm);
|
|
||||||
} else {
|
} else {
|
||||||
printlogf(log, DEBUG, "... ignored this event.");
|
printlogf(log, DEBUG, "... ignored this event.");
|
||||||
}
|
}
|
||||||
|
|
||||||
// in case of a new directory create new watches of the subdir
|
// in case of a new directory create new watches of the subdir
|
||||||
if (((IN_CREATE | IN_MOVED_TO) & event->mask) && (IN_ISDIR & event->mask)) {
|
if (((IN_CREATE | IN_MOVED_TO) & event->mask) && (IN_ISDIR & event->mask)) {
|
||||||
add_dirwatch(opts, watches, delays, excludes, inotify_fd, event->name, watch, watch->dir_conf);
|
add_dirwatch(opts, watches, delays, excludes, inotify_fd, event->name, watch, watch->dir_conf, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// in case of a removed directory remove watches from the subdir
|
// in case of a removed directory remove watches from the subdir
|
||||||
@ -1769,8 +1780,6 @@ master_loop(const struct global_options *opts,
|
|||||||
clock_t alarm;
|
clock_t alarm;
|
||||||
const struct log *log = &opts->log;
|
const struct log *log = &opts->log;
|
||||||
|
|
||||||
FD_ZERO(&readfds);
|
|
||||||
FD_SET(inotify_fd, &readfds);
|
|
||||||
|
|
||||||
if (opts->delay > 0) {
|
if (opts->delay > 0) {
|
||||||
if (clocks_per_sec <= 0) {
|
if (clocks_per_sec <= 0) {
|
||||||
@ -1802,12 +1811,14 @@ master_loop(const struct global_options *opts,
|
|||||||
}
|
}
|
||||||
// if select returns a positive number there is data on inotify
|
// if select returns a positive number there is data on inotify
|
||||||
// on zero the timemout occured.
|
// on zero the timemout occured.
|
||||||
|
FD_ZERO(&readfds);
|
||||||
|
FD_SET(inotify_fd, &readfds);
|
||||||
do_read = select(inotify_fd + 1, &readfds, NULL, NULL, &tv);
|
do_read = select(inotify_fd + 1, &readfds, NULL, NULL, &tv);
|
||||||
|
|
||||||
if (do_read) {
|
if (do_read) {
|
||||||
printlogf(log, DEBUG, "theres data on inotify.");
|
printlogf(log, DEBUG, "theres data on inotify.");
|
||||||
} else {
|
} else {
|
||||||
printlogf(log, DEBUG, "select() timeouted, doiong delays.");
|
printlogf(log, DEBUG, "select() timeout, doing delays.");
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// if nothing to wait for, enter a blocking read
|
// if nothing to wait for, enter a blocking read
|
||||||
@ -1822,10 +1833,9 @@ master_loop(const struct global_options *opts,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (len < 0) {
|
if (len < 0) {
|
||||||
if (!keep_going) {
|
if (keep_going) {
|
||||||
printlogf(log, NORMAL, "read exited due to TERM signal.");
|
// if !keep_going it is perfectly normal to be interupted.
|
||||||
} else {
|
printlogf(log, ERROR, "read error from inotify (%d:%s)", errno, strerror(errno));
|
||||||
printlogf(log, ERROR, "failed to read from inotify (%d:%s)", errno, strerror(errno));
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1848,8 +1858,7 @@ master_loop(const struct global_options *opts,
|
|||||||
// or the stack is empty. Using now time - times(NULL) - everytime
|
// or the stack is empty. Using now time - times(NULL) - everytime
|
||||||
// again as time may progresses while handling delayed entries.
|
// again as time may progresses while handling delayed entries.
|
||||||
while (delays->len > 0 && time_after_eq(times(NULL), delays->data[0]->alarm)) {
|
while (delays->len > 0 && time_after_eq(times(NULL), delays->data[0]->alarm)) {
|
||||||
printlogf(log, DEBUG, "time %d for '%s' arrived. now=%d", delays->data[0]->alarm, delays->data[0]->dirname, times(NULL));
|
rsync_dir(opts, delays->data[0], "delay expired");
|
||||||
rsync_dir(opts, delays->data[0]);
|
|
||||||
remove_first_delay(delays);
|
remove_first_delay(delays);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2443,11 +2452,11 @@ parse_options(struct global_options *opts, int argc, char **argv)
|
|||||||
for (target = &argv[optind + 1]; *target; target++) {
|
for (target = &argv[optind + 1]; *target; target++) {
|
||||||
dir_conf_add_target(NULL, odc, *target);
|
dir_conf_add_target(NULL, odc, *target);
|
||||||
if (first_target) {
|
if (first_target) {
|
||||||
printlogf(&opts->log, NORMAL, "command line options: syncing %s -> %s\n",
|
printlogf(&opts->log, NORMAL, "command line options: syncing %s -> %s",
|
||||||
odc->source, *target);
|
odc->source, *target);
|
||||||
first_target = false;
|
first_target = false;
|
||||||
} else {
|
} else {
|
||||||
printlogf(&opts->log, NORMAL, " and -> %s\n",
|
printlogf(&opts->log, NORMAL, " and -> %s",
|
||||||
*target);
|
*target);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2595,7 +2604,7 @@ one_main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
printlogf(log, NORMAL, "Starting up");
|
printlogf(log, NORMAL, "--- Starting up ---");
|
||||||
|
|
||||||
if (opts.pidfile) {
|
if (opts.pidfile) {
|
||||||
write_pidfile(log, opts.pidfile);
|
write_pidfile(log, opts.pidfile);
|
||||||
@ -2613,22 +2622,10 @@ one_main(int argc, char **argv)
|
|||||||
for (i = 0; i < opts.dir_conf_n; i++) {
|
for (i = 0; i < opts.dir_conf_n; i++) {
|
||||||
printlogf(log, NORMAL, "watching %s", opts.dir_confs[i].source);
|
printlogf(log, NORMAL, "watching %s", opts.dir_confs[i].source);
|
||||||
add_dirwatch(&opts, &watches, &delays, &excludes, inotify_fd,
|
add_dirwatch(&opts, &watches, &delays, &excludes, inotify_fd,
|
||||||
opts.dir_confs[i].source, NULL, &opts.dir_confs[i]);
|
opts.dir_confs[i].source, NULL, &opts.dir_confs[i], false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// clears delay FIFO again, because the startup recursive rsync will
|
|
||||||
// handle it eitherway or if started no-startup it has to be ignored.
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
printlogf(log, DEBUG, "dumping delay list.");
|
|
||||||
for(i = 0; i < delays.len; i++) {
|
|
||||||
delays.data[i]->delayed = false;
|
|
||||||
delays.data[i]->alarm = 0;
|
|
||||||
}
|
|
||||||
delays.len = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// startup recursive sync.
|
// startup recursive sync.
|
||||||
if (!opts.flag_nostartup) {
|
if (!opts.flag_nostartup) {
|
||||||
int i;
|
int i;
|
||||||
|
62
tests/directorycpr0.sh
Executable file
62
tests/directorycpr0.sh
Executable file
@ -0,0 +1,62 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
CON="\E[47;34m"
|
||||||
|
COFF="\033[0m"
|
||||||
|
|
||||||
|
echo -e "$CON***************************************************************$COFF"
|
||||||
|
echo -e "$CON** Testing the case of directory being cp -r'ed and touched. **$COFF"
|
||||||
|
echo -e "$CON** With zero delay **$COFF"
|
||||||
|
echo -e "$CON***************************************************************$COFF"
|
||||||
|
|
||||||
|
WORKSOURCE=$(mktemp -d)
|
||||||
|
WORKTARGET=$(mktemp -d)
|
||||||
|
PIDFILE=$(mktemp)
|
||||||
|
LOGFILE=$(mktemp)
|
||||||
|
|
||||||
|
echo -e "$CON* populating the filesystem$COFF"
|
||||||
|
mkdir -p "${WORKSOURCE}"/a/a
|
||||||
|
echo 'test' > "${WORKSOURCE}"/a/a/file
|
||||||
|
|
||||||
|
echo -e "$CON* starting lsyncd$COFF"
|
||||||
|
./lsyncd --logfile "${LOGFILE}" --pidfile "${PIDFILE}" --no-daemon --delay 0 "${WORKSOURCE}" "${WORKTARGET}"&
|
||||||
|
|
||||||
|
echo -e "$CON* waiting for lsyncd to start$COFF"
|
||||||
|
sleep 4s
|
||||||
|
|
||||||
|
# cp -r the directory
|
||||||
|
echo -e "$CON* making a lot of data$COFF"
|
||||||
|
for A in 1 2 3 4 5 6 7 8 9 10; do
|
||||||
|
cp -r "${WORKSOURCE}"/a "${WORKSOURCE}"/b${A}
|
||||||
|
echo 'test2' > "${WORKSOURCE}"/b${A}/a/another
|
||||||
|
done
|
||||||
|
mkdir -p "${WORKSOURCE}"/c/a
|
||||||
|
echo 'test3' > "${WORKSOURCE}"/c/a/file
|
||||||
|
for A in 1 2 3 4 5 6 7 8 9 10; do
|
||||||
|
cp -r "${WORKSOURCE}"/c "${WORKSOURCE}"/d${A}
|
||||||
|
echo 'test2' > "${WORKSOURCE}"/d${A}/a/another
|
||||||
|
done
|
||||||
|
|
||||||
|
echo -e "$CON*waiting until lsyncd does the job.$COFF"
|
||||||
|
sleep 20s
|
||||||
|
|
||||||
|
echo -e "$CON*killing lsyncd$COFF"
|
||||||
|
LSYNCPID=$(cat "${PIDFILE}")
|
||||||
|
if ! kill "${LSYNCPID}"; then
|
||||||
|
cat "${LOGFILE}"
|
||||||
|
diff -urN "${WORKSOURCE}" "${WORKTARGET}" || true
|
||||||
|
echo "kill failed"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
sleep 1s
|
||||||
|
|
||||||
|
#echo "log file contents"
|
||||||
|
#cat "${LOGFILE}"
|
||||||
|
##this should be grep.
|
||||||
|
|
||||||
|
echo -e "$CON*differences$COFF"
|
||||||
|
diff -urN "${WORKSOURCE}" "${WORKTARGET}"
|
||||||
|
|
||||||
|
rm "${PIDFILE}"
|
||||||
|
rm "${LOGFILE}"
|
||||||
|
rm -rf "${WORKTARGET}"
|
||||||
|
rm -rf "${WORKSOURCE}"
|
Loading…
x
Reference in New Issue
Block a user