2020-10-16 00:32:37 +08:00
|
|
|
#! /usr/bin/perl -T
|
|
|
|
# vim: set filetype=perl ts=4 sw=4 sts=4 et:
|
|
|
|
# NEEDGROUP osh-groupDelete
|
|
|
|
# SUDOERS %osh-groupDelete ALL=(root) NOPASSWD:/usr/bin/env perl -T /opt/bastion/bin/helper/osh-groupDelete *
|
2021-05-28 19:48:08 +08:00
|
|
|
# KEYSUDOERS # as an owner, we can delete our own group
|
|
|
|
# KEYSUDOERS SUPEROWNERS, %%GROUP%-owner ALL=(root) NOPASSWD: /usr/bin/env perl -T %BASEPATH%/bin/helper/osh-groupDelete --group %GROUP%
|
2020-10-16 00:32:37 +08:00
|
|
|
# FILEMODE 0700
|
2020-11-17 18:12:53 +08:00
|
|
|
# FILEOWN 0 0
|
2020-10-16 00:32:37 +08:00
|
|
|
|
|
|
|
#>HEADER
|
|
|
|
use common::sense;
|
2021-12-09 20:54:33 +08:00
|
|
|
use Getopt::Long qw(:config no_auto_abbrev no_ignore_case);
|
2020-10-16 00:32:37 +08:00
|
|
|
|
|
|
|
use File::Basename;
|
|
|
|
use lib dirname(__FILE__) . '/../../lib/perl';
|
|
|
|
use OVH::Bastion;
|
2021-11-30 19:20:28 +08:00
|
|
|
use OVH::Bastion::Helper;
|
2020-10-16 00:32:37 +08:00
|
|
|
|
|
|
|
# Fetch command options
|
|
|
|
my $fnret;
|
|
|
|
my ($result, @optwarns);
|
|
|
|
my ($group);
|
|
|
|
eval {
|
|
|
|
local $SIG{__WARN__} = sub { push @optwarns, shift };
|
|
|
|
$result = GetOptions("group=s" => sub { $group //= $_[1] },);
|
|
|
|
};
|
|
|
|
if ($@) { die $@ }
|
|
|
|
|
|
|
|
if (!$result) {
|
|
|
|
local $" = ", ";
|
|
|
|
HEXIT('ERR_BAD_OPTIONS', msg => "Error parsing options: @optwarns");
|
|
|
|
}
|
|
|
|
|
2021-12-16 01:11:03 +08:00
|
|
|
OVH::Bastion::Helper::check_spurious_args();
|
|
|
|
|
2020-10-16 00:32:37 +08:00
|
|
|
if (!$group) {
|
|
|
|
HEXIT('ERR_MISSING_PARAMETER', msg => "Missing argument 'group'");
|
|
|
|
}
|
|
|
|
|
|
|
|
#<HEADER
|
|
|
|
|
|
|
|
#>PARAMS:GROUP
|
|
|
|
# test if start by key, append if necessary
|
|
|
|
osh_debug("Checking group");
|
|
|
|
$fnret = OVH::Bastion::is_valid_group_and_existing(group => $group, groupType => "key");
|
|
|
|
$fnret or HEXIT($fnret);
|
|
|
|
|
|
|
|
# get returned untainted value
|
|
|
|
$group = $fnret->value->{'group'};
|
|
|
|
my $shortGroup = $fnret->value->{'shortGroup'};
|
|
|
|
|
|
|
|
#<PARAMS:GROUP
|
|
|
|
|
2021-05-28 19:48:08 +08:00
|
|
|
#>RIGHTSCHECK
|
|
|
|
if ($self eq 'root') {
|
|
|
|
osh_debug "Real root, skipping checks of permissions";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
# either we can delete any group
|
|
|
|
$fnret = OVH::Bastion::is_user_in_group(user => $self, group => "osh-groupDelete");
|
|
|
|
if (!$fnret) {
|
|
|
|
|
|
|
|
# or we can delete our own group as the owner of said group
|
|
|
|
$fnret = OVH::Bastion::is_group_owner(account => $self, group => $shortGroup, sudo => 1, superowner => 1);
|
|
|
|
if (!$fnret) {
|
|
|
|
HEXIT('ERR_SECURITY_VIOLATION', msg => "You're not allowed to run this, dear $self");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#<RIGHTSCHECK
|
|
|
|
|
2020-10-16 00:32:37 +08:00
|
|
|
#>CODE
|
|
|
|
# last security check
|
|
|
|
if (not -e "/home/$group/allowed.ip" or not -e "/home/keykeeper/$group") {
|
|
|
|
HEXIT('ERR_INVALID_GROUP', msg => "Sorry, but $shortGroup doesn't seem to be a legit bastion group");
|
|
|
|
}
|
|
|
|
|
2022-01-18 19:20:16 +08:00
|
|
|
# do the stuff
|
|
|
|
osh_info("Backing up group directory...");
|
|
|
|
|
|
|
|
if (!-d "/home/oldkeeper") {
|
2020-10-16 00:32:37 +08:00
|
|
|
mkdir "/home/oldkeeper";
|
|
|
|
}
|
|
|
|
chown 0, 0, "/home/oldkeeper";
|
|
|
|
chmod 0700, "/home/oldkeeper";
|
|
|
|
|
|
|
|
if (!-d "/home/oldkeeper/groups") {
|
|
|
|
mkdir "/home/oldkeeper/groups";
|
|
|
|
}
|
|
|
|
chown 0, 0, "/home/oldkeeper/groups";
|
|
|
|
chmod 0700, "/home/oldkeeper/groups";
|
|
|
|
|
|
|
|
my $suffix = 'at-' . time() . '.by-' . $self;
|
|
|
|
|
|
|
|
my $fulldir = "/home/oldkeeper/groups/$group.$suffix";
|
|
|
|
if (-e $fulldir) {
|
2022-01-18 19:20:16 +08:00
|
|
|
HEXIT('ERR_BACKUP_DIR_COLLISION', msg => "This shouldn't happen, $fulldir already exists!");
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
mkdir $fulldir;
|
|
|
|
chown 0, 0, $fulldir;
|
|
|
|
chmod 0700, $fulldir;
|
|
|
|
|
2022-01-18 19:20:16 +08:00
|
|
|
# from now on, as we're starting to move/remove/delete things, errors will be non fatal
|
|
|
|
# because we're trying to cleanup as much as we can, the idea is to log errors to the
|
|
|
|
# syslog, and then report to the user (and in the final formatted log) that we
|
|
|
|
# had issues and somebody might want to have a look
|
|
|
|
my $nbErrors = 0;
|
|
|
|
|
2022-06-17 21:40:39 +08:00
|
|
|
# File::Copy::move() sometimes craps itself when it gets -EXDEV from the OS, and doesn't
|
|
|
|
# compensate for it, while /bin/mv does...
|
|
|
|
$fnret = OVH::Bastion::execute_simple(cmd => ['mv', "/home/$group", "$fulldir/$group-home"], must_succeed => 1);
|
|
|
|
if (!$fnret) {
|
|
|
|
my $msg = substr($fnret->value->{'output'}, 0, 128);
|
|
|
|
$msg =~ s=[^a-zA-Z0-9./_-]=_=g;
|
2022-06-30 21:00:29 +08:00
|
|
|
warn_syslog("Error while backing up to-be-deleted '/home/$group' to '$fulldir/$group-home' ($msg)"
|
|
|
|
. ", continuing anyway...");
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
|
|
|
}
|
2022-06-30 21:00:29 +08:00
|
|
|
$fnret =
|
|
|
|
OVH::Bastion::execute_simple(cmd => ['mv', "/home/keykeeper/$group", "$fulldir/$group-keykeeper"], must_succeed => 1);
|
2022-06-17 21:40:39 +08:00
|
|
|
if (!$fnret) {
|
|
|
|
my $msg = substr($fnret->value->{'output'}, 0, 128);
|
|
|
|
$msg =~ s=[^a-zA-Z0-9./_-]=_=g;
|
2022-06-30 21:00:29 +08:00
|
|
|
warn_syslog("Error while backing up to-be-deleted '/home/keykeeper/$group' to '$fulldir/$group-keykeeper' ($msg)"
|
|
|
|
. ", continuing anyway...");
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
|
|
|
|
# now tar.gz the directory, this is important because inside we'll keep the
|
|
|
|
# old GID of the group, and we don't want GID-orphaned files on our filesystem, it's
|
|
|
|
# not a problem to have those inside a tarfile however.
|
|
|
|
my @tarcmd = qw{ tar czf };
|
|
|
|
push @tarcmd, $fulldir . '.tar.gz';
|
|
|
|
push @tarcmd, '--acls' if OVH::Bastion::has_acls();
|
|
|
|
push @tarcmd, '--one-file-system', '-p', '--remove-files', $fulldir;
|
2022-01-18 19:20:16 +08:00
|
|
|
|
2020-10-16 00:32:37 +08:00
|
|
|
$fnret = OVH::Bastion::execute(cmd => \@tarcmd, must_succeed => 1);
|
|
|
|
if (!$fnret) {
|
2022-01-18 19:20:16 +08:00
|
|
|
warn_syslog("Couldn't tar the backup homedir of this group (" . $fnret->msg . "), proceeding anyway.");
|
|
|
|
my $i = 0;
|
|
|
|
foreach (@{$fnret->value->{'stderr'} || []}) {
|
|
|
|
warn_syslog("tar: $_");
|
|
|
|
if (++$i >= 10) {
|
|
|
|
warn_syslog("more tar errors, suppressing");
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$nbErrors++;
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
2022-01-18 19:20:16 +08:00
|
|
|
|
|
|
|
if (-e "$fulldir.tar.gz") {
|
|
|
|
chmod 0000, "$fulldir.tar.gz";
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
|
2022-01-18 19:20:16 +08:00
|
|
|
# if the folder still exists, tar failed in some way, warn the admins
|
|
|
|
if (-d $fulldir) {
|
|
|
|
chmod 0000, $fulldir;
|
|
|
|
warn_syslog("While archiving the group '$group', $fulldir still exists, manual cleanup might be needed");
|
|
|
|
$nbErrors++;
|
|
|
|
}
|
|
|
|
osh_info("Backup done");
|
|
|
|
|
2020-10-16 00:32:37 +08:00
|
|
|
# remove dead symlinks in users homes
|
|
|
|
my $dh;
|
|
|
|
if (opendir($dh, "/home/allowkeeper")) {
|
|
|
|
while (my $dir = readdir($dh)) {
|
|
|
|
$dir =~ /^\./ and next;
|
|
|
|
$dir !~ /^([a-zA-Z0-9._-]+)$/ and next;
|
|
|
|
$dir = "/home/allowkeeper/$1"; # and untaint
|
|
|
|
-d $dir or next;
|
|
|
|
foreach my $file ("$dir/allowed.ip.$shortGroup", "$dir/allowed.partial.$shortGroup") {
|
|
|
|
if (-e $file || -l $file) {
|
|
|
|
osh_info "Removing $file...";
|
2022-01-18 19:20:16 +08:00
|
|
|
if (!unlink($file)) {
|
|
|
|
warn_syslog("Couldn't remove symlink '$file': $!");
|
|
|
|
$nbErrors++;
|
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close($dh);
|
|
|
|
}
|
|
|
|
else {
|
2022-01-18 19:20:16 +08:00
|
|
|
warn_syslog("Couldn't open /home/allowkeeper to cleanup symlinks: $!");
|
|
|
|
$nbErrors++;
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
# trying to remove main and gatekeeper and owner groups
|
|
|
|
foreach my $todelete ("$group-owner", "$group-aclkeeper", "$group-gatekeeper", $group) {
|
|
|
|
$fnret = OVH::Bastion::is_group_existing(group => $todelete);
|
|
|
|
if ($fnret) {
|
|
|
|
$todelete = $fnret->value->{'group'}; # untaint
|
|
|
|
my $members = $fnret->value->{'members'} || [];
|
|
|
|
if (@$members) {
|
|
|
|
osh_info "Found " . (scalar @$members) . " members, removing them from the group";
|
|
|
|
foreach my $member (@$members) {
|
|
|
|
osh_info "... removing $member from group $todelete";
|
|
|
|
$fnret = OVH::Bastion::sys_delmemberfromgroup(user => $member, group => $todelete, noisy_stderr => 1);
|
2022-01-18 19:20:16 +08:00
|
|
|
if ($fnret->err ne 'OK') {
|
2022-06-30 21:00:29 +08:00
|
|
|
warn_syslog("Error while attempting to remove member $member from group $todelete ("
|
|
|
|
. $fnret->msg
|
|
|
|
. "), proceeding anyway");
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($todelete eq $group) {
|
2022-06-30 21:00:29 +08:00
|
|
|
osh_info "Deleting main user of group $todelete...",
|
|
|
|
$fnret = OVH::Bastion::sys_userdel(user => $todelete, noisy_stderr => 1);
|
2022-01-18 19:20:16 +08:00
|
|
|
if ($fnret->err ne 'OK') {
|
2022-06-30 21:00:29 +08:00
|
|
|
warn_syslog("Error while attempting to delete main user of group $todelete ("
|
|
|
|
. $fnret->msg
|
|
|
|
. "), proceeding anyway");
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
# some OSes delete the main group of user if it has the same name
|
|
|
|
# and nobody else is a member of it, so check it still exists before
|
|
|
|
# trying to delete it
|
|
|
|
$fnret = OVH::Bastion::is_group_existing(group => $todelete);
|
|
|
|
if ($fnret) {
|
|
|
|
osh_info "Deleting group $todelete...";
|
|
|
|
$fnret = OVH::Bastion::sys_groupdel(group => $todelete, noisy_stderr => 1);
|
2022-01-18 19:20:16 +08:00
|
|
|
if ($fnret->err ne 'OK') {
|
2022-06-30 21:00:29 +08:00
|
|
|
warn_syslog(
|
|
|
|
"Error while attempting to delete group $todelete (" . $fnret->msg . "), proceeding anyway");
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
osh_info "Group $todelete not found, ignoring...";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# remove sudoers if it's there
|
2022-06-30 21:00:29 +08:00
|
|
|
$fnret = OVH::Bastion::execute(
|
|
|
|
cmd => [$OVH::Bastion::BASEPATH . '/bin/sudogen/generate-sudoers.sh', 'delete', 'group', $group],
|
|
|
|
must_succeed => 1,
|
|
|
|
noisy_stdout => 1
|
|
|
|
);
|
2020-12-03 19:04:01 +08:00
|
|
|
if (!$fnret) {
|
2020-12-15 21:57:58 +08:00
|
|
|
warn_syslog("Error during group deletion of '$group', couldn't delete sudoers file: " . $fnret->msg);
|
2022-01-18 19:20:16 +08:00
|
|
|
$nbErrors++;
|
2020-12-03 19:04:01 +08:00
|
|
|
}
|
2020-10-16 00:32:37 +08:00
|
|
|
|
|
|
|
OVH::Bastion::syslogFormatted(
|
|
|
|
severity => 'info',
|
|
|
|
type => 'group',
|
2022-01-18 19:20:16 +08:00
|
|
|
fields => [['action', 'delete'], ['group', $shortGroup], ['errors', $nbErrors]]
|
2020-10-16 00:32:37 +08:00
|
|
|
);
|
|
|
|
|
2022-06-30 21:00:29 +08:00
|
|
|
HEXIT(
|
|
|
|
'OK',
|
|
|
|
value => {group => $group, operation => 'deleted', errors => $nbErrors + 0},
|
|
|
|
msg => "Group $group has been deleted"
|
|
|
|
);
|