mc/src/chown.c

349 lines
8.4 KiB
C
Raw Normal View History

1998-02-27 07:54:42 +03:00
/* Chown command -- for the Midnight Commander
Copyright (C) 1994, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
2007 Free Software Foundation, Inc.
1998-02-27 07:54:42 +03:00
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
1998-02-27 07:54:42 +03:00
*/
/** \file chown.c
* \brief Source: chown command
*/
1998-02-27 07:54:42 +03:00
#include <config.h>
2005-02-08 12:04:03 +03:00
#include <errno.h>
1998-02-27 07:54:42 +03:00
#include <stdio.h>
2005-02-08 12:04:03 +03:00
#include <string.h>
2009-02-06 02:30:45 +03:00
#include <pwd.h>
#include <grp.h>
1998-02-27 07:54:42 +03:00
#include <sys/types.h>
#include <sys/stat.h>
2005-02-08 12:04:03 +03:00
#include <unistd.h>
1998-02-27 07:54:42 +03:00
#include "lib/global.h"
#include "lib/tty/tty.h"
#include "lib/skin/skin.h"
#include "lib/vfs/mc-vfs/vfs.h"
#include "dialog.h"
1998-02-27 07:54:42 +03:00
#include "widget.h"
#include "strutil.h"
1998-02-27 07:54:42 +03:00
/* Needed for the extern declarations of integer parameters */
#include "dir.h"
#include "panel.h" /* Needed for the externs */
#include "chmod.h"
#include "main.h" /* update_panels() */
#include "layout.h" /* repaint_screen() */
1998-02-27 07:54:42 +03:00
#include "chown.h"
#include "wtools.h" /* For init_box_colors */
#define UX 5
#define UY 2
#define GX 27
#define GY 2
#define BX 5
#define BY 15
#define TX 50
#define TY 2
#define BUTTONS 5
#define B_SETALL B_USER
2003-09-11 01:33:12 +04:00
#define B_SETUSR (B_USER + 1)
#define B_SETGRP (B_USER + 2)
1998-02-27 07:54:42 +03:00
static int need_update, end_chown;
static int current_file;
static int single_set;
static WListbox *l_user, *l_group;
static struct {
int ret_cmd, flags, y, x;
const char *text;
1998-02-27 07:54:42 +03:00
} chown_but[BUTTONS] = {
1999-01-31 01:45:53 +03:00
{ B_CANCEL, NORMAL_BUTTON, 0, 53, N_("&Cancel") },
{ B_ENTER, DEFPUSH_BUTTON, 0, 40, N_("&Set") },
{ B_SETUSR, NORMAL_BUTTON, 0, 25, N_("Set &users") },
{ B_SETGRP, NORMAL_BUTTON, 0, 11, N_("Set &groups") },
{ B_SETALL, NORMAL_BUTTON, 0, 0, N_("Set &all") },
1998-02-27 07:54:42 +03:00
};
#define LABELS 5
static struct {
int y, x;
WLabel *l;
} chown_label [LABELS] = {
2004-08-15 22:23:25 +04:00
{ TY+2, TX+2, NULL },
{ TY+4, TX+2, NULL },
{ TY+6, TX+2, NULL },
{ TY+8, TX+2, NULL },
{ TY+10,TX+2, NULL }
1998-02-27 07:54:42 +03:00
};
1998-12-03 00:27:27 +03:00
static void
chown_refresh (Dlg_head *h)
1998-02-27 07:54:42 +03:00
{
common_dialog_repaint (h);
tty_setcolor (COLOR_NORMAL);
1998-02-27 07:54:42 +03:00
draw_box (h, UY, UX, 12, 21);
draw_box (h, GY, GX, 12, 21);
draw_box (h, TY, TX, 12, 19);
1998-02-27 07:54:42 +03:00
dlg_move (h, TY + 1, TX + 1);
tty_print_string (_(" Name "));
dlg_move (h, TY + 3, TX + 1);
tty_print_string (_(" Owner name "));
dlg_move (h, TY + 5, TX + 1);
tty_print_string (_(" Group name "));
dlg_move (h, TY + 7, TX + 1);
tty_print_string (_(" Size "));
dlg_move (h, TY + 9, TX + 1);
tty_print_string (_(" Permission "));
1998-02-27 07:54:42 +03:00
tty_setcolor (COLOR_HOT_NORMAL);
dlg_move (h, UY, UX + 1);
tty_print_string (_(" User name "));
dlg_move (h, GY, GX + 1);
tty_print_string (_(" Group name "));
dlg_move (h, TY, TX + 1);
tty_print_string (_(" File "));
1998-02-27 07:54:42 +03:00
}
1998-12-03 00:27:27 +03:00
static char *
next_file (void)
1998-02-27 07:54:42 +03:00
{
while (!current_panel->dir.list[current_file].f.marked)
1998-02-27 07:54:42 +03:00
current_file++;
return current_panel->dir.list[current_file].fname;
1998-02-27 07:54:42 +03:00
}
static cb_ret_t
chown_callback (Dlg_head *h, Widget *sender,
dlg_msg_t msg, int parm, void *data)
1998-02-27 07:54:42 +03:00
{
switch (msg) {
1998-02-27 07:54:42 +03:00
case DLG_DRAW:
chown_refresh (h);
return MSG_HANDLED;
default:
return default_dlg_callback (h, sender, msg, parm, data);
1998-02-27 07:54:42 +03:00
}
}
static Dlg_head *
1998-12-03 00:27:27 +03:00
init_chown (void)
1998-02-27 07:54:42 +03:00
{
int i;
struct passwd *l_pass;
struct group *l_grp;
Dlg_head *ch_dlg;
1998-02-27 07:54:42 +03:00
do_refresh ();
end_chown = need_update = current_file = 0;
single_set = (current_panel->marked < 2) ? 3 : 0;
1998-02-27 07:54:42 +03:00
ch_dlg =
create_dlg (0, 0, 18, 74, dialog_colors, chown_callback, "[Chown]",
_(" Chown command "), DLG_CENTER | DLG_REVERSE);
1998-02-27 07:54:42 +03:00
for (i = 0; i < BUTTONS - single_set; i++)
add_widget (ch_dlg,
button_new (BY + chown_but[i].y, BX + chown_but[i].x,
chown_but[i].ret_cmd, chown_but[i].flags,
_(chown_but[i].text), 0));
1998-02-27 07:54:42 +03:00
/* Add the widgets for the file information */
for (i = 0; i < LABELS; i++) {
chown_label[i].l =
label_new (chown_label[i].y, chown_label[i].x, "");
add_widget (ch_dlg, chown_label[i].l);
1998-02-27 07:54:42 +03:00
}
/* get new listboxes */
l_user = listbox_new (UY + 1, UX + 1, 10, 19, NULL);
l_group = listbox_new (GY + 1, GX + 1, 10, 19, NULL);
1998-02-27 07:54:42 +03:00
/* add fields for unknown names (numbers) */
listbox_add_item (l_user, 0, 0, _("<Unknown user>"), NULL);
listbox_add_item (l_group, 0, 0, _("<Unknown group>"), NULL);
1998-02-27 07:54:42 +03:00
/* get and put user names in the listbox */
setpwent ();
1998-02-27 07:54:42 +03:00
while ((l_pass = getpwent ())) {
listbox_add_item (l_user, LISTBOX_APPEND_SORTED, 0, l_pass->pw_name, NULL);
1998-02-27 07:54:42 +03:00
}
endpwent ();
/* get and put group names in the listbox */
setgrent ();
1998-02-27 07:54:42 +03:00
while ((l_grp = getgrent ())) {
listbox_add_item (l_group, LISTBOX_APPEND_SORTED, 0, l_grp->gr_name, NULL);
1998-02-27 07:54:42 +03:00
}
endgrent ();
/* add listboxes to the dialogs */
1998-02-27 07:54:42 +03:00
add_widget (ch_dlg, l_group);
add_widget (ch_dlg, l_user);
return ch_dlg;
1998-02-27 07:54:42 +03:00
}
1998-12-03 00:27:27 +03:00
static void
chown_done (void)
1998-02-27 07:54:42 +03:00
{
if (need_update)
update_panels (UP_OPTIMIZE, UP_KEEPSEL);
1998-02-27 07:54:42 +03:00
repaint_screen ();
}
static void
1998-12-03 00:27:27 +03:00
do_chown (uid_t u, gid_t g)
1998-02-27 07:54:42 +03:00
{
if (mc_chown (current_panel->dir.list [current_file].fname, u, g) == -1)
message (D_ERROR, MSG_ERROR, _(" Cannot chown \"%s\" \n %s "),
current_panel->dir.list [current_file].fname, unix_error_string (errno));
1998-02-27 07:54:42 +03:00
do_file_mark (current_panel, current_file, 0);
1998-02-27 07:54:42 +03:00
}
1998-12-03 00:27:27 +03:00
static void
apply_chowns (uid_t u, gid_t g)
1998-02-27 07:54:42 +03:00
{
char *fname;
need_update = end_chown = 1;
do_chown (u,g);
do {
fname = next_file ();
do_chown (u,g);
} while (current_panel->marked);
1998-02-27 07:54:42 +03:00
}
#define chown_label(n,txt) label_set_text (chown_label [n].l, txt)
1998-12-03 00:27:27 +03:00
void
chown_cmd (void)
1998-02-27 07:54:42 +03:00
{
char *fname;
struct stat sf_stat;
WLEntry *fe;
Dlg_head *ch_dlg;
1998-02-27 07:54:42 +03:00
uid_t new_user;
gid_t new_group;
char buffer [BUF_TINY];
1998-02-27 07:54:42 +03:00
do { /* do while any files remaining */
ch_dlg = init_chown ();
1998-02-27 07:54:42 +03:00
new_user = new_group = -1;
if (current_panel->marked)
1998-02-27 07:54:42 +03:00
fname = next_file (); /* next marked file */
else
fname = selection (current_panel)->fname; /* single file */
1998-02-27 07:54:42 +03:00
if (mc_stat (fname, &sf_stat) != 0) { /* get status of file */
1998-02-27 07:54:42 +03:00
destroy_dlg (ch_dlg);
break;
}
/* select in listboxes */
fe = listbox_search_text (l_user, get_owner(sf_stat.st_uid));
if (fe)
listbox_select_entry (l_user, fe);
fe = listbox_search_text (l_group, get_group(sf_stat.st_gid));
if (fe)
listbox_select_entry (l_group, fe);
chown_label (0, str_trunc (fname, 15));
chown_label (1, str_trunc (get_owner (sf_stat.st_uid), 15));
chown_label (2, str_trunc (get_group (sf_stat.st_gid), 15));
size_trunc_len (buffer, 15, sf_stat.st_size, 0);
1998-02-27 07:54:42 +03:00
chown_label (3, buffer);
chown_label (4, string_perm (sf_stat.st_mode));
run_dlg (ch_dlg);
switch (ch_dlg->ret_value) {
case B_CANCEL:
end_chown = 1;
break;
case B_SETUSR:
{
struct passwd *user;
user = getpwnam (l_user->current->text);
if (user){
new_user = user->pw_uid;
apply_chowns (new_user, new_group);
}
break;
}
case B_SETGRP:
{
struct group *grp;
grp = getgrnam (l_group->current->text);
if (grp){
new_group = grp->gr_gid;
apply_chowns (new_user, new_group);
}
break;
}
case B_SETALL:
case B_ENTER:
{
struct group *grp;
struct passwd *user;
grp = getgrnam (l_group->current->text);
if (grp)
new_group = grp->gr_gid;
user = getpwnam (l_user->current->text);
if (user)
new_user = user->pw_uid;
if (ch_dlg->ret_value==B_ENTER) {
need_update = 1;
if (mc_chown (fname, new_user, new_group) == -1)
message (D_ERROR, MSG_ERROR, _(" Cannot chown \"%s\" \n %s "),
1998-02-27 07:54:42 +03:00
fname, unix_error_string (errno));
} else
apply_chowns (new_user, new_group);
break;
}
}
if (current_panel->marked && ch_dlg->ret_value != B_CANCEL){
do_file_mark (current_panel, current_file, 0);
1998-02-27 07:54:42 +03:00
need_update = 1;
}
destroy_dlg (ch_dlg);
} while (current_panel->marked && !end_chown);
1998-02-27 07:54:42 +03:00
chown_done ();
}