diff options
-rwxr-xr-x | userdrake | 859 |
1 files changed, 859 insertions, 0 deletions
diff --git a/userdrake b/userdrake new file mode 100755 index 0000000..1752820 --- /dev/null +++ b/userdrake @@ -0,0 +1,859 @@ +#!/usr/bin/perl +################################################################################ +# # +# # +# Copyright (C) 2003 MandrakeSoft # +# +# Daouda Lo <daouda@mandrakesoft.com> # +# # +# This program is free software; you can redistribute it and/or modify # +# it under the terms of the GNU General Public License Version 2 as # +# published by the Free Software Foundation. # +# # +# This program is distributed in the hope that it will be useful, # +# but WITHOUT ANY WARRANTY; without even the implied warranty of # +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # +# GNU General Public License for more details. # +# # +# You should have received a copy of the GNU General Public License # +# along with this program; if not, write to the Free Software # +# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. # +################################################################################ + +use strict; +use lib qw(/usr/lib/libDrakX); +use standalone; + +use common; +use any; +use ugtk2 qw(:all); +use interactive; +use POSIX qw(mktime ceil); +use USER; +use Devel::Peek; + +my $conffile = '/etc/sysconfig/userdrake'; +my $secfile = '/etc/sysconfig/msec'; +my $pixdir = '/usr/share/userdrake/pixmaps/'; + +c::bind_textdomain_codeset('userdrake', 'UTF8'); +sub translate { + my ($s) = @_; + my $r = ''; + if ($s) { + $r = c::dgettext('userdrake', $s); + $r eq $s and $r = Locale::gettext::iconv(c::dgettext('urpmi', $s), undef, "UTF-8"); + c::set_tagged_utf8($r); + } + $r; +} +sub sprintf_fixutf8 { + my $need_upgrade; + $need_upgrade |= to_bool(c::is_tagged_utf8($_)) + 1 foreach @_; + if ($need_upgrade == 3) { c::upgrade_utf8($_) foreach @_ }; + sprintf shift, @_; +} +sub N { + my $s = shift @_; my $t = translate($s); + sprintf_fixutf8 $t, @_; +} + +my $in = interactive->vnew('su'); + +my $us = {} ; +$us->{VERSION} = 0.92; +my $error = 0; +my $GetValue = -65533; +my $stringsearch = ''; +my %prefs = getVarsFromSh($conffile); +my %sec = getVarsFromSh($secfile); +my $sysfilter = text2bool($prefs{FILTER}); + +sub HelpSystem { system("drakhelp Drakxtools-Guide.html/userdrake.html &") }; + +$us->{wnd} = ugtk2->new(N("Mandrake Linux Users Management Tool") . " " . $us->{VERSION}, center => 1); +gtkset_size_request($us->{wnd}{window}, 660, 460); +$us->{wnd}{rwindow}->signal_connect(delete_event => \&QuitGlobal); +my $utree_model = Gtk2::ListStore->new(Gtk2::GType->STRING, Gtk2::GType->INT, Gtk2::GType->STRING, Gtk2::GType->STRING, Gtk2::GType->STRING, Gtk2::GType->STRING ); +my $gtree_model = Gtk2::ListStore->new(Gtk2::GType->STRING, Gtk2::GType->INT, Gtk2::GType->STRING ); +my ($usertree, $grouptree); +$usertree = CreateTree($utree_model); $grouptree = CreateTree($gtree_model); +my ($checkedit, $checkdel, $checkall); +my ($menu, $factory) = create_factory_menu($us->{wnd}{rwindow}, + ({ path => N("/_File"), type => '<Branch>' }, + { path => N("/_File") . N("/_Refresh"), callback => sub { Refresh($sysfilter, $stringsearch)} }, + { path => N("/_File") . N("/_Quit"), callback => \&QuitGlobal }, + { path => N("/_Actions"), type => '<Branch>' }, + { path => N("/_Actions") . N("/_Add User"), callback => \&AddUser }, + { path => N("/_Actions") . N("/Add _Group"), callback => \&AddGroup }, + { path => N("/_Actions") . N("/_Edit"), callback => \&Edit }, + { path => N("/_Actions") . N("/_Delete"), callback => \&Delete }, + { path => N("/_Options"), type => '<Branch>' }, + { path => N("/_Options") . N("/_Filter system users"), type => '<CheckItem>', + callback => sub { $sysfilter = $checkall->active; Refresh($sysfilter, $stringsearch) } }, + { path => N("/_Help"), type => '<Branch>' }, + { path => N("/_Help").N("/_Help"), callback => \&HelpSystem }, + { path => N("/_Help").N("/_Report Bug"), callback => sub { system("drakhelp https://qa.mandrakesoft.com &") } }, + { path => N("/_Help").N("/_About..."), callback => \&About } + ) + ); +$checkall = $factory->get_widget("<main>" . N("/Options") . N("/Filter system users")); +$checkedit = $factory->get_widget("<main>" . N("/Actions") . N("/Edit")); +$checkdel = $factory->get_widget("<main>" . N("/Actions") . N("/Delete")); +if (defined $checkall) { + $checkall->set_active($sysfilter); +} else { + print STDERR "BUG with LANGUAGE $ENV{LANGUAGE}\n"; +} +my $toolb = Gtk2::Toolbar->new; +my $searchBox = Gtk2::HBox->new(0,5); +$searchBox->pack_end(my $fbut = Gtk2::Button->new("Apply filter"), 0, 0, 4); +$searchBox->pack_end(my $filter = Gtk2::Entry->new, 0, 0, 4); +gtkappend_page(my $nb = Gtk2::Notebook->new, gtkpack(create_scrolled_window($usertree)), gtkshow(Gtk2::Label->new(N("Users")))); +gtkappend_page($nb, gtkpack(create_scrolled_window($grouptree)), gtkshow(Gtk2::Label->new(N("Group Users")))); +$nb->set_show_border(0); +gtkadd($us->{wnd}{window}, + gtkpack_(Gtk2::VBox->new(0, 0), + 0, $menu, + 0, $toolb, + 0, $searchBox, + 1, $nb )); +my @ucolsize = (80, 50, 50, 120, 80, 80, -1); my @gcolsize =(100, 80, 160, -1) ; +each_index { + my $col = Gtk2::TreeViewColumn->new_with_attributes($_, Gtk2::CellRendererText->new, 'text' => $::i); + $col->set_min_width($ucolsize[$::i]); + $usertree->append_column($col); +} (N("User Name"), N("User ID"), N("Primary Group"), N("Full Name"), N("Login Shell"), N("Home Directory")); + +each_index { + my $col = Gtk2::TreeViewColumn->new_with_attributes($_, Gtk2::CellRendererText->new, 'text' => $::i); + $col->set_min_width($gcolsize[$::i]); + $grouptree->append_column($col); +} (N("Group Name"), N("Group ID"), N("Group Members")); +my @toolbwg; +map { + my $t = $toolb->append_item($_->[0], $_->[1], $_->[2], + Gtk2::Image->new_from_file($pixdir . $_->[2] . '.png'), $_->[3], $toolb); push(@toolbwg, $t); + $toolb->append_space; +} ( [ N("Add User"), N("Add a user to the system"), 'user_add', \&AddUser ], + [ N("Add Group"), N("Add a group to the system"), 'group_add', \&AddGroup ], + [ N("Edit"), N("Edit selected row"), 'user_conf', \&Edit ], + [ N("Delete"), N("Delete selected row"), 'user_del', \&Delete ], + [ N("Refresh"), N("Refresh the list"), 'refresh', sub { Refresh($sysfilter, $stringsearch) } ], + [ N("Help"), N("Generic help on userdrake"), 'help', \&HelpSystem ] + ); +my ($tbuser, $tbgroup, $tbedit, $tbdel, $tbref, $tbhelp) = @toolbwg; +foreach (($tbedit, $tbdel, $checkedit, $checkdel)) { $_->set_sensitive(0) }; +my $ctx = USER::ADMIN->new; +$fbut->signal_connect('clicked', sub { $stringsearch = $filter->get_text() ; Refresh($sysfilter, $stringsearch) }); +Refresh($sysfilter, $stringsearch); +my $tmm = ConvTime(19, 06, 2003); +$nb->signal_connect('switch-page' => sub { NotebookSwitch() }); +$us->{wnd}{rwindow}->show_all; +Gtk2->main; +ugtk2->exit; + +sub TreeUnselect { + my $treev = shift; + $treev->get_selection->unselect_all; + foreach (($tbedit, $tbdel, $checkedit, $checkdel)) { $_->set_sensitive(0) }; +} +sub NotebookSwitch { + my $page = $nb->get_current_page(); + TreeUnselect($usertree) ; TreeUnselect($grouptree); +} +sub RefreshUsersFull { + my ($filterusers, $strfilt) = @_; + my ($users, $gid, $group, $groupnm); + $users = $ctx->UsersEnumerateFull(); + $utree_model->clear(); + my @UserReal; + LOOP: foreach my $l (@$users) { next LOOP if ($filterusers && $l->Uid($GetValue) <= 499 || $l->Uid($GetValue) == 65534) ; push(@UserReal, $l) if $l->UserName($GetValue) =~ /^\Q$strfilt/; }; + foreach my $l (@UserReal) { + $a = $l->Gid($GetValue); $group = $ctx->LookupGroupById($a); $groupnm = ''; + $group and $groupnm = $group->GroupName($GetValue); + $utree_model->append_set([ 0 => $l->UserName($GetValue), 1 => $l->Uid($GetValue), 2 => $groupnm, 3 => $l->Gecos($GetValue), 4 => $l->LoginShell($GetValue), 5 => $l->HomeDir($GetValue) ]); + } +} +sub RefreshGroupsFull { + my ($filtergroups, $strfilt) = @_; + my $groups = $ctx->GroupsEnumerateFull(); + $gtree_model->clear(); + my @GroupReal; + LOOP:foreach my $g (@$groups) { next LOOP if ($filtergroups && $g->Gid($GetValue) <= 499 || $g->Gid($GetValue) == 65534); push(@GroupReal, $g) if $g->GroupName($GetValue) =~ /^\Q$strfilt/ }; + foreach my $g (@GroupReal) { + my $a = $g->Gid($GetValue); my $group = $ctx->LookupGroupById($a); + my $u_b_g = $ctx->EnumerateUsersByGroup($g->GroupName($GetValue)); + my $listUbyG = join(',', @$u_b_g); + $gtree_model->append_set([ 0 => $g->GroupName($GetValue), 1 => $g->Gid($GetValue), 2 => $listUbyG ]); + } +} +sub Refresh { + my ($filt, $strfilt) = @_; + RefreshUsersFull($filt, $strfilt); + RefreshGroupsFull($filt, $strfilt); +} +sub AddUser { + my $w = ugtk2->new(N("Create New User"), grab => 1); + my $dontcreatehomedir = 0; my $is_system = 0; + my %u; + gtkadd($w->{window}, + gtkpack_(Gtk2::VBox->new(0, 2), + 0, BuildUui(), + 0, Gtk2::HSeparator->new, + 0, $us->{o}->{createhomedir} = Gtk2::CheckButton->new(N("Create home Directory")), + 0, gtkpack_(Gtk2::HBox->new(0, 4), + 0, Gtk2::Label->new(N("Home Directory: ")), + 0, $us->{o}->{homedir} = Gtk2::Entry->new() + ), + 0, $us->{o}->{privategroup} = Gtk2::CheckButton->new(N("Create a private group for the user")), + 0, $us->{o}->{userid} = Gtk2::CheckButton->new(N("Specify user ID manually")), + 0, gtkset_sensitive(my $h = Gtk2::HBox->new(0, 4), 0), + 0, Gtk2::HSeparator->new, + 0, gtkpack(Gtk2::HBox->new(1, 20), + map { + my $r = $_->[1]; + gtksignal_connect(Gtk2::Button->new($_->[0]), clicked => sub { + if (!$r) { + $u{username} = $us->{o}->{login}->get_text(); $error = 0; + if (!valid($u{username})) { RaiseError($us->{error}) }; + my $nm = $ctx->LookupUserByName($u{username}); + if ($nm) { RaiseError(N("User already exists, please choose another User Name")); $us->{o}->{login}->set_text('') }; + $u{passwd} = $us->{o}->{passwd}->get_text(); + if ($u{passwd} ne $us->{o}->{confpasswd}->get_text()) { RaiseError(N("Password Mismatch")) }; + if ($sec{SECURE_LEVEL} > 3 && length($u{passwd}) < 6) { RaiseError(N("This password is too simple. \n Good passwords should be > 6 caracters")) }; + my $userEnt = $ctx->InitUser($u{username}, $is_system); + if ($us->{o}->{createhomedir}->get_active) { + $dontcreatehomedir = 0; + $u{homedir} = $us->{o}->{homedir}->get_text(); + $userEnt->HomeDir($u{homedir}); + } else { + $dontcreatehomedir = 1 + } + if ($us->{o}->{userid}->get_active) { + if (($u{uid} = $us->{o}->{uid}->get_value) < 500) { + my $uidchoice = GimmeChoice(N("User Uid is < 500"), N("Creating a user with a UID less than 500 is not recommended.\n Are you sure you want to do this?\n\n")); + $uidchoice and $userEnt->Uid($u{uid}); + } else { $userEnt->Uid($u{uid}) } + }; + if ($us->{o}{privategroup}->get_active) { + if (!$error) { + #Check if group exist + my $gr = $ctx->LookupGroupByName($u{username}); + if ($gr) { + my $groupchoice = ChooseGroup(); + if ($groupchoice == 0 && !$error) { + #You choose to put it in the existing group + $u{gid} = $gr->Gid($GetValue); + } elsif ($groupchoice == 1) { + # Put it in 'users' group + $u{gid} = Add2UsersGroup($u{username}); + } + } else { + #it's a new group: Add it + my $newgroup = $ctx->InitGroup($u{username},$is_system); + $u{gid} = $newgroup->Gid($GetValue); + $ctx->GroupAdd($newgroup); + } + } + } else { + $u{gid} = Add2UsersGroup($u{username}); + #debug Info (to be removed before final) + } + if(!$error) { + $u{gecos} = $us->{o}->{fullname}->get_text(); $u{loginshell} = $us->{o}->{shells}->entry->get_text(); + $userEnt->Gecos($u{gecos}); $userEnt->LoginShell($u{loginshell}); $userEnt->Gid($u{gid}); + $userEnt->ShadowMin(-1); $userEnt->ShadowMax(99999); + $userEnt->ShadowWarn(-1); $userEnt->ShadowInact(-1); + $ctx->UserAdd($userEnt, $is_system, $dontcreatehomedir); + $ctx->UserSetPass($userEnt, $u{passwd}); + Refresh($sysfilter, $stringsearch); + } + } + !$error and Gtk2->main_quit; $error = 0 }) + } ([ N("Cancel"), 1 ], [ N("Ok"), 0 ])), + ) + ); + map { $us->{o}->{$_}->set_active } (qw(privategroup createhomedir)); + $us->{o}->{login}->signal_connect('focus_out_event' => sub { + my $fullname = $us->{o}->{fullname}->get_text(); + $us->{o}->{homedir}->set_text("/home/".$us->{o}->{login}->get_text()); + $fullname or $us->{o}->{fullname}->set_text($us->{o}->{login}->get_text()) + }); + $us->{o}->{uid} = Gtk2::SpinButton->new(Gtk2::Adjustment->new(500, 1, 65000, 1, 10, 10), 1, 0); + $h->pack_end($us->{o}->{uid}, 0, 0, 4); + $h->pack_end(Gtk2::Label->new(N("UID: ")), 0, 0, 4); + GrayBox($us->{o}->{userid}, $h, 0); + $w->{rwindow}->show_all; + $w->main; +} +sub Add2UsersGroup { + my $name = shift; + my $usersgroup = $ctx->LookupGroupByName('users'); + $usersgroup->MemberName($name, 1); + return $usersgroup->Gid($GetValue); +} +sub ChooseGroup() { + my $w = ugtk2->new(N("Choose Group"), grab => 1, center => 1); + my $choice; + my @radio = gtkradio(N("Add to the existing group"), (N("Add to the existing group"), N("Add to the 'users' group"))); + gtkadd($w->{window}, + gtkpack__(Gtk2::VBox->new(0,5), + Gtk2::Label->new(N("A group with this name already exists. What would you like to do?")), + gtkpack(Gtk2::VBox->new(0,0), @radio), + Gtk2::HSeparator->new, + gtkpack(create_hbox(), + gtksignal_connect(Gtk2::Button->new(N("Ok")), clicked => sub { + each_index { $_->get_active and $choice = $::i } @radio; + Gtk2->main_quit; + }), + gtksignal_connect(Gtk2::Button->new(N("Cancel")), clicked => sub { $w->{retval} = 0; $error = 1; Gtk2->main_quit })))); + $w->main; + $choice; +} +sub GimmeChoice { + my ($title, $text) = @_; + my $w = ugtk2->new($title, grab => 1, center => 1); + my $choice; + gtkadd($w->{window}, + gtkpack__(Gtk2::VBox->new(0,5), + Gtk2::Label->new($text), + Gtk2::HSeparator->new, + gtkpack(create_hbox(), + gtksignal_connect(Gtk2::Button->new(N("Yes")), clicked => sub { + $choice = 1; + Gtk2->main_quit; + }), + gtksignal_connect(Gtk2::Button->new(N("No")), clicked => sub { $choice = $w->{retval} = 0; $error = 1; Gtk2->main_quit })))); + $w->main; + $choice; +} +sub AddGroup { + my $w = ugtk2->new(N("Create New User"), grab => 1); + my $mode = 0; my %g; my $is_system = 0; + gtkadd($w->{window}, + gtkpack_(Gtk2::VBox->new(0, 2), + 0, BuildGui(), + 0, Gtk2::HSeparator->new, + 0, $us->{o}->{groupid} = Gtk2::CheckButton->new(N("Specify group ID manually")), + 0, gtkset_sensitive(my $h = Gtk2::HBox->new(0, 4), 0), + 0, Gtk2::HSeparator->new, + 0, gtkpack(Gtk2::HBox->new(1, 20), + map { + my $r = $_->[1]; + gtksignal_connect(Gtk2::Button->new($_->[0]), clicked => sub { + if (!$r) { + $g{groupname} = $us->{o}->{groupname}->get_text(); $error = 0; + if (!valid($g{groupname})) { RaiseError($us->{error}) }; + my $nm = $ctx->LookupGroupByName($g{groupname}); + if ($nm) { RaiseError(N("Group already exists, please choose another Group Name")); $us->{o}->{groupname}->set_text('') }; + my $groupEnt = $ctx->InitGroup($g{groupname}, $is_system); + if ($us->{o}->{groupid}->get_active) { + if (($g{gid} = $us->{o}->{gid}->get_value) < 500) { + my $gidchoice = GimmeChoice(N(" Group Gid is < 500"), N("Creating a group with a GID less than 500 is not recommended.\n Are you sure you want to do this?\n\n")); + $gidchoice and $groupEnt->Gid($g{gid}); + } else { $groupEnt->Gid($g{gid}) } + }; + if(!$error) { + $ctx->GroupAdd($groupEnt); + Refresh($sysfilter, $stringsearch); + } + } + !$error and Gtk2->main_quit; $error = 0; }) + } ([ N("Cancel"), 1 ], [ N("Ok"), 0 ])), + ) + ); + $us->{o}->{gid} = Gtk2::SpinButton->new(Gtk2::Adjustment->new(500, 1, 65000, 1, 10, 10), 1, 0); + $h->pack_end($us->{o}->{gid}, 0, 0, 4); + $h->pack_end(Gtk2::Label->new(N("GID: ")), 0, 0, 4); + $us->{o}->{groupid}->signal_connect('clicked' => sub { $mode = !$mode; $h->set_sensitive($mode) }); + $w->{rwindow}->show_all; + $w->main; + +} +sub UpdateOrDelUsersInGroup { + my ($name, $action) = @_; + my $groups = $ctx->GroupsEnumerateFull(); + if ($action) { + foreach my $g (@$groups) { + my $members = $g->MemberName(1, 0); + if (InArray($name, $members)) { + eval { $g->MemberName($name, 2) }; + printf("\nAbout to modify %s\n", $g->GroupName($GetValue)); + eval { $ctx->GroupModify($g) }; + printf("\nModified %s\n", $g->GroupName($GetValue)); + } + } + } else { + print ("Updating MemberName \n"); + } +} +sub GetNameEntFromIter { + my ($tree, $model, $rank) = @_; + my (undef, $iter) = $tree->get_selection->get_selected; + my $name = $model->get($iter, $rank); + $iter->free; + $name +} +sub FillUserInfo { + my $ent = shift; + $us->{o}->{fullname}->set_text($ent->Gecos($GetValue)); + $us->{o}->{passwd}->set_text(' '); + $us->{o}->{confpasswd}->set_text(' '); + $us->{o}->{shells}->entry->set_text($ent->LoginShell($GetValue)); + $us->{o}->{homedir}->set_text($ent->HomeDir($GetValue)) +} +sub Delete { + my $page = $nb->get_current_page(); + $us->{wnd}{rwindow}->set_sensitive(0); + gtkset_mousecursor_wait(); $error = 0; + if ($page <= 0) { + my $username = GetNameEntFromIter($usertree, $utree_model, 0); + my $userEnt = $ctx->LookupUserByName($username); + $ctx->UserDel($userEnt); + UpdateOrDelUsersInGroup($username, 1); + #Let's check out the user's primary group + my $usergid = $userEnt->Gid($GetValue); + my $groupEnt = $ctx->LookupGroupById($usergid); my $member = $groupEnt->MemberName(1, 0); + if ($groupEnt && (scalar(@$member) == 0)) { $groupEnt->Gid($GetValue) > 499 and $ctx->GroupDel($groupEnt) }; + my $removehome = GimmeChoice(N(" Remove Home Directory"), N("Do you want to delete the user's home directory and mail spool?")); + $removehome and $ctx->Clean($userEnt); + Refresh($sysfilter, $stringsearch) + } elsif ($page == 1) { + my $groupname = GetNameEntFromIter($grouptree, $gtree_model, 0); + my $groupEnt = $ctx->LookupGroupByName($groupname); + my $members = $ctx->EnumerateUsersByGroup($groupname); + GLOOP:foreach my $username (@$members) { + my $userEnt = $ctx->LookupUserByName($username); + if ($userEnt && $userEnt->Gid($GetValue) == $groupEnt->Gid($GetValue)) { + RaiseError(N("%s is a primary group for user %s\n Remove the user first", $groupname, $username)); + last GLOOP + } + } + if (!$error) { $ctx->GroupDel($groupEnt); Refresh($sysfilter, $stringsearch) } + } + $us->{wnd}{rwindow}->set_sensitive(1); + gtkset_mousecursor_normal(); +} +# Gtk Facilities +sub CreateTree { + my ($tree_model) = @_; + my $tree = Gtk2::TreeView->new_with_model($tree_model); + $tree->get_selection->set_mode('browse'); + $tree->set_headers_visible(1); + $tree->set_rules_hint(1); + $tree->get_selection->signal_connect('changed' => sub { foreach (($tbedit, $tbdel, $checkedit, $checkdel)) { $_->set_sensitive(1);} }); + $tree->signal_connect(button_press_event => sub { + my (undef, $event) = @_; + my (undef, $iter) = $tree->get_selection->get_selected; + return unless $iter; + foreach (($tbedit, $tbdel, $checkedit, $checkdel)) { $_->set_sensitive(1) }; + Edit() if $event->type eq '2button-press'; + $iter->free; + }); + $tree->signal_connect(key_press_event => sub { + my (undef, $event) = @_; + my (undef, $iter) = $tree->get_selection->get_selected; + return unless $iter; + Edit() if ($event->keyval == Gtk2::Gdk::Event::Key->Sym_Return); + $iter->free; + }); + $tree +} +sub GtkEntryHidePass { + my ($text) = @_; + my $e = Gtk2::Entry->new(); + $text and $e->set_text($text); + $e->set_visibility(0); + $e +} +sub GtkEntrySized { + my $i = shift; + my $e = Gtk2::Entry->new_with_max_length($i); + $e->set_size_request(30, 20); + $e +} +sub BuildUui { + gtkpack_(my $vbox = Gtk2::VBox->new(0, 2), + 1, create_packtable({ homogeneous => 1, col_spacings => 5, row_spacings => 5 }, + [ N("Login") . " :", $us->{o}->{login} = Gtk2::Entry->new() ], + [ N("Full Name") . " :", $us->{o}->{fullname} = Gtk2::Entry->new() ], + [ N("Password") . " :", $us->{o}->{passwd} = GtkEntryHidePass() ], + [ N("Confirm Password:") . " :", $us->{o}->{confpasswd} = GtkEntryHidePass() ], + [ N("Login Shell") . " :", $us->{o}->{shells} = Gtk2::Combo->new ] + ) + ); + $us->{o}->{shells}->set_popdown_strings(@{$ctx->GetUserShells()}); + $us->{o}->{shells}->entry->set_text("/bin/bash"); + $vbox +} +sub BuildGui { + my $groupentry = @_; + gtkpack_(my $vbox = Gtk2::VBox->new(0, 2), + 1, create_packtable({ homogeneous => 1, col_spacings => 5, row_spacings =>5 }, + [ N("Group Name") . " :", $us->{o}->{groupname} = Gtk2::Entry->new() ] + ) + ); + $vbox +} +sub Edit { + my $w = ugtk2->new('userdrake', grab => 1); + my $model = Gtk2::ListStore->new(Gtk2::GType->OBJECT, Gtk2::GType->STRING, Gtk2::GType->INT); + my $tree = Gtk2::TreeView->new_with_model($model); my %g; my %u; + my ($groupname, $username, $groupEnt, $userEnt, $members, $times, $min, $max, $warn, $inact, $primgid, $temp); + my @primgroup; + my @pix = ($pixdir.'selected.png', $pixdir.'unselected.png'); + $tree->get_selection->set_mode('browse'); + $tree->append_column(my $check = Gtk2::TreeViewColumn->new_with_attributes(undef, Gtk2::CellRendererPixbuf->new, 'pixbuf' => 0)); + $tree->append_column(my $col = Gtk2::TreeViewColumn->new_with_attributes(undef, Gtk2::CellRendererText->new, 'text' => 1)); + $tree->set_headers_visible(0); + my $p = $nb->get_current_page(); + my $nbU = Gtk2::Notebook->new; my $nbG = Gtk2::Notebook->new; + if ($p <= 0) { + gtkappend_page($nbU, + gtkpack_(Gtk2::VBox->new(0, 2), + 1, BuildUui(), + 1, create_packtable({ homogeneous => 1, col_spacings => 5, row_spacings => 5 }, + [ N("Home") . " :", $us->{o}->{homedir} = Gtk2::Entry->new() ] + ) + ), gtkshow(Gtk2::Label->new(N("User Data")))); + my $m = 0; my $s = 0; + gtkappend_page($nbU, + gtkpack_(Gtk2::VBox->new(0, 2), + 0, $us->{o}->{acheckexpire} = Gtk2::CheckButton->new(N("Enable account expiration")), + 0, gtkpack__(my $h = Gtk2::HBox->new(0, 10), + 0, Gtk2::Label->new(N("Account expires MM/DD/YYYY :")), + 0, map { my $s = $_->[0]; my $t = $_->[1]; + $us->{o}->{$s} = GtkEntrySized($t), + } (['expm', 2], ['expd', 2], ['expy', 4]), + ), + 0, Gtk2::HSeparator->new, + 0, $us->{o}->{lockuser} = Gtk2::CheckButton->new(N("Lock User Account")) + ), gtkshow(Gtk2::Label->new(N("Account Info")))); + GrayBox($us->{o}->{acheckexpire}, $h, 0); + gtkappend_page($nbU, + gtkpack_(Gtk2::VBox->new(0, 2), + 0, gtkpack__( Gtk2::HBox->new(0,5), + 0, Gtk2::Label->new(N("User last changed password on : ")), + 0, my $dayStr = Gtk2::Label->new(""), 0, my $month = Gtk2::Label->new(""), 0, my $dayInt = Gtk2::Label->new(""), 0, my $year = Gtk2::Label->new("") + ), + 0, Gtk2::HSeparator->new, + 0, $us->{o}->{pcheckexpire} = Gtk2::CheckButton->new(N("Enable Password Expiration")), + 1, gtkpack_(my $v = Gtk2::VBox->new(0,1), + 1, create_packtable({ homogeneous => 1, col_spacings => 5, row_spacings => 5 }, + [ N("Days before change allowed :"), $us->{o}->{dbca} = Gtk2::Entry->new(0) ], + [ N("Days before change required :"), $us->{o}->{dbcr} = Gtk2::Entry->new(0) ], + [ N("Days warning before change :"), $us->{o}->{bwbc} = Gtk2::Entry->new(0) ], + [ N("Days before account inactive :"), $us->{o}->{dbai} = Gtk2::Entry->new(0) ] + ) + )), gtkshow(Gtk2::Label->new(N("Password Info")))); + GrayBox($us->{o}->{pcheckexpire}, $v, 0); + gtkappend_page($nbU, + gtkpack_(Gtk2::VBox->new(0, 2), + 0, Gtk2::Label->new(N("Select the groups that the user will be a member of:")), + 1, create_scrolled_window($tree), + 0, gtkpack_( Gtk2::HBox->new(0, 1), + 0, Gtk2::Label->new(N("Primary Group")), + 1, $us->{o}->{primgroup} = Gtk2::Combo->new, + ) + ), gtkshow(Gtk2::Label->new(N("Groups")))); + $username = GetNameEntFromIter($usertree, $utree_model,0); $us->{o}->{login}->set_text($username); + $userEnt = $ctx->LookupUserByName($username); FillUserInfo($userEnt); + my $Uid = $userEnt->Uid($GetValue); + my $expire = $userEnt->ShadowExpire($GetValue); + if ($expire && $expire != -1) { + $us->{o}->{acheckexpire}->set_active(1); $h->set_sensitive(1); + $times = TimeOfArray($expire, 1); + printf("day = %s ** Month = %s , Year = %s", $times->{dayint}, $times->{month}, $times->{year}); + $us->{o}->{expd}->set_text($times->{dayint}); $us->{o}->{expm}->set_text($times->{month}); + $us->{o}->{expy}->set_text($times->{year}); + } + #root account should never be locked + !$Uid and $us->{o}->{lockuser}->set_sensitive(0); + # Check if user account is locked + $ctx->IsLocked($userEnt) and $us->{o}->{lockuser}->set_active(1); + my $lastchg = $userEnt->ShadowLastChange($GetValue); + if ($lastchg) { $times = TimeOfArray($lastchg, 0); + $dayStr->set_text($times->{daystr}); $month->set_text($times->{month}); $dayInt->set_text($times->{dayint}); $year->set_text($times->{year}); + } + $min = $userEnt->ShadowMin($GetValue); $max =$userEnt->ShadowMax($GetValue); $warn = $userEnt->ShadowWarn($GetValue); $inact = $userEnt->ShadowInact($GetValue); + if ($min && $min != -1 || $max && $max != 99999 || $warn && $warn != 7 && $warn != -1 || $inact && $inact != -1) { + $us->{o}->{pcheckexpire}->set_active(1); $v->set_sensitive(1); + } + $min && $min != -1 and $us->{o}->{dbca}->set_text($min); $max && $max != -1 and $us->{o}->{dbcr}->set_text($max); + $warn && $warn != -1 and $us->{o}->{bwbc}->set_text($warn); $inact && $inact != -1 and $us->{o}->{dbai}->set_text($inact); + my $grps = $ctx->GroupsEnumerate(); my @sgroups = sort(@$grps); + $members = $ctx->EnumerateGroupsByUser($username); + $primgid = $userEnt->Gid($GetValue); + my $Gent = $ctx->LookupGroupById($primgid); + foreach my $group (@sgroups) { + if (any { $_ eq $group } @$members) { + $model->append_set([ 0 => gtkcreate_pixbuf($pix[0]), 1 => $group, 2 => 1]); push(@primgroup, $group); + } else { $model->append_set([ 0 => gtkcreate_pixbuf($pix[1]), 1 => $group, 2 => 0]); } + } + $us->{o}->{primgroup}->set_popdown_strings(@primgroup); + $Gent and $us->{o}->{primgroup}->entry->set_text($Gent->GroupName($GetValue)); + + } elsif ($p == 1) { + $nbG->set_size_request(300, 200); + gtkappend_page($nbG, gtkpack_(Gtk2::VBox->new(0, 2), 1, BuildGui()), gtkshow(Gtk2::Label->new(N("Group Data")))); + gtkappend_page($nbG, + gtkpack_(Gtk2::VBox->new(0, 1), + 0, Gtk2::Label->new(N("Select the users to join this group :")), + 1, create_scrolled_window($tree)), gtkshow(Gtk2::Label->new(N("Group Users")))); + $groupname = GetNameEntFromIter($grouptree, $gtree_model, 0); $us->{o}->{groupname}->set_text($groupname); + # Don't allow change on group name since there is a bug in lu_user_modify group + $us->{o}->{groupname}->set_editable(0); + $groupEnt = $ctx->LookupGroupByName($groupname); + my $users = $ctx->UsersEnumerate(); my @susers = sort(@$users); + $members = $ctx->EnumerateUsersByGroup($groupname); + foreach my $user (@susers) { + if (any { $_ eq $user } @$members) { + $model->append_set([ 0 => gtkcreate_pixbuf($pix[0]), 1 => $user, 2 => 1]); + } else { $model->append_set([ 0 => gtkcreate_pixbuf($pix[1]), 1 => $user, 2 => 0]); } + } + } + my ($ch, $name, $uEnt, $gEnt, $ugid, $Exp); + gtkadd($w->{window}, + gtkpack_(Gtk2::VBox->new(0,5), + 1, !$p ? $nbU : $nbG, + 0, Gtk2::HSeparator->new, + 0, gtkpack(Gtk2::HBox->new(1, 20), + map { + my $retv = $_->[1]; + gtksignal_connect(Gtk2::Button->new($_->[0]), clicked => sub { + if (!$retv) { + if ($p <= 0) { + $error = 0; + %u = ( + username => $us->{o}->{login}->get_text(), gecos => $us->{o}->{fullname}->get_text(), + homedir => $us->{o}->{homedir}->get_text(), pw => $us->{o}->{passwd}->get_text(), + confm => $us->{o}->{confpasswd}->get_text(), shell => $us->{o}->{shells}->entry->get_text() + ); + if(!valid($u{username})) { RaiseError($us->{error}) } + if ($u{pw} ne $u{confm}) { + RaiseError(N("Password Mismatch")); + } elsif (($u{pw} eq $u{confm}) && $u{pw} ne ' ') { + if ($sec{SECURE_LEVEL} > 3 && length($u{pw}) < 6) { RaiseError(N("This password is too simple. \n Good passwords should be > 6 caracters")) } + !$error and $ctx->UserSetPass($userEnt, $u{pw}); + } + if (!$error) { + $userEnt->UserName($u{username}); $userEnt->Gecos($u{gecos}); + $userEnt->HomeDir($u{homedir}); $userEnt->LoginShell($u{shell}); + $username = $userEnt->UserName($GetValue); + $model->foreach(sub { + my ($mod, $path, $iter) = @_; + $ch = $mod->get($iter, 2); + $name = $mod->get($iter, 1); + $gEnt = $ctx->LookupGroupByName($name); + $ugid = $gEnt->Gid($GetValue); + my $m = $gEnt->MemberName(1,0); + if ($ch == 1) { + if (!InArray($username, $m) && $primgid != $ugid){ + eval { $gEnt->MemberName($username, 1) }; + $ctx->GroupModify($gEnt); + } + } else { + if (InArray($username, $m)) { + eval { $gEnt->MemberName($username, 2) }; + $ctx->GroupModify($gEnt); + } + } + $iter->free; + return 0; + }, undef); + if ($us->{o}->{primgroup}->entry->get_text() eq '') { + RaiseError(N("Please select at least one group for the user")); + } elsif (!$error) { + my $ent = $ctx->LookupGroupByName($us->{o}->{primgroup}->entry->get_text()); + $ugid = $ent->Gid($GetValue); + $userEnt->Gid($ugid); + if ($us->{o}->{acheckexpire}->get_active()) { + print "test"; + my $yr = $us->{o}->{expy}->get_text(); my $mo = $us->{o}->{expm}->get_text(); + my $dy = $us->{o}->{expd}->get_text(); + $yr && $dy && $mo or RaiseError(N("Please specify Year, Month and Day \n for Account Expiration ")); + $Exp = ConvTime($dy, $mo, $yr); + !$error and $userEnt->ShadowExpire($Exp); + } else { $userEnt->ShadowExpire(ceil(-1)) } + if ($us->{o}->{pcheckexpire}->get_active()) { + my $allowed = int($us->{o}->{dbca}->get_text()); my $required = int($us->{o}->{dbcr}->get_text()); + my $warning = int($us->{o}->{bwbc}->get_text()); my $inactive = int($us->{o}->{dbai}->get_text()); + + $allowed && $required && $warning && $inactive or RaiseError(N("Please specify all field for password aging")); + if(!$error) { + $userEnt->ShadowMin($allowed); $userEnt->ShadowMax($required); + $userEnt->ShadowWarn($warning); $userEnt->ShadowInact($inactive); + } + } else { + $userEnt->ShadowMin(-1); $userEnt->ShadowMax(99999); + $userEnt->ShadowWarn(-1); $userEnt->ShadowInact(-1); + } + !$error and $ctx->UserModify($userEnt); + if ($us->{o}->{lockuser}->get_active()) { + !$ctx->IsLocked($userEnt) and $ctx->Lock($userEnt); + } else { $ctx->IsLocked($userEnt) and $ctx->UnLock($userEnt) } + !$error and Refresh($sysfilter, $stringsearch); + } + } + + } elsif ($p == 1) { + $g{groupname} = $us->{o}->{groupname}->get_text(); $error = 0; + if (!valid($g{groupname})) { RaiseError($us->{error}) }; + if (!$error && $groupname ne $g{groupname}) {$groupEnt->GroupName($g{groupname})} + $groupname = $groupEnt->GroupName($GetValue); + $members = $ctx->EnumerateUsersByGroup($groupname); + my $gid = $groupEnt->Gid($GetValue); + !$error and $model->foreach(sub { + my ($mod, $path, $iter) = @_; + $ch = $mod->get($iter, 2); + $name = $mod->get($iter, 1); + if ($ch == 1) { + if (!InArray($name, $members)) { + $uEnt = $ctx->LookupUserByName($name); + $ugid = $uEnt->Gid($GetValue); + if ($ugid != $gid) { + eval { $groupEnt->MemberName($name,1) }; + } + } + } else { + if (InArray($name, $members)) { + $uEnt = $ctx->LookupUserByName($name); + if ($uEnt->Gid($GetValue) == $groupEnt->Gid($GetValue)) { + $model->set($iter, [0 => gtkcreate_pixbuf($pix[0])]); + $model->set($iter, [2 => 1]); + RaiseError(N("You cannot remove user '%s' from their primary group", $name)); + } + if (!$error) { + eval { $groupEnt->MemberName($name,2) }; + } + } + } + $iter->free; + return 0; + }, undef); + if (!$error) { + $ctx->GroupModify($groupEnt); + Refresh($sysfilter, $stringsearch); + } + } + } + !$error and Gtk2->main_quit; $error = 0; }) + } ([ N("Cancel"), 1 ], [ N("Ok"), 0 ])), + )); + $tree->show; + $w->{rwindow}->show_all; + $tree->signal_connect(button_press_event => sub { + my ($returns, $path, $column) = $tree->get_path_at_pos($_[1]->x, $_[1]->y); + if ($returns) { + my $eq = Gtk2->equals($check, $column); + if ($eq) { + my $iter = Gtk2::TreeIter->new; + $iter = $model->get_iter($path); + if ($iter) { + my $cp = $model->get($iter, 2); + my $item = $model->get($iter, 1); + $model->set($iter, [0 => gtkcreate_pixbuf($pix[$cp])]); + $model->set($iter, [2 => !$cp]); + if ($p <= 0) { + if (!$cp) { + !InArray($item, \@primgroup) and push(@primgroup, $item) + } else { + InArray($item, \@primgroup) and @primgroup = RemoveFromArray($item, \@primgroup) + } + $us->{o}->{primgroup}->set_popdown_strings(@primgroup); + } + $iter->free; + } + } + $path->free; + } + }); + $w->main +} +sub ConvTime { + my ($day, $month, $year) = @_; + my ($tm, $days, $mon, $yr); + $mon = $month - 1; $yr = $year - 1900; + $tm = POSIX::mktime(0, 0, 0, $day, $mon, $yr); + $days = ceil($tm / (24 * 60 *60)); + return $days; +} +sub TimeOfArray { + my ($reltime, $cm) = @_; + my $h; my $t; my %mth = (Jan => 1, Feb => 2, Mar => 3, Apr=> 4, May => 5, Jun => 6, Jul => 7, Aug => 8, Sep => 9, Oct => 10, Nov => 11, Dec => 12); + $t = gmtime($reltime * 24 * 60 * 60) =~ /(\S+)\s+(\S+)\s+(\d+)\s+(\S+)\s+(\d+)/; + $h->{daystr} = $1; $h->{month} = $2; $h->{dayint} = $3; $h->{year} = $5 ; + $cm and $h->{month} = $mth{$2}; + $h +} +sub InArray { + my ($item, $arr) = @_; + return any { $item eq $_} @$arr +} +sub RemoveFromArray { + my ($item, $arr) = @_; + my ($t, $s) = partition { $item ne $_ } @$arr; + return @$t; +} +sub GrayBox { + my ($o, $v, $m) = @_; + $v->set_sensitive($m); + $o->signal_connect('clicked' => sub { $m = !$m; $v->set_sensitive($m) }); +} +sub NewDialog { + my ($title, $no_button) = @_; + my $dialog = gtkset_border_width(Gtk2::Dialog->new(), 10); + $dialog->set_transient_for($us->{wnd}{rwindow}); + $dialog->set_position('center-on-parent'); + $dialog->set_title($title); + $dialog->action_area->pack_start(gtkadd(Gtk2::HButtonBox->new, + gtksignal_connect(Gtk2::Button->new(N("Close")), clicked => sub { $dialog->destroy }) + ), + 0,0,0) unless $no_button; + gtkset_modal($dialog, 1); +} +sub About { + my $window_about = NewDialog(N("Userdrake")); + my $tree_model = Gtk2::TreeStore->new(Gtk2::GType->STRING, Gtk2::GType->STRING, Gtk2::GType->STRING); + my $list = Gtk2::TreeView->new_with_model($tree_model); + $list->can_focus(0); + each_index { $list->append_column(Gtk2::TreeViewColumn->new_with_attributes(undef, Gtk2::CellRendererText->new, 'text' => $::i)) } 0..2; + $list->set_headers_visible(0); + foreach my $row ([ N("Authors: "), '', '' ], [ '', 'Daouda Lo', '<daouda@mandrakesoft.com>' ], ['', '', '']) { + $tree_model->append_set(undef, [ map_index { $::i => $_ } @$row ])->free; + } + $list->get_selection()->set_mode('none'); + gtkpack_($window_about->vbox, + -r "$pixdir/userdrake.png" ? + (0, Gtk2::Image->new_from_file("$pixdir/userdrake.png")) : (1, gtkmodify_font(Gtk2::Label->new(N("Users Management \n") . $us->{VERSION}), 'Bold 18'),), + 0, Gtk2::Label->new("\n" . N("Copyright (C) 2003 Mandrakesoft SA") . "\n"), + 1, $list, + ); + $window_about->show_all; +} +sub valid { + $_[0] or $us->{error} = N("Name field is empty please provide a name"), return 0; + $_[0] =~ /^[a-z]+?[a-z0-9_-]*?$/ or $us->{error} = N("The name must contain only lower cased letters, numbers, `-' and `_'"), return 0; + length($_[0]) <= 32 or $us->{error} = N("Name is too long"), return 0; + return 1; +} +sub RaiseError { + my $w = ugtk2->new(N("Error"), grab => 1, center => 1, transient => $us->{wnd}->{rwindow}); + $error = 1; + gtkadd($w->{window}, + gtkpack_(Gtk2::VBox->new, + 1, Gtk2::Label->new($_[0]), + 0, Gtk2::HSeparator->new, + 0, gtksignal_connect(Gtk2::Button->new(N("Ok")), clicked => sub { Gtk2->main_quit }) + ), + ); + $w->main +} +sub QuitGlobal { + setVarsInSh($conffile, { + FILTER => bool2text($sysfilter), + }); + gtkset_mousecursor_normal(); + Gtk2->main_quit; +} |