dsIIr4/bin/
dsIIr4/extra/creremote/
dsIIr4/extra/wolfpaw/
dsIIr4/lib/cmds/admins/
dsIIr4/lib/cmds/common/
dsIIr4/lib/cmds/creators/include/
dsIIr4/lib/cmds/creators/include/SCCS/
dsIIr4/lib/daemon/services/
dsIIr4/lib/doc/
dsIIr4/lib/domains/Ylsrim/
dsIIr4/lib/domains/Ylsrim/adm/
dsIIr4/lib/domains/Ylsrim/armor/
dsIIr4/lib/domains/Ylsrim/broken/
dsIIr4/lib/domains/Ylsrim/fish/
dsIIr4/lib/domains/Ylsrim/meal/
dsIIr4/lib/domains/Ylsrim/npc/
dsIIr4/lib/domains/Ylsrim/virtual/
dsIIr4/lib/domains/Ylsrim/weapon/
dsIIr4/lib/domains/campus/adm/
dsIIr4/lib/domains/campus/etc/
dsIIr4/lib/domains/campus/meals/
dsIIr4/lib/domains/campus/npc/
dsIIr4/lib/domains/campus/save/
dsIIr4/lib/domains/campus/txt/
dsIIr4/lib/domains/campus/txt/ai/charles/
dsIIr4/lib/domains/campus/txt/ai/charles/bak2/
dsIIr4/lib/domains/campus/txt/ai/charles/bak2/bak1/
dsIIr4/lib/domains/campus/txt/ai/charly/
dsIIr4/lib/domains/campus/txt/ai/charly/bak/
dsIIr4/lib/domains/campus/txt/jenny/
dsIIr4/lib/domains/default/creator/
dsIIr4/lib/domains/default/doors/
dsIIr4/lib/domains/default/etc/
dsIIr4/lib/domains/default/virtual/
dsIIr4/lib/domains/default/weap/
dsIIr4/lib/domains/town/virtual/
dsIIr4/lib/lib/comp/
dsIIr4/lib/lib/lvs/
dsIIr4/lib/lib/user/
dsIIr4/lib/lib/virtual/
dsIIr4/lib/log/
dsIIr4/lib/obj/book_source/
dsIIr4/lib/obj/include/
dsIIr4/lib/realms/template/
dsIIr4/lib/realms/template/adm/
dsIIr4/lib/realms/template/area/armor/
dsIIr4/lib/realms/template/area/npc/
dsIIr4/lib/realms/template/area/obj/
dsIIr4/lib/realms/template/area/room/
dsIIr4/lib/realms/template/area/weap/
dsIIr4/lib/realms/template/bak/
dsIIr4/lib/realms/template/cmds/
dsIIr4/lib/save/
dsIIr4/lib/save/kills/o/
dsIIr4/lib/secure/cfg/classes/
dsIIr4/lib/secure/cmds/creators/include/
dsIIr4/lib/secure/cmds/players/
dsIIr4/lib/secure/cmds/players/include/
dsIIr4/lib/secure/daemon/include/
dsIIr4/lib/secure/lib/
dsIIr4/lib/secure/lib/include/
dsIIr4/lib/secure/lib/net/include/
dsIIr4/lib/secure/lib/std/
dsIIr4/lib/secure/modules/
dsIIr4/lib/secure/npc/
dsIIr4/lib/secure/obj/include/
dsIIr4/lib/secure/room/
dsIIr4/lib/secure/save/
dsIIr4/lib/secure/save/boards/
dsIIr4/lib/secure/save/players/g/
dsIIr4/lib/secure/tmp/
dsIIr4/lib/secure/verbs/creators/
dsIIr4/lib/shadows/
dsIIr4/lib/spells/
dsIIr4/lib/std/board/
dsIIr4/lib/std/lib/
dsIIr4/lib/tmp/
dsIIr4/lib/verbs/admins/include/
dsIIr4/lib/verbs/common/
dsIIr4/lib/verbs/common/include/
dsIIr4/lib/verbs/creators/include/
dsIIr4/lib/verbs/players/include/SCCS/
dsIIr4/lib/verbs/rooms/
dsIIr4/lib/verbs/rooms/include/
dsIIr4/lib/www/
dsIIr4/v22.2b14-dsouls2/
dsIIr4/v22.2b14-dsouls2/ChangeLog.old/
dsIIr4/v22.2b14-dsouls2/Win32/
dsIIr4/v22.2b14-dsouls2/compat/
dsIIr4/v22.2b14-dsouls2/compat/simuls/
dsIIr4/v22.2b14-dsouls2/include/
dsIIr4/v22.2b14-dsouls2/mudlib/
dsIIr4/v22.2b14-dsouls2/testsuite/
dsIIr4/v22.2b14-dsouls2/testsuite/clone/
dsIIr4/v22.2b14-dsouls2/testsuite/command/
dsIIr4/v22.2b14-dsouls2/testsuite/data/
dsIIr4/v22.2b14-dsouls2/testsuite/etc/
dsIIr4/v22.2b14-dsouls2/testsuite/include/
dsIIr4/v22.2b14-dsouls2/testsuite/inherit/
dsIIr4/v22.2b14-dsouls2/testsuite/inherit/master/
dsIIr4/v22.2b14-dsouls2/testsuite/log/
dsIIr4/v22.2b14-dsouls2/testsuite/single/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/compiler/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/efuns/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/operators/
dsIIr4/v22.2b14-dsouls2/testsuite/u/
dsIIr4/v22.2b14-dsouls2/tmp/
dsIIr4/win32/
/*    /secure/cmds/creator/update.c
 *    from the Dead Souls LPC Library
 *    destroys an object then reloads it
 *    created by Descartes of Borg 950330
 *    - added recursive update option (blitz 960711)
 *    Version: @(#) update.c 1.2@(#)
 *    Last modified: 96/12/17
 */

#include <lib.h>
#include <rooms.h>

inherit LIB_DAEMON;
inherit LIB_HELP;

#define U_RECURSIVE       (1 << 1)
#define U_INTERACTIVE     (1 << 2)
#define U_AUTOMATED       (1 << 3)

mapping LocationsMap = ([]);

static int eventUpdate(string args, int flags);

static void CacheAndCarry(object *obs){
    if(!sizeof(obs)) return;
    foreach(object fellow in obs){
	string ubi = fellow->GetProperty("LastLocation");
	if(ubi) LocationsMap[fellow->GetKeyName()] = ubi;
    }
    obs->eventMove(ROOM_VOID);
}

static void ReturnAndRelease(object *dudes, string file){
    if(!sizeof(dudes)) return;
    if(!file) return;
    dudes->eventMove(file);
    foreach(object fellow in dudes){
	if(sizeof(LocationsMap[fellow->GetKeyName()])){
	    fellow->SetProperty("LastLocation",LocationsMap[fellow->GetKeyName()]);
	}
    }
}


static void create() {
    SetHelp("Syntax: <update [-r] [file list]>\n"
      "Destructs the master copy of the file named "
      "and then attempts to reload a new version "
      "of it.\n"
      "The -r flag attempts to update all "
      "files in the target's inheritance tree.\n");
}

mixed cmd(string args) {
    object *obs, ob;
    string *files, *tmpfiles;
    mixed tmp;
    string file;
    int i, flags;

    if( sizeof(args) ) {
	string str = args;
	args = "";
	foreach(string foo in explode(str, " ")) {
	    if(!sizeof(foo)) continue;
	    switch(foo) {
	    case "-r" : flags |= U_RECURSIVE; break;
	    case "-e" : flags |= U_INTERACTIVE; break;
	    case "-a" : flags |= U_AUTOMATED; break;
	    default: args += " " + foo;
	    }   
	}
    }
    if( args == "" || !args ) {
	if(!this_player()) return "No player.";
	ob = environment(this_player());
	if( !ob ) return "You have no environment.";
	file = base_name(ob);
	this_player()->eventPrint("Updating environment");
	obs = filter(all_inventory(ob), (: userp :));
	//if( sizeof(obs) ) obs->eventMove(ROOM_VOID);
	if( sizeof(obs) ) CacheAndCarry(obs);
	if( !eventUpdate(base_name(ob), flags) ) {
	    obs->eventPrint("You are thrown into the void as your "
	      "surroundings violently destruct.");
	    return "Error in reloading environment.";
	}
	obs = filter(obs, (: $1 :));
	//if( sizeof(obs) ) obs->eventMove(file);
	if( sizeof(obs) ) ReturnAndRelease(obs, file);
	return 1;
    }
    if(this_player()){
	tmpfiles = map(explode(args, " "),
	  function(string x) {
	      string tmp = (string)this_player()->query_cwd();
	      if( x[<2..] != ".c" ) x = x + ".c";
	      return absolute_path(tmp, x);
	    });
	  tmpfiles = map(tmpfiles,
	    (: ((file_size($1) == -2) ?
		(($1[<1] == '/') ? ($1 + "*.c") : ($1 + "/*.c")) : $1)
	    :));
	  i = sizeof(tmpfiles);
	  files = ({});
	  while(i--) {
	      if( sizeof(tmp = (string *)this_player()->wild_card(tmpfiles[i])) )
		  files += tmp;
	      else this_player()->eventPrint(tmpfiles[i] + ": File not found.");
	  }
	  i = sizeof(files);
	  while(i--) eventUpdate(files[i], flags);
      }
	return 1;
    }

    static int eventUpdate(string args, int flags) {
	object ob;
	string tmp;

	if( flags & U_RECURSIVE ) {
	    string *ancestors;
	    int i;

	    if( !eventUpdate(args, flags ^ U_RECURSIVE) ) return 0;
	    if( !(ob = find_object(args)) ) return 0;
	    ancestors = deep_inherit_list(ob);
	    if(this_player() && (flags & U_RECURSIVE) && !(flags & U_AUTOMATED))  
		this_player()->eventPrint("(%^CYAN%^Recursive "
		  "update: " + args + "%^RESET%^)\n");
	    i = sizeof(ancestors);
	    while(i--) if( !eventUpdate(ancestors[i], flags ^ U_RECURSIVE) ) {
		    if(this_player()) 
			this_player()->eventPrint("Recursive update failed.");
		    return 0;
		}        
	}
	if( args[<2..] == ".c" ) args = args[0..<3];
	ob = find_object(args);
	if(!ob) ob = load_object(args);
	if( ob ) {
	    if( tmp = catch( ob->eventDestruct()) && this_player() )
		this_player()->eventPrint(args + ": error in eventDestruct()");
	    if( ob ) destruct(ob);
	    if( ob && this_player())
		this_player()->eventPrint(args + ": Failed to destruct old object.");
	}
	if( args == base_name(this_object()) && this_player() ) {
	    this_player()->eventPrint("Cannot reload update after destruct.\n"
	      "It will be reloaded at next reference.");
	    return 0;
	}
	tmp = catch(call_other(args, "???"));
	if(this_player() && !(flags & U_AUTOMATED) ){
	    if( !tmp ) 
		this_player()->eventPrint(args + ": Ok");
	    else this_player()->eventPrint(args + ": Error in update\n" + tmp);
	}
	return 1;
    }