ds2.10/bin/
ds2.10/extra/
ds2.10/extra/crat/
ds2.10/extra/creremote/
ds2.10/extra/mingw/
ds2.10/extra/wolfpaw/
ds2.10/fluffos-2.16-ds05/
ds2.10/fluffos-2.16-ds05/Win32/
ds2.10/fluffos-2.16-ds05/compat/
ds2.10/fluffos-2.16-ds05/compat/simuls/
ds2.10/fluffos-2.16-ds05/include/
ds2.10/fluffos-2.16-ds05/testsuite/
ds2.10/fluffos-2.16-ds05/testsuite/clone/
ds2.10/fluffos-2.16-ds05/testsuite/command/
ds2.10/fluffos-2.16-ds05/testsuite/data/
ds2.10/fluffos-2.16-ds05/testsuite/etc/
ds2.10/fluffos-2.16-ds05/testsuite/include/
ds2.10/fluffos-2.16-ds05/testsuite/inherit/
ds2.10/fluffos-2.16-ds05/testsuite/inherit/master/
ds2.10/fluffos-2.16-ds05/testsuite/log/
ds2.10/fluffos-2.16-ds05/testsuite/single/
ds2.10/fluffos-2.16-ds05/testsuite/single/tests/compiler/
ds2.10/fluffos-2.16-ds05/testsuite/single/tests/efuns/
ds2.10/fluffos-2.16-ds05/testsuite/single/tests/operators/
ds2.10/fluffos-2.16-ds05/testsuite/u/
ds2.10/lib/cmds/admins/
ds2.10/lib/cmds/common/
ds2.10/lib/cmds/creators/include/
ds2.10/lib/daemon/services/
ds2.10/lib/daemon/tmp/
ds2.10/lib/doc/
ds2.10/lib/doc/bguide/
ds2.10/lib/doc/efun/all/
ds2.10/lib/doc/efun/arrays/
ds2.10/lib/doc/efun/buffers/
ds2.10/lib/doc/efun/compile/
ds2.10/lib/doc/efun/floats/
ds2.10/lib/doc/efun/functions/
ds2.10/lib/doc/efun/general/
ds2.10/lib/doc/efun/mixed/
ds2.10/lib/doc/efun/numbers/
ds2.10/lib/doc/efun/parsing/
ds2.10/lib/doc/help/classes/
ds2.10/lib/doc/help/races/
ds2.10/lib/doc/lfun/
ds2.10/lib/doc/lfun/all/
ds2.10/lib/doc/lfun/lib/abilities/
ds2.10/lib/doc/lfun/lib/armor/
ds2.10/lib/doc/lfun/lib/bank/
ds2.10/lib/doc/lfun/lib/bot/
ds2.10/lib/doc/lfun/lib/clay/
ds2.10/lib/doc/lfun/lib/clean/
ds2.10/lib/doc/lfun/lib/clerk/
ds2.10/lib/doc/lfun/lib/client/
ds2.10/lib/doc/lfun/lib/combat/
ds2.10/lib/doc/lfun/lib/connect/
ds2.10/lib/doc/lfun/lib/container/
ds2.10/lib/doc/lfun/lib/corpse/
ds2.10/lib/doc/lfun/lib/creator/
ds2.10/lib/doc/lfun/lib/daemon/
ds2.10/lib/doc/lfun/lib/damage/
ds2.10/lib/doc/lfun/lib/deterioration/
ds2.10/lib/doc/lfun/lib/donate/
ds2.10/lib/doc/lfun/lib/door/
ds2.10/lib/doc/lfun/lib/equip/
ds2.10/lib/doc/lfun/lib/file/
ds2.10/lib/doc/lfun/lib/fish/
ds2.10/lib/doc/lfun/lib/fishing/
ds2.10/lib/doc/lfun/lib/flashlight/
ds2.10/lib/doc/lfun/lib/follow/
ds2.10/lib/doc/lfun/lib/ftp_client/
ds2.10/lib/doc/lfun/lib/ftp_data_connection/
ds2.10/lib/doc/lfun/lib/fuel/
ds2.10/lib/doc/lfun/lib/furnace/
ds2.10/lib/doc/lfun/lib/genetics/
ds2.10/lib/doc/lfun/lib/holder/
ds2.10/lib/doc/lfun/lib/id/
ds2.10/lib/doc/lfun/lib/interactive/
ds2.10/lib/doc/lfun/lib/lamp/
ds2.10/lib/doc/lfun/lib/leader/
ds2.10/lib/doc/lfun/lib/light/
ds2.10/lib/doc/lfun/lib/limb/
ds2.10/lib/doc/lfun/lib/living/
ds2.10/lib/doc/lfun/lib/load/
ds2.10/lib/doc/lfun/lib/look/
ds2.10/lib/doc/lfun/lib/manipulate/
ds2.10/lib/doc/lfun/lib/meal/
ds2.10/lib/doc/lfun/lib/messages/
ds2.10/lib/doc/lfun/lib/player/
ds2.10/lib/doc/lfun/lib/poison/
ds2.10/lib/doc/lfun/lib/position/
ds2.10/lib/doc/lfun/lib/post_office/
ds2.10/lib/doc/lfun/lib/potion/
ds2.10/lib/doc/lfun/lib/room/
ds2.10/lib/doc/lfun/lib/server/
ds2.10/lib/doc/lfun/lib/spell/
ds2.10/lib/doc/lfun/lib/torch/
ds2.10/lib/doc/lfun/lib/vendor/
ds2.10/lib/doc/lfun/lib/virt_sky/
ds2.10/lib/doc/lfun/lib/weapon/
ds2.10/lib/doc/lfun/lib/worn_storage/
ds2.10/lib/doc/lpc/constructs/
ds2.10/lib/doc/lpc/etc/
ds2.10/lib/doc/lpc/intermediate/
ds2.10/lib/doc/lpc/types/
ds2.10/lib/doc/misc/
ds2.10/lib/doc/old/
ds2.10/lib/doc/phints/
ds2.10/lib/domains/
ds2.10/lib/domains/Praxis/adm/
ds2.10/lib/domains/Praxis/attic/
ds2.10/lib/domains/Praxis/cemetery/mon/
ds2.10/lib/domains/Praxis/data/
ds2.10/lib/domains/Praxis/death/
ds2.10/lib/domains/Praxis/mountains/
ds2.10/lib/domains/Praxis/obj/armour/
ds2.10/lib/domains/Praxis/obj/magic/
ds2.10/lib/domains/Praxis/obj/weapon/
ds2.10/lib/domains/Praxis/orc_valley/
ds2.10/lib/domains/Ylsrim/
ds2.10/lib/domains/Ylsrim/adm/
ds2.10/lib/domains/Ylsrim/armor/
ds2.10/lib/domains/Ylsrim/broken/
ds2.10/lib/domains/Ylsrim/fish/
ds2.10/lib/domains/Ylsrim/meal/
ds2.10/lib/domains/Ylsrim/npc/
ds2.10/lib/domains/Ylsrim/obj/
ds2.10/lib/domains/Ylsrim/virtual/
ds2.10/lib/domains/Ylsrim/weapon/
ds2.10/lib/domains/alpha/room/
ds2.10/lib/domains/alpha/virtual/
ds2.10/lib/domains/campus/adm/
ds2.10/lib/domains/campus/etc/
ds2.10/lib/domains/campus/meals/
ds2.10/lib/domains/campus/txt/ai/charles/
ds2.10/lib/domains/campus/txt/ai/charles/bak2/
ds2.10/lib/domains/campus/txt/ai/charles/bak2/bak1/
ds2.10/lib/domains/campus/txt/ai/charly/
ds2.10/lib/domains/campus/txt/ai/charly/bak/
ds2.10/lib/domains/campus/txt/jenny/
ds2.10/lib/domains/cave/doors/
ds2.10/lib/domains/cave/etc/
ds2.10/lib/domains/cave/meals/
ds2.10/lib/domains/cave/weap/
ds2.10/lib/domains/default/chamber/
ds2.10/lib/domains/default/creator/
ds2.10/lib/domains/default/doors/
ds2.10/lib/domains/default/etc/
ds2.10/lib/domains/default/vehicle/
ds2.10/lib/domains/default/virtual/
ds2.10/lib/domains/town/save/
ds2.10/lib/domains/town/txt/shame/
ds2.10/lib/domains/town/virtual/
ds2.10/lib/domains/town/virtual/bottom/
ds2.10/lib/domains/town/virtual/space/
ds2.10/lib/estates/
ds2.10/lib/ftp/
ds2.10/lib/lib/comp/
ds2.10/lib/lib/daemons/
ds2.10/lib/lib/daemons/include/
ds2.10/lib/lib/lvs/
ds2.10/lib/lib/user/
ds2.10/lib/lib/virtual/
ds2.10/lib/log/
ds2.10/lib/log/adm/
ds2.10/lib/log/archive/
ds2.10/lib/log/chan/
ds2.10/lib/log/errors/
ds2.10/lib/log/law/adm/
ds2.10/lib/log/law/email/
ds2.10/lib/log/law/names/
ds2.10/lib/log/law/sites-misc/
ds2.10/lib/log/law/sites-register/
ds2.10/lib/log/law/sites-tempban/
ds2.10/lib/log/law/sites-watch/
ds2.10/lib/log/open/
ds2.10/lib/log/reports/
ds2.10/lib/log/router/
ds2.10/lib/log/secure/
ds2.10/lib/log/watch/
ds2.10/lib/obj/book_source/
ds2.10/lib/obj/include/
ds2.10/lib/powers/prayers/
ds2.10/lib/powers/spells/
ds2.10/lib/realms/template/
ds2.10/lib/realms/template/adm/
ds2.10/lib/realms/template/area/
ds2.10/lib/realms/template/area/armor/
ds2.10/lib/realms/template/area/npc/
ds2.10/lib/realms/template/area/obj/
ds2.10/lib/realms/template/area/room/
ds2.10/lib/realms/template/area/weap/
ds2.10/lib/realms/template/bak/
ds2.10/lib/realms/template/cmds/
ds2.10/lib/save/kills/o/
ds2.10/lib/secure/cfg/classes/
ds2.10/lib/secure/cmds/builders/
ds2.10/lib/secure/cmds/creators/include/
ds2.10/lib/secure/cmds/players/include/
ds2.10/lib/secure/daemon/imc2server/
ds2.10/lib/secure/daemon/include/
ds2.10/lib/secure/lib/
ds2.10/lib/secure/lib/include/
ds2.10/lib/secure/lib/net/include/
ds2.10/lib/secure/lib/std/
ds2.10/lib/secure/log/adm/
ds2.10/lib/secure/log/bak/
ds2.10/lib/secure/log/intermud/
ds2.10/lib/secure/log/network/
ds2.10/lib/secure/modules/
ds2.10/lib/secure/npc/
ds2.10/lib/secure/obj/include/
ds2.10/lib/secure/room/
ds2.10/lib/secure/save/
ds2.10/lib/secure/save/backup/
ds2.10/lib/secure/save/boards/
ds2.10/lib/secure/save/players/g/
ds2.10/lib/secure/tmp/
ds2.10/lib/secure/upgrades/files/
ds2.10/lib/secure/verbs/creators/
ds2.10/lib/std/board/
ds2.10/lib/std/lib/
ds2.10/lib/verbs/admins/include/
ds2.10/lib/verbs/builders/
ds2.10/lib/verbs/common/
ds2.10/lib/verbs/common/include/
ds2.10/lib/verbs/creators/
ds2.10/lib/verbs/creators/include/
ds2.10/lib/verbs/rooms/
ds2.10/lib/verbs/rooms/include/
ds2.10/lib/www/client/
ds2.10/lib/www/errors/
ds2.10/lib/www/images/
ds2.10/win32/
// This file written mostly by Tim Johnson (Tim@TimMUD)
void check_discs();

string *banned_muds = ({});
mapping bad_connects = ([]);

static void process_startup_req(int protocol, mixed info, int fd){
    // Handles startup stuff.
    //  Loads info into newinfo mapping.
    //  Error if mud is already connected.
    mapping newinfo;
    string site_ip;
    string *brethren = ({});
    // router name is info[4], I'll just pretend I'm every router though, ha-ha!
    // also, should verify that all the fields are the right type

    check_discs();
    server_log("%^BLUE%^process_startup_req: protocol="+protocol+", mud="+info[2]);

    if(member_array(info[2], banned_muds) != -1) {
        server_log("%^RED%^"+info[2]+" denied. reason: banned.");
        trr("---\n","blue");
        return;
    }

    if(!info[2] || !sizeof(info[2])) {
        server_log("%^RED%^"+info[2]+" denied. reason: null name");
        trr("---\n","blue");
        return;
    }

    foreach(mixed element in keys(mudinfo)){
        if(member_array(element,keys(connected_muds)) == -1) continue;
        if(mudinfo[element] && mudinfo[element]["ip"] == clean_fd(socket_address(fd)))
            brethren += ({ element });
    }
    trr("Number of muds with the same ip: "+sizeof(brethren));
    if(sizeof(brethren) > MAXMUDS && !mudinfo[info[2]]){
        write_data(fd,({ "error", 5, router_name, 0, info[2], 0,
                    "not-allowed", "Too many muds from your IP.", info }) );
        return;
    }
    if(member_array(fd,keys(this_object()->query_connected_fds())) != -1){
        string existing_mud = this_object()->query_connected_fds()[fd];
        trr("Socket status: "+identify(socket_status(fd)),"red");
        server_log("%^RED%^FD CONFLICT, NEW MUD: "+info[2]+", FD: "+fd);
        server_log("%^RED%^FD CONFLICT, OLD MUD: "+existing_mud+", FD: "+fd);
        trr("---\n","blue");
        if(mudinfo[existing_mud]){
            mudinfo[existing_mud]["disconnect_time"] = time();
        }
        if(!existing_mud || existing_mud != info[2]){
        map_delete(connected_muds, existing_mud);
        schedule_broadcast(existing_mud);
        }
    }

    if(sizeof(info)<18){ 
        // smallest protocol is protocol 1/2 which have size 18
        write_data(fd,({
                    "error",
                    5,
                    router_name,
                    0,
                    info[2],
                    0,
                    "bad-pkt",
                    "Bad startup_req packet",
                    info
                    }));
        return;
    }
    site_ip=socket_address(fd);
    site_ip = clean_fd(site_ip);
    newinfo = ([
            "name":info[2],
            "ip":site_ip,
            "connect_time":time(),
            "disconnect_time":0,
            "password":info[6],
            "old_mudlist_id":info[7],
            "old_chanlist_id":info[8],
            "player_port":info[9],
            "imud_tcp_port":info[10],
            "imud_udp_port":info[11],
            "mudlib":info[12],
            "base_mudlib":info[13],
            "driver":info[14],
            "mud_type":info[15],
            "open_status":info[16],
            "protocol":protocol,
            "restart_delay":-1,
            "router" : my_name,
            ]);

    if(protocol == 2 && sizeof(info) == 20 ) protocol = 3;

    switch(protocol){
        case 1:
        case 2:
            if(sizeof(info)!=18){
                server_log("error: wrong size packet. Got: "+sizeof(info)+", wanted 18");
                trr("---\n","blue");
                write_data(fd,({
                            "error",
                            5,
                            router_name,
                            0,
                            info[2],
                            0,
                            "bad-pkt",
                            "Bad startup_req packet",
                            info
                            }));
                return;
            }
            newinfo["services"]=info[17];
            newinfo["admin_email"]="Unknown"; // only in protocol 3
            newinfo["other_data"]=0; // only in protocol 3
            break;
        case 3:
            if(sizeof(info)!=20){
                server_log("error. wrong size packet. Got: "+sizeof(info)+", wanted 20");
                trr("---\n","blue");

                write_data(fd,({
                            "error",
                            5,
                            router_name,
                            0,
                            info[2],
                            0,
                            "bad-pkt",
                            "Bad startup_req packet",
                            info
                            }));
                return;
            }
            newinfo["admin_email"]=info[17];
            newinfo["services"]=info[18];
            newinfo["other_data"]=info[19];
            break;
        default:
            server_log("error. I dunno what.");

            write_data(fd,({
                        "error",
                        5,
                        router_name,
                        0,
                        info[2], // mud name
                        0,
                        "not-imp",
                        "unknown protocol: "+protocol,
                        info
                        }));
            return;
    }
    // Check valid values...
    if(
            !stringp(newinfo["open_status"]) ||
            !stringp(newinfo["admin_email"]) ||
            !stringp(newinfo["mud_type"]) ||
            !stringp(newinfo["driver"]) ||
            !stringp(newinfo["base_mudlib"]) ||
            !stringp(newinfo["mudlib"]) ||
            !mapp(newinfo["services"]) ||
            !intp(newinfo["imud_udp_port"]) ||
            !intp(newinfo["imud_tcp_port"]) ||
            !intp(newinfo["player_port"]) ||
            !intp(newinfo["password"]) ||
            !intp(newinfo["old_chanlist_id"]) ||
            !intp(newinfo["old_mudlist_id"]) ||
            (!mapp(newinfo["other_data"]) && newinfo["other_data"]!=0)
      ){
        write_data(fd,({
                    "error",
                    5,
                    router_name,
                    0,
                    info[2],
                    0,
                    "bad-pkt",
                    "Bad startup_req packet",
                    info
                    }));
        return;
    }
    if(!undefinedp(connected_muds[info[2]])){
        // if MUD is already connected
        trr("ROUTER_D: mud already connected on fd"+connected_muds[info[2]],"red");
        trr("ROUTER_D: Xref back: mud on fd"+connected_muds[info[2]]+" is "+
                "supposed to be "+this_object()->query_connected_fds()[this_object()->query_connected_muds()[info[2]]],"red");
        server_log("mud already connected");
        trr("---\n","blue");
        if(this_object()->query_mudinfo()[info[2]]["ip"] != explode(socket_address(fd)," ")[0] ||
                mudinfo[info[2]]["password"] != newinfo["password"]){
            write_data(fd,({
                        "error",
                        5,
                        router_name,
                        0,
                        info[2], // mud name
                        0,
                        "bad-proto", // doesn't seem to me like it should be bad-proto...
                        // see what the official one uses for this
                        // it might just boot the earlier MUD off?
                        "MUD already connected", // Error message
                        info
                        }));

            //This message to the already-connected mud serves 
            //to alert the legit mud that something may be wrong, and
            //also will trigger a closure of the socket if it turns out
            //to actually be in a zombie state.
            write_data(connected_muds[info[2]],({
                        "error",
                        5,
                        router_name,
                        0,
                        info[2], // mud name
                        0,
                        "bad-mojo", 
                        "Another mud is trying to be you.", 
                        ({ explode(socket_address(fd)," ")[0], info[12], info[13], info[17], info[9] })
                        }));
            return;
        }
        if(this_object()->query_mudinfo()[info[2]]["ip"] == explode(socket_address(fd)," ")[0] &&
                mudinfo[info[2]]["password"] == newinfo["password"] ){
            server_log("Since it's the same ip and password, I'll remove the current connection");
            if(mudinfo[info[2]]) mudinfo[info[2]]["disconnect_time"] = time();
            map_delete(connected_muds, info[2]);
            schedule_broadcast(info[2]);
        }
    }

    if(mudinfo[info[2]] && sizeof(mudinfo[info[2]]) && mudinfo[info[2]]["password"] != newinfo["password"]){
        // if MUD is already known, not connected, and wrong password
        if(newinfo["ip"]==mudinfo[info[2]]["ip"]){
            // same IP as last time and it isn't Dead Souls
            server_log("Wrong password, but right IP");
            trr("Expected "+mudinfo[info[2]]["password"]+
                    " but got "+newinfo["password"]);
            write_data(fd,({
                        "error",5,router_name,0,info[2],0,
                        "warning", // nothing in error summary that seems applicable?
                        // Change later if I find a better error code
                        "wrong password, but I'll allow you since it's the same IP as last time",0
                        }));
        }
        else{
            trr("wrong password, and from a new IP","red");
            trr("---\n","blue");
            if(!bad_connects[newinfo["ip"]]) bad_connects[newinfo["ip"]] = 1;
            else bad_connects[newinfo["ip"]] = bad_connects[newinfo["ip"]] + 1;
            if(bad_connects[newinfo["ip"]] > 100){
                trr("ROUTER_D: "+newinfo["ip"]+" blacklisted!","red");
                this_object()->AddBlacklistedMud(newinfo["ip"]);
                bad_connects[newinfo["ip"]] = 0;
            }
            server_log("%^RED%^WRONG PASSWORD, AND FROM A NEW IP");
            write_data(fd,({
                        "error",
                        5,
                        router_name,
                        0,
                        info[2], // mud name
                        0,
                        "not-allowed", // Change later!
                        "wrong password, and from a new IP", // Error message
                        info
                        }));
            return;
        }
    }
    if(!mudinfo[info[2]] || !newinfo["password"] || mudinfo[info[2]]["password"] != newinfo["password"] ){
        // if new MUD, assign it a password
        newinfo["password"]=random_numbers(9,1);
        trr("ROUTER_D: Assigning password "+newinfo["password"],"white");
        // Change this maybe... see if the password is supposed to be in a certain range
    }
    else {
        trr("ROUTER_D: password: Known: "+mudinfo[info[2]]["password"]+", current: "+newinfo["password"],"green");
    }
    // MUD should be okay at this point.
    mudinfo[info[2]]=newinfo; // update the mudinfo
    connected_muds[info[2]] = fd; // add this MUD to list of connected muds
    send_startup_reply(info[2]); // reply to MUD
    mudinfo_update_counter++;
    mudinfo_updates[info[2]]=mudinfo_update_counter;
    send_mudlist_updates(info[2], newinfo["old_mudlist_id"]);
    send_full_mudlist(info[2]);
    broadcast_mudlist(info[2]);
    broadcast_chanlist("foo",info[2]);
    if(bad_connects[newinfo["ip"]]) bad_connects[newinfo["ip"]] = 0;
    if(member_array("channel", keys(newinfo["services"])) != -1){

    }
    else {
        trr("It looks like "+info[2]+" doesn't have a channel service?!?","blue");
    }
    trr(timestamp()+" process_startup_req: for mud: "+info[2]+" complete.\n---\n","blue");
}