dw_fluffos_v2/
dw_fluffos_v2/fluffos-2.9-ds2.05/
dw_fluffos_v2/fluffos-2.9-ds2.05/ChangeLog.old/
dw_fluffos_v2/fluffos-2.9-ds2.05/Win32/
dw_fluffos_v2/fluffos-2.9-ds2.05/compat/
dw_fluffos_v2/fluffos-2.9-ds2.05/compat/simuls/
dw_fluffos_v2/fluffos-2.9-ds2.05/include/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/clone/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/command/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/data/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/etc/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/include/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/inherit/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/inherit/master/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/log/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/single/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/single/tests/compiler/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/single/tests/efuns/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/single/tests/operators/
dw_fluffos_v2/fluffos-2.9-ds2.05/testsuite/u/
dw_fluffos_v2/fluffos-2.9-ds2.05/tmp/
dw_fluffos_v2/fluffos-2.9-ds2.05/windows/
dw_fluffos_v2/lib/
dw_fluffos_v2/lib/binaries/cmds/
dw_fluffos_v2/lib/binaries/cmds/creator/
dw_fluffos_v2/lib/binaries/cmds/living/
dw_fluffos_v2/lib/binaries/cmds/player/
dw_fluffos_v2/lib/binaries/d/admin/obj/
dw_fluffos_v2/lib/binaries/d/liaison/
dw_fluffos_v2/lib/binaries/global/virtual/
dw_fluffos_v2/lib/binaries/global/virtual/setup_compiler/
dw_fluffos_v2/lib/binaries/obj/handlers/autodoc/
dw_fluffos_v2/lib/binaries/obj/handlers/terrain_things/
dw_fluffos_v2/lib/binaries/obj/misc/
dw_fluffos_v2/lib/binaries/obj/misc/buckets/
dw_fluffos_v2/lib/binaries/obj/monster/
dw_fluffos_v2/lib/binaries/obj/reactions/
dw_fluffos_v2/lib/binaries/obj/reagents/
dw_fluffos_v2/lib/binaries/secure/cmds/creator/
dw_fluffos_v2/lib/binaries/secure/master/
dw_fluffos_v2/lib/binaries/std/
dw_fluffos_v2/lib/binaries/std/dom/
dw_fluffos_v2/lib/binaries/std/effects/object/
dw_fluffos_v2/lib/binaries/std/guilds/
dw_fluffos_v2/lib/binaries/std/languages/
dw_fluffos_v2/lib/binaries/std/races/
dw_fluffos_v2/lib/binaries/std/room/
dw_fluffos_v2/lib/binaries/std/room/basic/
dw_fluffos_v2/lib/binaries/std/shops/
dw_fluffos_v2/lib/binaries/std/shops/inherit/
dw_fluffos_v2/lib/binaries/www/
dw_fluffos_v2/lib/cmds/guild-race/
dw_fluffos_v2/lib/cmds/guild-race/crafts/
dw_fluffos_v2/lib/cmds/guild-race/other/
dw_fluffos_v2/lib/cmds/playtester/
dw_fluffos_v2/lib/cmds/playtester/senior/
dw_fluffos_v2/lib/d/admin/
dw_fluffos_v2/lib/d/admin/log/
dw_fluffos_v2/lib/d/admin/mapper/31-10-01/mapmaker/event/
dw_fluffos_v2/lib/d/admin/meetings/
dw_fluffos_v2/lib/d/admin/obj/
dw_fluffos_v2/lib/d/admin/room/we_care/
dw_fluffos_v2/lib/d/admin/save/
dw_fluffos_v2/lib/d/dist/
dw_fluffos_v2/lib/d/dist/mtf/
dw_fluffos_v2/lib/d/dist/pumpkin/
dw_fluffos_v2/lib/d/dist/pumpkin/chars/
dw_fluffos_v2/lib/d/dist/pumpkin/desert/
dw_fluffos_v2/lib/d/dist/pumpkin/gumboot/
dw_fluffos_v2/lib/d/dist/pumpkin/hospital/
dw_fluffos_v2/lib/d/dist/pumpkin/inherit/
dw_fluffos_v2/lib/d/dist/pumpkin/map/
dw_fluffos_v2/lib/d/dist/pumpkin/plain/
dw_fluffos_v2/lib/d/dist/pumpkin/pumpkin/
dw_fluffos_v2/lib/d/dist/pumpkin/save/
dw_fluffos_v2/lib/d/dist/pumpkin/squash/
dw_fluffos_v2/lib/d/dist/pumpkin/terrain/
dw_fluffos_v2/lib/d/dist/pumpkin/woods/
dw_fluffos_v2/lib/d/dist/start/
dw_fluffos_v2/lib/d/learning/TinyTown/buildings/
dw_fluffos_v2/lib/d/learning/TinyTown/map/
dw_fluffos_v2/lib/d/learning/TinyTown/roads/
dw_fluffos_v2/lib/d/learning/add_command/
dw_fluffos_v2/lib/d/learning/arms_and_weps/
dw_fluffos_v2/lib/d/learning/chars/
dw_fluffos_v2/lib/d/learning/cutnpaste/
dw_fluffos_v2/lib/d/learning/examples/npcs/
dw_fluffos_v2/lib/d/learning/examples/player_houses/npcs/
dw_fluffos_v2/lib/d/learning/examples/terrain_map/basic/
dw_fluffos_v2/lib/d/learning/functions/
dw_fluffos_v2/lib/d/learning/handlers/
dw_fluffos_v2/lib/d/learning/help_topics/npcs/
dw_fluffos_v2/lib/d/learning/help_topics/objects/
dw_fluffos_v2/lib/d/learning/help_topics/rcs_demo/
dw_fluffos_v2/lib/d/learning/help_topics/rooms/
dw_fluffos_v2/lib/d/learning/help_topics/rooms/crowd/
dw_fluffos_v2/lib/d/learning/help_topics/rooms/situations/
dw_fluffos_v2/lib/d/learning/items/
dw_fluffos_v2/lib/d/learning/save/
dw_fluffos_v2/lib/d/liaison/
dw_fluffos_v2/lib/d/liaison/NEWBIE/doc/
dw_fluffos_v2/lib/d/liaison/NEWBIE/save/oldlog/
dw_fluffos_v2/lib/db/
dw_fluffos_v2/lib/doc/
dw_fluffos_v2/lib/doc/creator/
dw_fluffos_v2/lib/doc/creator/autodoc/include/reaction/
dw_fluffos_v2/lib/doc/creator/autodoc/include/ritual_system/
dw_fluffos_v2/lib/doc/creator/autodoc/include/talker/
dw_fluffos_v2/lib/doc/creator/autodoc/include/terrain_map/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/baggage/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/clock/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/clothing/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/cont_save/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/corpse/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/money/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/monster/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/scabbard/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/service_provider/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/state_changer/
dw_fluffos_v2/lib/doc/creator/autodoc/obj/wand/
dw_fluffos_v2/lib/doc/creator/autodoc/std/book_dir/
dw_fluffos_v2/lib/doc/creator/autodoc/std/key/
dw_fluffos_v2/lib/doc/creator/autodoc/std/learning/
dw_fluffos_v2/lib/doc/creator/autodoc/std/map/
dw_fluffos_v2/lib/doc/creator/autodoc/std/race/
dw_fluffos_v2/lib/doc/creator/autodoc/std/weapon_logic/
dw_fluffos_v2/lib/doc/creator/files/
dw_fluffos_v2/lib/doc/creator/policy/
dw_fluffos_v2/lib/doc/creator/room/
dw_fluffos_v2/lib/doc/effects/
dw_fluffos_v2/lib/doc/ideas/
dw_fluffos_v2/lib/doc/known_command/
dw_fluffos_v2/lib/doc/lpc/basic_manual/
dw_fluffos_v2/lib/doc/lpc/intermediate/
dw_fluffos_v2/lib/doc/new/add_command/
dw_fluffos_v2/lib/doc/new/handlers/
dw_fluffos_v2/lib/doc/new/living/
dw_fluffos_v2/lib/doc/new/living/race/
dw_fluffos_v2/lib/doc/new/living/spells/
dw_fluffos_v2/lib/doc/new/player/
dw_fluffos_v2/lib/doc/new/room/guild/
dw_fluffos_v2/lib/doc/new/room/outside/
dw_fluffos_v2/lib/doc/new/room/storeroom/
dw_fluffos_v2/lib/doc/object/
dw_fluffos_v2/lib/doc/playtesters/
dw_fluffos_v2/lib/doc/policy/
dw_fluffos_v2/lib/doc/weapons/
dw_fluffos_v2/lib/global/handlers/
dw_fluffos_v2/lib/global/virtual/setup_compiler/
dw_fluffos_v2/lib/include/
dw_fluffos_v2/lib/include/cmds/
dw_fluffos_v2/lib/include/effects/
dw_fluffos_v2/lib/include/npc/
dw_fluffos_v2/lib/include/shops/
dw_fluffos_v2/lib/net/daemon/chars/
dw_fluffos_v2/lib/net/inherit/
dw_fluffos_v2/lib/net/intermud3/
dw_fluffos_v2/lib/net/intermud3/services/
dw_fluffos_v2/lib/net/obj/
dw_fluffos_v2/lib/net/save/
dw_fluffos_v2/lib/net/smnmp/
dw_fluffos_v2/lib/net/snmp/
dw_fluffos_v2/lib/obj/amulets/
dw_fluffos_v2/lib/obj/b_day/
dw_fluffos_v2/lib/obj/examples/
dw_fluffos_v2/lib/obj/food/alcohol/
dw_fluffos_v2/lib/obj/food/chocolates/
dw_fluffos_v2/lib/obj/food/fruits/
dw_fluffos_v2/lib/obj/food/meat/
dw_fluffos_v2/lib/obj/food/nuts/
dw_fluffos_v2/lib/obj/food/seafood/
dw_fluffos_v2/lib/obj/food/vegetables/
dw_fluffos_v2/lib/obj/fungi/
dw_fluffos_v2/lib/obj/furnitures/artwork/
dw_fluffos_v2/lib/obj/furnitures/bathroom/
dw_fluffos_v2/lib/obj/furnitures/beds/
dw_fluffos_v2/lib/obj/furnitures/cabinets/
dw_fluffos_v2/lib/obj/furnitures/chairs/
dw_fluffos_v2/lib/obj/furnitures/chests/
dw_fluffos_v2/lib/obj/furnitures/clocks/
dw_fluffos_v2/lib/obj/furnitures/crockery/
dw_fluffos_v2/lib/obj/furnitures/cupboards/
dw_fluffos_v2/lib/obj/furnitures/cushions/
dw_fluffos_v2/lib/obj/furnitures/fake_plants/
dw_fluffos_v2/lib/obj/furnitures/lamps/
dw_fluffos_v2/lib/obj/furnitures/mirrors/
dw_fluffos_v2/lib/obj/furnitures/outdoor/
dw_fluffos_v2/lib/obj/furnitures/safes/
dw_fluffos_v2/lib/obj/furnitures/shelves/
dw_fluffos_v2/lib/obj/furnitures/sideboards/
dw_fluffos_v2/lib/obj/furnitures/sofas/
dw_fluffos_v2/lib/obj/furnitures/stoves/
dw_fluffos_v2/lib/obj/furnitures/tables/
dw_fluffos_v2/lib/obj/furnitures/wardrobes/
dw_fluffos_v2/lib/obj/handlers/
dw_fluffos_v2/lib/obj/handlers/autodoc/
dw_fluffos_v2/lib/obj/jewellery/anklets/
dw_fluffos_v2/lib/obj/jewellery/bracelets/
dw_fluffos_v2/lib/obj/jewellery/earrings/
dw_fluffos_v2/lib/obj/jewellery/misc/
dw_fluffos_v2/lib/obj/jewellery/necklaces/
dw_fluffos_v2/lib/obj/jewellery/rings/
dw_fluffos_v2/lib/obj/media/
dw_fluffos_v2/lib/obj/misc/buckets/
dw_fluffos_v2/lib/obj/misc/jars/
dw_fluffos_v2/lib/obj/misc/papers/
dw_fluffos_v2/lib/obj/misc/player_shop/
dw_fluffos_v2/lib/obj/misc/shops/
dw_fluffos_v2/lib/obj/misc/traps/
dw_fluffos_v2/lib/obj/monster/
dw_fluffos_v2/lib/obj/monster/godmother/
dw_fluffos_v2/lib/obj/monster/transport/
dw_fluffos_v2/lib/obj/plants/inherit/
dw_fluffos_v2/lib/obj/potions/
dw_fluffos_v2/lib/open/boards/
dw_fluffos_v2/lib/save/autodoc/
dw_fluffos_v2/lib/save/bank_accounts/
dw_fluffos_v2/lib/save/boards/frog/
dw_fluffos_v2/lib/save/books/bed_catalog/
dw_fluffos_v2/lib/save/creators/
dw_fluffos_v2/lib/save/mail/
dw_fluffos_v2/lib/save/mail/p/
dw_fluffos_v2/lib/save/soul/data/
dw_fluffos_v2/lib/save/tasks/
dw_fluffos_v2/lib/save/vaults/
dw_fluffos_v2/lib/secure/cmds/lord/
dw_fluffos_v2/lib/secure/config/
dw_fluffos_v2/lib/secure/items/
dw_fluffos_v2/lib/secure/player/
dw_fluffos_v2/lib/soul/
dw_fluffos_v2/lib/soul/i/
dw_fluffos_v2/lib/soul/j/
dw_fluffos_v2/lib/soul/k/
dw_fluffos_v2/lib/soul/o/
dw_fluffos_v2/lib/soul/q/
dw_fluffos_v2/lib/soul/to_approve/
dw_fluffos_v2/lib/soul/u/
dw_fluffos_v2/lib/soul/v/
dw_fluffos_v2/lib/soul/wish_list/
dw_fluffos_v2/lib/soul/y/
dw_fluffos_v2/lib/soul/z/
dw_fluffos_v2/lib/std/creator/
dw_fluffos_v2/lib/std/effects/
dw_fluffos_v2/lib/std/effects/attached/
dw_fluffos_v2/lib/std/effects/external/
dw_fluffos_v2/lib/std/effects/fighting/
dw_fluffos_v2/lib/std/effects/other/
dw_fluffos_v2/lib/std/environ/
dw_fluffos_v2/lib/std/guilds/
dw_fluffos_v2/lib/std/hospital/
dw_fluffos_v2/lib/std/house/
dw_fluffos_v2/lib/std/house/onebedhouse/
dw_fluffos_v2/lib/std/house/onebedhut/
dw_fluffos_v2/lib/std/house/tworoomflat/
dw_fluffos_v2/lib/std/languages/
dw_fluffos_v2/lib/std/liquids/
dw_fluffos_v2/lib/std/nationality/
dw_fluffos_v2/lib/std/nationality/accents/
dw_fluffos_v2/lib/std/nationality/accents/national/
dw_fluffos_v2/lib/std/nationality/accents/regional/
dw_fluffos_v2/lib/std/npc/goals/
dw_fluffos_v2/lib/std/npc/goals/basic/
dw_fluffos_v2/lib/std/npc/goals/misc/
dw_fluffos_v2/lib/std/npc/inherit/
dw_fluffos_v2/lib/std/npc/plans/
dw_fluffos_v2/lib/std/npc/plans/basic/
dw_fluffos_v2/lib/std/outsides/
dw_fluffos_v2/lib/std/races/shadows/
dw_fluffos_v2/lib/std/room/basic/topography/
dw_fluffos_v2/lib/std/room/controller/
dw_fluffos_v2/lib/std/room/controller/topography/
dw_fluffos_v2/lib/std/room/furniture/games/
dw_fluffos_v2/lib/std/room/furniture/inherit/
dw_fluffos_v2/lib/std/room/inherit/carriage/
dw_fluffos_v2/lib/std/room/inherit/topography/
dw_fluffos_v2/lib/std/room/punishments/
dw_fluffos_v2/lib/std/room/topography/area/
dw_fluffos_v2/lib/std/room/topography/iroom/
dw_fluffos_v2/lib/std/room/topography/milestone/
dw_fluffos_v2/lib/std/shadows/
dw_fluffos_v2/lib/std/shadows/attached/
dw_fluffos_v2/lib/std/shadows/curses/
dw_fluffos_v2/lib/std/shadows/disease/
dw_fluffos_v2/lib/std/shadows/fighting/
dw_fluffos_v2/lib/std/shadows/room/
dw_fluffos_v2/lib/std/shops/controllers/
dw_fluffos_v2/lib/std/shops/objs/
dw_fluffos_v2/lib/std/shops/player_shop/
dw_fluffos_v2/lib/std/shops/player_shop/office_code/
dw_fluffos_v2/lib/std/socket/
dw_fluffos_v2/lib/www/
dw_fluffos_v2/lib/www/external/autodoc/
dw_fluffos_v2/lib/www/external/java/telnet/Documentation/
dw_fluffos_v2/lib/www/external/java/telnet/Documentation/images/
dw_fluffos_v2/lib/www/external/java/telnet/examples/
dw_fluffos_v2/lib/www/external/java/telnet/tools/
dw_fluffos_v2/lib/www/pics/
dw_fluffos_v2/lib/www/secure/creator/
dw_fluffos_v2/lib/www/secure/editors/
dw_fluffos_v2/lib/www/secure/survey_results/
dw_fluffos_v2/win32/
#include <network.h>
#include <mail.h>
#include <pop3.h>
#include <mime.h>
#include <player_handler.h>

inherit SERVER;

private void parse_comm(class pop_session sess, string str);

private string *pop_states = ({
  "AUTHORIZATION",
  "AUTHORIZATION",
  "TRANSACTION",
  "UPDATE"
});
private mapping Sockets;
private nosave mapping cache;
private nosave int reads, hits;
private mapping last;

protected void create() {
  server::create();
  SetSocketType(STREAM);
  SetDestructOnClose(1);
  Sockets = ([]);
  call_out("setup", 2);
  call_out("clean_sockets", 180);
  unguarded((: restore_object, "/net/save/pop3" :));
  if(!last)
    last = ([ ]);
} /* create() */

/** @ignore yes */
protected void setup() {
  if (eventCreateSocket(PORT_POP3) < 0) {
    if (this_object()) {
      destruct(this_object());
    }
  }
} /* setup() */

/** @ignore yes */
protected void eventNewConnection(int fd) {
  class pop_session sess;
  
  server::eventNewConnection(fd);
  sess = new(class pop_session, fd : fd, state : POP_AUTH_USER, command : "",
             deleted : ({}), messages : ({}));
  Sockets[fd] = sess;
  eventWrite(fd, sprintf("+OK POP3 %s Discworld v%s server ready\r\n",
                         query_host_name(), POP3_VERSION));
} /* eventNewConnection() */

/** @ignore yes */
protected void eventRead(int fd, string str) {
  string *bits, bit;
  class pop_session sess = Sockets[fd];

  if (!sess) {
    return;
  }
  if (str) {
    sess->command += str;
    if (strsrch(str, '\n') == -1) {
      return;
    }
  }
  sess->command = replace_string(sess->command, "\r", "");
  bits = explode(sess->command, "\n");
  if (!sizeof(bits)) {
    eventWrite(fd, "-ERR Null command\r\n");
  } else {
    foreach (bit in bits) {
      parse_comm(sess, bit);
    }
  }
  sess->command = "";
} /* eventRead() */

/** @ignore yes */
protected void eventSocketClosed(int fd) {
  class pop_session sess = Sockets[fd];

  if (!sess) {
    return;
  }
  map_delete(Sockets, fd);
} /* eventSocketClosed() */

private void sign_off(class pop_session sess) {
  int *deleted;
  
  if (sizeof(sess->deleted)) {
    deleted = map(sess->deleted, (: $2->headers[$1-1]->number :), sess);
    FOLDER_H->delete_it(sess->user_name, "inbox", deleted);
  }
  eventWrite(sess->fd, "+OK Sayonara\r\n", 1);
  unguarded((: save_object, "/net/save/pop3" :));
} /* sign_off() */

private void load_message(class pop_session sess, int message) {
  string mess;
  int which;

  reads++;

  which = sess->headers[message]->number;

  if(!cache)
    cache = ([ ]);
  if(cache[which]) {
    hits++;
    sess->messages[message] = cache[which];
    return;
  }
  
  mess = FOLDER_H->load_message(sess->user_name, "inbox",
                                sess->headers[message]->number);
  mess = MIME->rewrite_header(mess);
  mess = replace_string(mess, "\n", "\r\n");

  if (mess[<1] != '\n') {
    mess += "\r\n";
  }
  sess->messages[message] = mess;
  sess->sizes[message] = strlen(mess);
  
  cache[which] = copy(mess);
} /* load_message() */

private void load_folder(class pop_session sess) {
  int *tmp, max, i;
  
  sess->headers = FOLDER_H->get_messages(sess->user_name, "inbox");
  tell_creator("ceres", "Headers: %d\n", sizeof(sess->headers));
  sess->messages = allocate(sizeof(sess->headers));
  sess->sizes = allocate(sizeof(sess->headers));
  sess->num_messages = sizeof(sess->headers);

  // the sizes will start out as an approximation. As we read actual messages
  // the real size will be put in place.

  tmp = sess->sizes;
  
  for(i=0; i<sizeof(sess->headers); i++)
    sess->sizes[i] = 1000;

  // If we'd already got some sizes then copy them into the new size array.
  if(tmp) {
    (sizeof(tmp) > sizeof(sess->headers)) ? max = sizeof(tmp) :
      max = sizeof(sess->headers);
    for(i=0; i<max; i++)
      sess->sizes[i] = tmp[i];
  }
} /* load_folder() */

private void parse_comm(class pop_session sess, string str) {
  string *bits, rest, cmd, message, header;
  int *sizes, lines, i, number, fd = sess->fd;
  
  if (strsrch(lower_case(str), "pass") == -1) {
    TP("Parsing " + str + ".\n");
  }
  bits = explode(str, " ");
  cmd = bits[0];
  if (sizeof(bits) > 1 ) {
    rest = implode(bits[1..], " ");
  } else {
    rest = "";
  }
  sess->time = time();
  switch(lower_case(cmd)) {
  case "dele":
    tell_creator("ceres", "Requesting deletion of %d\n", number);

    CHECK_STATE(POP_TRANSACTION);
    CHECK_CMD(1, "-ERR Missing message number argument\r\n");
    if ((sscanf(bits[1], "%d", number) != 1) || number < 1 ||
        number > sess->num_messages) {
      eventWrite(fd, "-ERR No such message\r\n");
    } else if (member_array(number, sess->deleted) != -1) {
      eventWrite(fd, sprintf("-ERR Message number %d already deleted\r\n",
                             number));
    } else {
      tell_creator("ceres", "Deleting %d\n", number);
      sess->deleted += ({ number });
      eventWrite(fd, "+OK Message deleted\r\n");
    }
    break;
  case "last":
    CHECK_STATE(POP_TRANSACTION);
    eventWrite(fd, sprintf("+OK %d\r\n", last[sess->user_name]));
    tell_creator("ceres", "Returning %d for last\n", last[sess->user_name]);
    break;
  case "list":
    CHECK_STATE(POP_TRANSACTION);
    if (sizeof(bits) > 1) {
      if ((sscanf(bits[1], "%d", number) != 1) || number < 1 ||
          number > sess->num_messages) {
        eventWrite(fd, "-ERR No such message\r\n");
      } else if (member_array(number, sess->deleted) != -1) {
        eventWrite(fd, sprintf("-ERR Message number %d already deleted\r\n",
                               number));
      } else {
        if (!sizeof(sess->headers)) {
          load_folder(sess);
        }
        eventWrite(fd, sprintf("+OK %d %d\r\n", number,
                               sess->sizes[number-1]));
      }
    } else {
      eventWrite(fd, "+OK Mailbox scan listing follows\r\n");
      if (!sizeof(sess->headers)) {
        load_folder(sess);
      }
      i = 0;
      if (!sizeof(sess->deleted)) {
        map(sess->sizes,
            function(int size, int ref idx, int fd) {
              reset_eval_cost();
              eventWrite(fd, sprintf("%d %d\r\n", ++idx, size));
            }, ref i, fd);
      } else {
        map(sess->sizes,
            function (int size, int ref idx, int fd, int *deleted) {
              reset_eval_cost();
              idx++;
              if (member_array(idx, deleted) == -1) {
                eventWrite(fd, sprintf("%d %d\r\n", idx, size));
              }
            }, ref i, fd, sess->deleted);
      }
      eventWrite(fd, ".\r\n");
    }
    break;
  case "noop":
    CHECK_STATE(POP_TRANSACTION);
    eventWrite(fd, "+OK No-op to you too!\r\n");
    break;
  case "pass":
    CHECK_STATE(POP_AUTH_PASS);
    CHECK_CMD(1, "-ERR Missing password argument\r\n");
    if (!PLAYER_HANDLER->test_password(sess->user_name, rest)) {
      sess->state = POP_AUTH_USER;
      eventWrite(fd, "-ERR Bad login\r\n");
    } else {
      sess->state = POP_TRANSACTION;
      load_folder(sess);
      //      sess->headers = FOLDER_H->get_messages(sess->user_name, "inbox");
      //      sess->num_messages = sizeof(sess->headers);
      eventWrite(fd, sprintf("+OK Mailbox open, %d messages\r\n",
                             sess->num_messages));
    }
    break;
  case "quit":
    switch (sess->state) {
    case POP_AUTH_USER:
    case POP_AUTH_PASS:
      eventWrite(fd, "+OK Sayonara\r\n", 1);
      break;
    case POP_TRANSACTION:
      sess->state = POP_UPDATE;
      sign_off(sess);
      break;
    }
    break;
  case "retr":
    CHECK_STATE(POP_TRANSACTION);
    CHECK_CMD(1, "-ERR Missing message number argument\r\n");
    if ((sscanf(bits[1], "%d", number) != 1) || number < 1 ||
        number > sess->num_messages) {
      eventWrite(fd, "-ERR No such message\r\n");
    } else if (member_array(number, sess->deleted) != -1) {
      eventWrite(fd, sprintf("-ERR Message number %d already deleted\r\n",
                             number));
    } else {
      if (!sizeof(sess->headers)) {
        load_folder(sess);
      }
      last[sess->user_name] = number;
      if(!sess->messages[number-1])
        load_message(sess, number-1);
      
      message = replace_string(sess->messages[number-1], "\n.", "\n..");
      eventWrite(fd, sprintf("+OK %d octets\r\n", sess->sizes[number-1]));
      eventWrite(fd, message + ".\r\n");
    }
    break;
  case "rset":
    CHECK_STATE(POP_TRANSACTION);
    sess->deleted = ({});
    last[sess->user_name] = 0;
    eventWrite(fd, "+OK Reset state\r\n");
    break;
  case "stat":
    CHECK_STATE(POP_TRANSACTION);
    if (!sizeof(sess->headers)) {
      load_folder(sess);
    }
    if (!sizeof(sess->deleted)) {
      tell_creator("ceres", "headers: %d, sizes %d\n", sizeof(sess->headers),
                   sizeof(sess->sizes));
      eventWrite(fd, sprintf("+OK %d %d\r\n", sess->num_messages,
                             implode(sess->sizes, (: $1 + $2 :))));
    } else {
      i = 0;
      sizes = map(sess->sizes, function(int size, int ref idx, int *deleted) {
                                 reset_eval_cost();
                                 idx++;
                                 if (member_array(idx, deleted) != -1) {
                                   return 0;
                                 } else {
                                   return size;
                                 }
                               }, ref i, sess->deleted);
      eventWrite(fd, sprintf("+OK %d %d\r\n",
                             (sess->num_messages - sizeof(sess->deleted)),
                             implode(sizes,
                                     (: reset_eval_cost(), $1 + $2 :))));
    }
    break;
  case "top":
    CHECK_STATE(POP_TRANSACTION);
    CHECK_CMD(1, "-ERR Missing message number argument\r\n");
    if ((sscanf(bits[1], "%d", number) != 1) || number < 1 ||
        number > sess->num_messages) {
      eventWrite(fd, "-ERR No such message\r\n");
    } else if (member_array(number, sess->deleted) != -1) {
      eventWrite(fd, sprintf("-ERR Message number %d already deleted\r\n",
                             number));
    } else {
      if (!sizeof(sess->headers)) {
        load_folder(sess);
      }

      last[sess->user_name] = number;
      if(!sess->messages[number-1])
        load_message(sess, number-1);

      if (sizeof(bits) > 2 && (sscanf(bits[2], "%d", lines) == 1) &&
          lines < sess->sizes[number-1]) {
        i = strsrch(sess->messages[number-1], "\r\n\r\n");
        header = sess->messages[number-1][0..i+3];
        message = replace_string(sess->messages[number-1][i+4..], "\n.",
                                 "\n..");
        eventWrite(fd, sprintf("+OK %d octets\r\n", sess->sizes[number-1]));
        eventWrite(fd, header);
        if (lines) {
          bits = explode(message, "\r\n");
          eventWrite(fd, implode(bits[0..lines-1], "\r\n") + "\r\n.\r\n");
        } else {
          eventWrite(fd, ".\r\n");
        }
      } else {
        message = replace_string(sess->messages[number-1], "\n.", "\n..");
        eventWrite(fd, sprintf("+OK %d octets\r\n", sess->sizes[number-1]));
        eventWrite(fd, message + ".\r\n");
      }
    }   
    break;
  case "uidl":
    CHECK_STATE(POP_TRANSACTION);
    if (sizeof(bits) > 1) {
      if ((sscanf(bits[1], "%d", number) != 1) || number < 1 ||
          number > sess->num_messages) {
        eventWrite(fd, "-ERR No such message\r\n");
      } else if (member_array(number, sess->deleted) != -1) {
        eventWrite(fd, sprintf("-ERR Message number %d already deleted\r\n",
                               number));
      } else {
        if (!sizeof(sess->headers)) {
          load_folder(sess);
        }
        eventWrite(fd, sprintf("+OK %d %d\r\n", number,
                               sess->headers[number-1]->number));
      }
    } else {
      eventWrite(fd, "+OK Unique-ID listing follows\r\n");
      if (!sizeof(sess->headers)) {
        load_folder(sess);
      }
      i = 0;
      if (!sizeof(sess->deleted)) {
        map(sess->headers,
            function(class mail_header hdr, int ref idx, int fd) {
              reset_eval_cost();
              eventWrite(fd, sprintf("%d %d\r\n", ++idx, hdr->number));
            }, ref i, fd);
      } else {
        map(sess->headers,
            function (class mail_header hdr, int ref idx, int fd, int *deleted) {
              reset_eval_cost();
              idx++;
              if (member_array(idx, deleted) == -1) {
                eventWrite(fd, sprintf("%d %d\r\n", idx, hdr->number));
              }
            }, ref i, fd, sess->deleted);
      }
      eventWrite(fd, ".\r\n");
    }
    break;
  case "user":
    CHECK_STATE(POP_AUTH_USER);
    CHECK_CMD(1, "-ERR Missing username argument\r\n");
    sess->user_name = bits[1];
    sess->state = POP_AUTH_PASS;
    eventWrite(fd, "+OK User name accepted, password please\r\n");
    break;
  case "":
    eventWrite(fd, "-ERR Null command\r\n");
    break;
  default:
    eventWrite(fd, sprintf("-ERR Unknown command in %s state\r\n",
                           pop_states[sess->state]));
    break;
  }
} /* parse_comm() */

/** @ignore yes */
protected void close_connection(class pop_session sess) {
  if (!sess) {
    return;
  }
  eventWrite(sess->fd, "-ERR Autologout; idle for too long\r\n", 1);
} /* close_connection() */

/** @ignore yes */
protected void clean_sockets() {
  class pop_session sess;
       
  foreach (sess in values(Sockets)) {
    if (!sess->time) {
      sess->time = time();
      continue;
    }
    if (time() - sess->time > 1800)
      close_connection(sess);
  }
  call_out("clean_sockets", 180);
} /* clean_sockets() */

mixed *stats() {
  if(!cache)
    cache = ([ ]);
  return  ({
    ({ "reads", reads, }),
      ({ "cache hit percent", reads != 0 ? (hits * 100) / reads : 0, }),
        ({ "messages in cache", sizeof(keys(cache)), }), 
          });
}