ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/ext/commands.ext
Revision: 1.50
Committed: Sun Jul 1 03:17:39 2007 UTC (16 years, 10 months ago) by root
Branch: MAIN
Changes since 1.49: +1 -1 lines
Log Message:
*** empty log message ***

File Contents

# User Rev Content
1 root 1.47 #! perl # mandatory depends=irc
2 root 1.1
3     use POSIX ();
4    
5     # miscellaneous commands
6    
7     sub rename_to($$$) {
8     my ($ob, $from, $to) = @_;
9    
10 root 1.12 $to =~ /^[a-zA-Z0-9.,=#\/%$!^ ]*$/
11 root 1.1 or return $ob->message ("rename: name must consist only of letters, digits, spaces and a few other things.");
12    
13     127 >= length $to
14     or return $ob->message ("rename: new name must be <= 127 characters.");
15    
16     my $item;
17    
18     if (length $from) {
19     $item = $ob->find_best_object_match ($from)
20     or return $ob->message ("rename: could not find a matching item to rename.");
21     } else {
22 root 1.31 $item = $ob->find_marked_object
23 root 1.1 or return $ob->message ("rename: no from name and no marked item found to rename.");
24     }
25    
26     $item->custom_name (length $to ? $to : undef);
27    
28     if (length $to) {
29     $item->custom_name ($to);
30     $ob->message ("Your " . $item->base_name . " will now be called $to.");
31     } else {
32     $item->custom_name (undef);
33     $ob->message ("You stop calling your " . $item->base_name . " with weird names.");
34     }
35    
36     $ob->esrv_update_item (cf::UPD_NAME, $item);
37    
38     1
39     }
40    
41 root 1.42 sub who_listing(;$$) {
42     my ($privileged, $select) = @_;
43 root 1.1
44     my ($numwiz, $numafk) = (0, 0);
45     my @pl;
46    
47     foreach my $pl (cf::player::list) {
48 root 1.5 my $ns = $pl->ns or next;
49 root 1.1 my $ob = $pl->ob;
50    
51     next unless $ob->map
52     && ($privileged || !$pl->hidden);
53    
54     $numwiz++ if $ob->flag (cf::FLAG_WIZ);
55 root 1.5 $numafk++ if $ns->afk;
56 root 1.1
57     push @pl, $pl;
58     }
59    
60     (
61     "Total Players in The World. (" . (scalar @pl) . ") -- WIZ($numwiz) AFK($numafk) BOT(0)",
62 root 1.42 (grep /$select/,
63     map {
64     my ($pl, $ob, $ns) = ($_, $_->ob, $_->ns);
65    
66     "* " . $ob->name . "/" . $ob->level . " " . (length $pl->own_title ? $pl->own_title : "the " . $pl->title)
67 root 1.48 . ($pl->gender ? " [f]" : " [m]")
68 root 1.42 . ($pl->peaceful ? " [peaceful]" : " [HOSTILE]")
69     . ($ns->afk ? " [AFK]" : "")
70     . ($ob->flag (cf::FLAG_WIZ) ? " [WIZ]" : "")
71     . " [" . $pl->ns->version . "]"
72     . " [" . ($pl->peaceful || $privileged ? $ob->map->visible_name : $ob->region->name) . "]"
73     . (sprintf " [rtt %.3fs]", $pl->ns->rtt * 1e-6)
74     . ($privileged ? " " . $pl->ns->host : "")
75     } sort { (lc $a->ob->name) cmp (lc $b->ob->name) } @pl
76 root 1.1 ),
77 root 1.45 eval { "* IRC: " . join ", ", ext::irc::users },
78 root 1.1 )
79     }
80    
81     cf::register_command who => sub {
82     my ($ob, $arg) = @_;
83    
84 root 1.42 $ob->speed_left ($ob->speed_left - 4);
85 root 1.1
86 root 1.42 $ob->reply (undef, (join "\n", who_listing $ob->may ("extended_who"), $arg), cf::NDI_UNIQUE | cf::NDI_DK_ORANGE);
87 root 1.1
88     1
89     };
90    
91 root 1.40 cf::register_command seen => sub {
92     my ($pl, $args) = @_;
93    
94     if (my ($login) = $args =~ /(\S+)/) {
95     if ($login eq $pl->name) {
96     $pl->message ("Very funny, $login. Ha. Ha.", cf::NDI_UNIQUE);
97     } elsif (cf::player::find_active $login) {
98     $pl->message ("$login is right here on this server!", cf::NDI_UNIQUE);
99     } elsif (cf::player::exists $login
100     and stat cf::player::path $login) {
101     my $time = (stat _)[9];
102    
103     $pl->message ("$login was last seen here "
104     . (POSIX::strftime "%Y-%m-%d %H:%M:%S +0000", gmtime $time)
105     . " which was " . (int +(time - $time) / 3600) . " hours ago.", cf::NDI_UNIQUE);
106     } else {
107     $pl->message ("No player named $login is known to me.", cf::NDI_UNIQUE);
108     }
109     } else {
110     $pl->message ("Usage: seen <player>", cf::NDI_UNIQUE);
111     }
112     };
113    
114 pippijn 1.27 cf::register_command body => sub {
115     my ($ob) = @_;
116    
117     # Too hard to try and make a header that lines everything up, so just
118     # give a description. (comment from C++)
119 root 1.44 my $reply =
120     "The first column is the name of the body location.\n"
121     . "The second column is how many of those locations your body has.\n"
122     . "The third column is how many slots in that location are available.\n";
123 pippijn 1.27
124 root 1.44 $reply .= sprintf "%-20s %3s %5s\n", "Location", "You", "Avail";
125 root 1.43 for (0 .. cf::NUM_BODY_LOCATIONS - 1) {
126 root 1.44 my $msg = cf::object::slot_nonuse_name $_;
127     $msg =~ s/^.*? a //;
128     $reply .= sprintf "%-20s %3d %5d\n", $msg, $ob->slot_info ($_), $ob->slot_used ($_)
129 root 1.43 if $ob->slot_info ($_) or $ob->slot_used ($_);
130 pippijn 1.27 }
131    
132 root 1.44 $reply .= "You are not allowed to wear armor\n"
133 pippijn 1.27 unless $ob->flag (cf::FLAG_USE_ARMOUR);
134 root 1.44 $reply .= "You are not allowed to use weapons\n"
135 pippijn 1.27 unless $ob->flag (cf::FLAG_USE_WEAPON);
136    
137 root 1.44 $ob->reply (undef, $reply);
138    
139 pippijn 1.27 1
140     };
141    
142 pippijn 1.28 cf::register_command mark => sub {
143     my ($pl, $arg) = @_;
144    
145 pippijn 1.32 if (length $arg) {
146 pippijn 1.28 my $ob = $pl->find_best_object_match ($arg);
147    
148     return $pl->reply (undef, "Could not find an object that matches $arg")
149     unless $ob;
150    
151 pippijn 1.29 $pl->contr->mark ($ob);
152 pippijn 1.28 $pl->reply (undef, (sprintf "Marked item %s", $ob->name, $ob->title));
153 pippijn 1.32 } else {
154     my $ob = $pl->find_marked_object;
155    
156     $pl->reply (undef, $ob
157     ? (sprintf "%s %s * is marked.", $ob->name, $ob->title)
158     : "You have no marked object.");
159 pippijn 1.28 }
160    
161     1
162     };
163    
164     for my $cmd ("run", "fire") {
165     my $oncmd = "${cmd}_on";
166     cf::register_command $cmd => sub {
167     my ($ob, $arg) = @_;
168    
169 pippijn 1.34 $ob->reply (undef, "Can't $cmd into a non adjacent square.")
170 pippijn 1.28 if $arg < 0 or $arg >= 9;
171    
172 pippijn 1.34 $ob->contr->$oncmd (1);
173     $ob->move_player ($arg);
174 pippijn 1.28
175     1
176     };
177 pippijn 1.34
178 pippijn 1.28 cf::register_command "${cmd}_stop" => sub {
179     my ($ob) = @_;
180    
181     $ob->contr->$oncmd (0);
182    
183     1
184     };
185     }
186    
187 pippijn 1.26 cf::register_command mapinfo => sub {
188     my ($ob) = @_;
189    
190 root 1.49 my $map = $ob->contr->observe->map
191 pippijn 1.26 or return;
192     $ob->reply (undef, (sprintf "%s (%s) %s", $map->name, $map->path, $ob->region->longname));
193     $ob->reply (undef, (sprintf "players: %d difficulty: %d size: %d start: %dx%d timeout: %d",
194     $map->players, $map->difficulty, $map->width, $map->height, $map->enter_x, $map->enter_y, $map->timeout))
195     if $ob->flag (cf::FLAG_WIZ);
196     $ob->reply (undef, $map->msg);
197    
198     1
199     };
200    
201 pippijn 1.25 cf::register_command whereami => sub {
202     my ($ob) = @_;
203    
204 root 1.50 my $reg = $ob->contr->observe->region;
205 pippijn 1.25 $ob->reply (undef, (sprintf "You are %s.\n%s", $reg->longname, $reg->msg));
206    
207     1
208     };
209    
210 root 1.31 sub _set_mode($$$@) {
211     my ($name, $ob, $arg, $slot, @choices) = @_;
212 pippijn 1.23
213 root 1.31 my $oldmode = $ob->contr->$slot;
214 pippijn 1.23
215 root 1.31 return $ob->reply (undef, "$name is set to $choices[$oldmode]")
216 pippijn 1.23 unless $arg;
217    
218 root 1.31 my ($idx) = grep $choices[$_] eq $arg, 0 .. $#choices
219     or return $ob->reply (undef, "$name: Unknown options '$arg', valid options are @choices"), 1;
220 pippijn 1.23
221 root 1.31 $ob->contr->$slot ($idx);
222     $ob->reply (undef, "$name" . ($oldmode == $ob->contr->unapply ? "" : " now") . " set to " . $choices[$ob->contr->$slot]);
223     }
224    
225     cf::register_command applymode => sub {
226     my ($ob, $arg) = @_;
227    
228     _set_mode "applymode", $ob, $arg, unapply => qw(nochoice never always);
229 pippijn 1.23
230     1
231     };
232    
233     cf::register_command petmode => sub {
234     my ($ob, $arg) = @_;
235    
236 root 1.31 _set_mode "petmode", $ob, $arg, petmode => qw(normal sad defend arena);
237 pippijn 1.23
238     1
239     };
240    
241 pippijn 1.21 cf::register_command usekeys => sub {
242     my ($ob, $arg) = @_;
243    
244 root 1.31 _set_mode "usekeys", $ob, $arg, usekeys => qw(inventory keyrings containers);
245 pippijn 1.22
246     1
247 pippijn 1.21 };
248    
249 pippijn 1.19 cf::register_command afk => sub {
250     my ($ob, $arg) = @_;
251    
252     $ob->contr->ns->afk ($ob->contr->ns->afk ? 0 : 1);
253     $ob->reply (undef, $ob->contr->ns->afk ? "You are now AFK" : "You are no longer AFK");
254 pippijn 1.22
255     1
256 pippijn 1.19 };
257    
258 pippijn 1.21 cf::register_command sound => sub {
259     my ($ob, $arg) = @_;
260    
261     $ob->contr->ns->sound ($ob->contr->ns->sound ? 0 : 1);
262     $ob->reply (undef, $ob->contr->ns->sound ? "The sounds are enabled." : "Silence is golden...");
263 pippijn 1.22
264     1
265 pippijn 1.21 };
266    
267 pippijn 1.20 cf::register_command brace => sub {
268     my ($ob, $arg) = @_;
269    
270     $ob->contr->braced ($ob->contr->braced ? 0 : 1);
271     $ob->reply (undef, $ob->contr->braced ? "You are braced." : "Not braced.");
272 pippijn 1.22
273     1
274 pippijn 1.20 };
275    
276 root 1.35 cf::register_command 'output-rate' => sub {
277     my ($ob, $arg) = @_;
278    
279     return $ob->reply (undef, sprintf "Output rate is presently %dbps.", $ob->contr->ns->max_rate / $cf::TICK)
280     unless $arg > 0;
281    
282 root 1.39 # minimum is 2k/s
283     $ob->contr->ns->max_rate ((List::Util::max 2048, $arg) * $cf::TICK);
284 root 1.36 $ob->reply (undef, sprintf "Output rate now set to %dbps.", $ob->contr->ns->max_rate / $cf::TICK);
285 root 1.35
286     1
287     };
288    
289 pippijn 1.24 cf::register_command 'output-count' => sub {
290     my ($ob, $arg) = @_;
291    
292     return $ob->reply (undef, "Output count is presently " . $ob->contr->outputs_count)
293     unless $arg > 0;
294    
295 root 1.41 $arg = 4 if $arg < 4;
296    
297 pippijn 1.24 $ob->contr->outputs_count ($arg);
298     $ob->reply (undef, "Output count now set to " . $ob->contr->outputs_count);
299    
300     1
301     };
302    
303     cf::register_command 'output-sync' => sub {
304     my ($ob, $arg) = @_;
305    
306 root 1.38 return $ob->reply (undef, sprintf "Output sync time is presently %.1fs", $ob->contr->outputs_sync * $cf::TICK)
307     unless length $arg;
308 pippijn 1.24
309 root 1.41 $arg = 0.5 if $arg < 0.5;
310    
311 root 1.38 $ob->contr->outputs_sync ($arg / $cf::TICK);
312     $ob->reply (undef, sprintf "Output sync time now set to %.1fs", $ob->contr->outputs_sync * $cf::TICK);
313 pippijn 1.24
314     1
315     };
316    
317 pippijn 1.20 # XXX: This has a bug. After one sets his wimpy level to 0 and resets it to
318     # some other level (which may also be 0), this does not get echoed,
319     # but it does get set.
320     cf::register_command wimpy => sub {
321     my ($ob, $arg) = @_;
322    
323     my $wimpy = $ob->run_away;
324     return $ob->reply (undef, "Your current wimpy level is $wimpy.")
325     if $arg eq "";
326    
327     return $ob->run_away ($arg) && $ob->reply (undef, "Your new wimpy level is $arg.")
328     if $arg =~ /^\d+$/ and $arg <= 100;
329    
330     $ob->reply (undef, "Incorrect parameters for wimpy: $arg");
331 pippijn 1.22
332     1
333 pippijn 1.20 };
334    
335     cf::register_command peaceful => sub {
336     my ($ob, $arg) = @_;
337    
338     $ob->reply (undef, "You cannot change your peaceful setting with this command."
339     ." Please speak to the priest in the temple of Gorokh"
340     ." if you want to become hostile or in temple of Valriel"
341     ." if you want to become peaceful again.");
342    
343     #$ob->contr->peaceful ($ob->contr->peaceful ? 0 : 1);
344     #$ob->reply (undef, $ob->contr->peaceful ? "You will attack other players." : "You will not attack other players.");
345 pippijn 1.22
346     1
347 pippijn 1.20 };
348    
349 root 1.1 cf::register_command rename => sub {
350     my ($ob, $arg) = @_;
351    
352     $ob->speed_left ($ob->speed_left - 0.25);
353    
354     if ($arg =~ /^\s* (?: <([^>]+)> \s+)? to \s+ <([^>]*)> \s*$/x) {
355     # compatibility syntax
356     rename_to $ob, $1, $2;
357     } elsif ($arg =~ /
358     ^\s*
359     (?:
360     (?: "((?:[^"]+|\\.)*)" | (\S+) )
361     \s+)?
362     to \s+
363     (?: "((?:[^"]+|\\.)*)" | (\S+) )
364     \s*$
365     /x) {
366     # does not unquote $1 or $3
367     rename_to $ob, $2||$1, $4||$3;
368     } else {
369     $ob->reply (undef, 'Syntax error. Rename usage: rename ["oldname"] to "newname"');
370     }
371    
372     1
373     };
374    
375     cf::register_command uptime => sub {
376     my ($ob, $arg) = @_;
377    
378     my $startup = POSIX::strftime "%Y-%m-%d %H:%M:%S %Z", localtime $cf::UPTIME;
379     my $runtime = sprintf "%.1f", (time - $cf::UPTIME) / 86400;
380     $ob->reply (undef, "server started $startup, uptime ${runtime}\d");
381    
382     1
383     };
384    
385 root 1.8 my %IN_MEMORY = (
386     cf::MAP_IN_MEMORY => "I",
387     cf::MAP_SWAPPED => "S",
388     cf::MAP_LOADING => "L",
389     );
390    
391 root 1.7 cf::register_command maps => sub {
392     my ($ob, $arg) = @_;
393    
394     no re 'eval'; $arg = qr<$arg>;
395    
396 root 1.13 my $format = "%2s %1s %3s %5s %.60s\n";
397 root 1.7
398 root 1.15 $ob->reply (undef, (sprintf $format, "Pl", "I", "Svd", "Reset", "Name"), cf::NDI_BLACK | cf::NDI_UNIQUE);
399 root 1.7
400     for (sort keys %cf::MAP) {
401     my $map = $cf::MAP{$_}
402     or next;
403    
404     next unless $map->path =~ $arg;
405 root 1.17 next if $map->{deny_list};
406 root 1.7
407 root 1.10 my $svd = int $cf::RUNTIME - $map->{last_save};
408     $svd = "++" if $svd > 99;
409    
410 root 1.14 $ob->reply (undef,
411     (sprintf $format,
412     (scalar $map->players),
413     $IN_MEMORY{$map->in_memory} || "?",
414     $svd,
415     (int $map->reset_at - $cf::RUNTIME),
416 root 1.17 $map->visible_name),
417 root 1.14 cf::NDI_BLACK | cf::NDI_UNIQUE);
418 root 1.7 }
419    
420     1
421     };
422