Help with massive uid/gid change.

Tom Christiansen tchrist at convex.COM
Sat Dec 1 08:36:05 AEST 1990


In article <2365 at taurus.BITNET> <ofer%math.tau.ac.il at CUNYVM.CUNY.EDU> writes:
>Hello
>I have to change the uid/gid numbers for all the users on a system
>I'd like to do it doing one pass over the file system and not by
>doing n times 'find ....' for n=the number of users.
>I have an old /etc/passwd file and a new one and wish to use them
>as the base for this number swapping process.
>I have perl and all standard unix available.

I believe that UofCO has a compiled version of this kind of thing that's
quite quick by going under the file system, which may or may not bother
you.  I don't have a copy though, so wrote my own.

This is how it works.  You make a file, called "howtomv" by default.
In it you put new mappings, for example:

    #comment
    user jimbob 10002      # more comment
    user stevebob 854
    group bobbros 200
    user daemon 1854
    user notes 1

Then you run the script.  It will give you a passwd.new and a group.new
file that you can inspect, and it won't really do the work if you use -n.
I've checked for a lot of blunders, but not all.  One thing I don't check
for is if userid A wants to move to userid B and B is occupied, it won't
let you.  It should really check to see whether B is moving and leaving a
vacancy, but it doesn't.  By all means inspect the code closely first, run
with -n to see what it wants to do, and keep a backup of your disk.  No
warranties, etc etc.  Still, it worked well enough for me when I needed it.

--tom

#!/usr/bin/perl
#
# mvids - "moves" uids and gids
# Tom Christiansen <tchrist at convex.com>
#
# usage: mvids [-n] [-f howtomvfile] [starting-dir]
#	
# Takes list of new user and group ids.
# Fixes passwd and group files.
# Traverses local file system starting with starting-dir
#  	updating all changed ids
#
# -n means don't really change anything 
# -f is if you don't like the default description file name of howtomv


# read descriptions from howtomv file with format: 
#	type name number
# e.g.:
#	user tom 1023
#	group staff 200


$| = 1;
$oops = 0;

require 'getopts.pl';

do Getopts('dnf:');

$FILE = $opt_f || "howtomv"; 

$DIR = $opt_d ? "." : "/etc";

$topdir = shift || '/';

die "usage: $0 [-n] [-f howtomv] [starting-dir]\n" 	if $#ARGV > -1;

die "$topdir: Not a directory"	unless -d $topdir;

open FILE || die "Can't open directions file \"$FILE\": $!\n";
while (<FILE>) {
    s/\s*#.*//;
    next if /^$/;
    unless (/^(user|group)\s+(\w+)\s+(\d+)/) {
	print STDERR "malformed line at line $. of $FILE: $_";
	$oops++; next;
    } 
    if ($3 > 32000) {
	print STDERR "$1 $2 has id that's too big ($3)\n";
	$oops++; next;
    } 
    if ($3 == 0) {
	print STDERR "Too dangerous to move $1 $2 to 0\n";
	$oops++; next;
    } 
    if ($2 eq 'root') {
	print STDERR "You don't really want to move root\n";
	$oops++; $next;
    } 
    if ($1 eq 'user') {
	if (defined $n_pwn2i{$2}) {
	    print STDERR "Saw user $2 again at line $. of $FILE\n";
	    $oops++; next;
	}
	if (defined $n_pwi2n{$3}) {
	    print STDERR "Saw uid $3 again at line $. of $FILE\n";
	    $oops++; next;
	}
	$uids++;
	$n_pwn2i{$2} = $3;
	$n_pwi2n{$3} = $2;
    } else {
	if (defined $n_grn2i{$2}) {
	    print STDERR "Saw group $2 again at line $. of $FILE\n";
	    $oops++; next;
	} 
	if (defined $n_gri2n{$3}) {
	    print STDERR "Saw gid $3 again at line $. of $FILE\n";
	    $oops++; next;
	}
	$gids++;
	$n_grn2i{$2} = $3;
	$n_gri2n{$3} = $2;
    } 
} 

$PWD  = "$DIR/passwd";
$NPWD = "$PWD.new";

if ($uids) {
    open PWD 			|| die "Can't open $PWD: $!\n";
    open (NPWD, ">$NPWD") 	|| die "Can't create $NPWD: $!\n";

    while (<PWD>) {
	((($name,$uid) = /^(\w+):[^:]*:(\d+):/)) 
	    || die "Bad passwd entry at line $.\n";
	if (defined $n_pwi2n{$uid} && !defined $n_pwn2i{$name}) {
	    printf STDERR "Can't move user %s to uid %d -- %s already has it\n",
		    $n_pwi2n{$uid}, $uid, $name;
	    $oops++;
	    next;
	} 
	$pwn2i{$name} = $uid;
	s/:$uid:/:$n_pwn2i{$name}:/	if defined $n_pwn2i{$name};
	print NPWD;
    }
    close PWD;
    close NPWD;

    foreach $user (keys %pwnam) {
	unless (defined $pwn2i{$user}) {
	    print STDERR "Can't move non-existent user $user\n";
	    $oops++;
	} 
    } 

}

if ($gids) {
    $GRP = "$DIR/group";
    $NGRP = "$GRP.new";
    open GRP 			|| die "Can't open $GRP: $!\n";
    open (NGRP , ">$NGRP") 	|| die "Can't create $NGRP: $!\n";

    while (<GRP>) {
	((($name,$gid) = /^(\w+):[^:]*:(\d+):/)) 
	    || die "Bad group entry at line $.\n";
	if (defined $n_gri2n{$gid} && !defined $n_grn2i{$name}) {
	    printf STDERR "Can't move gid %s to %d -- %s already has it\n",
		    $n_gri2n{$gid}, $gid, $name;
	    $oops++;
	    next;
	} 
	$grn2i{$name} = $gid;
	s/:$gid:/:$n_grn2i{$name}:/	if defined $n_grn2i{$name};
	print NGRP;
    } 
    close GRP;
    close NGRP;

    foreach $group (keys %grnam) {
	unless (defined $grn2i{$group}) {
	    print STDERR "Can't move non-existent group $group\n";
	    $oops++;
	} 
    } 

}

die "$0: $oops error" . ($oops > 1 ? "s" : "").
	" in remapping directions.\n"		if $oops;


die "$0: no ids to move\n" unless $uids || $gids;

# ok, now do it

open(FIND, "find $topdir \\( -fstype nfs -prune \\) -o -ls |")
    || die "Can't open find pipe";

while (<FIND>) {
    split;
    $uid = $gid = -1;
    ($file, $user, $group) = ($_[11], $_[5], $_[6]);

    if (defined $n_pwn2i{$user}) {
	$uid = $n_pwn2i{$user};
	print "changing owner $user of $file from ",
		"$pwn2i{$user} to $n_pwn2i{$user}\n";
    }
    if (defined $n_grn2i{$group}) {
	$gid = $n_grn2i{$group};
	print "changing group $group of $file from ",
		"$grn2i{$group} to $n_grn2i{$group}\n";
    }

    if (!$opt_n && ($uid != -1 || $gid != -1)) {
	if (!chown $uid, $gid, $file) {
	    printf STDERR "couldn't chown $file to $uid.$gid: $!\n";
	    $oops++;
	} 
    } 
} 

unless ($opt_n) {
    if ($uids) {
	rename($PWD, "$PWD.bak") 	
	    || die "Can't mv $PWD to $PWD.bak: $!\n";
	rename($NPWD, $PWD)		
	    || die "Can't mv $NPWD to $PWD: $!\n";
    }
    if ($gids) {
	rename($GRP, "$GRP.bak")
	    || die "Can't mv $GRP to $GRP.bak: $!\n";
	rename($NGRP, $GRP)
	    || die "Can't mv $NGRP to $GRP: $!\n";
    }
} 

exit ($oops != 0);



More information about the Alt.sources mailing list