ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/lib/cf.pm
(Generate patch)

Comparing deliantra/server/lib/cf.pm (file contents):
Revision 1.65 by root, Tue Sep 12 22:18:55 2006 UTC vs.
Revision 1.140 by root, Fri Jan 5 20:04:02 2007 UTC

1package cf; 1package cf;
2
3use utf8;
4use strict;
2 5
3use Symbol; 6use Symbol;
4use List::Util; 7use List::Util;
5use Storable; 8use Storable;
6use Opcode; 9use Opcode;
7use Safe; 10use Safe;
8use Safe::Hole; 11use Safe::Hole;
9 12
13use Coro 3.3 ();
14use Coro::Event;
15use Coro::Timer;
16use Coro::Signal;
17use Coro::Semaphore;
18use Coro::AIO;
19
20use Digest::MD5;
21use Fcntl;
22use IO::AIO 2.31 ();
23use YAML::Syck ();
10use Time::HiRes; 24use Time::HiRes;
11use Event; 25
12$Event::Eval = 1; # no idea why this is required, but it is 26use Event; $Event::Eval = 1; # no idea why this is required, but it is
13 27
14use strict; 28# work around bug in YAML::Syck - bad news for perl6, will it be as broken wrt. unicode?
29$YAML::Syck::ImplicitUnicode = 1;
15 30
16_init_vars; 31$Coro::main->prio (Coro::PRIO_MAX); # run main coroutine ("the server") with very high priority
17 32
33sub WF_AUTOCANCEL () { 1 } # automatically cancel this watcher on reload
34
18our %COMMAND = (); 35our %COMMAND = ();
36our %COMMAND_TIME = ();
37our %EXTCMD = ();
38
19our @EVENT; 39our @EVENT;
20our $LIBDIR = maps_directory "perl"; 40our $LIBDIR = datadir . "/ext";
21 41
22our $TICK = MAX_TIME * 1e-6; 42our $TICK = MAX_TIME * 1e-6;
23our $TICK_WATCHER; 43our $TICK_WATCHER;
24our $NEXT_TICK; 44our $NEXT_TICK;
45our $NOW;
46
47our %CFG;
48
49our $UPTIME; $UPTIME ||= time;
50our $RUNTIME;
51
52our %MAP; # all maps
53our $LINK_MAP; # the special {link} map
54our $RANDOM_MAPS = cf::localdir . "/random";
55our %EXT_CORO;
56
57binmode STDOUT;
58binmode STDERR;
59
60# read virtual server time, if available
61unless ($RUNTIME || !-e cf::localdir . "/runtime") {
62 open my $fh, "<", cf::localdir . "/runtime"
63 or die "unable to read runtime file: $!";
64 $RUNTIME = <$fh> + 0.;
65}
66
67mkdir cf::localdir;
68mkdir cf::localdir . "/" . cf::playerdir;
69mkdir cf::localdir . "/" . cf::tmpdir;
70mkdir cf::localdir . "/" . cf::uniquedir;
71mkdir $RANDOM_MAPS;
72
73# a special map that is always available
74our $LINK_MAP;
75our $EMERGENCY_POSITION;
76
77#############################################################################
78
79=head2 GLOBAL VARIABLES
80
81=over 4
82
83=item $cf::UPTIME
84
85The timestamp of the server start (so not actually an uptime).
86
87=item $cf::RUNTIME
88
89The time this server has run, starts at 0 and is increased by $cf::TICK on
90every server tick.
91
92=item $cf::LIBDIR
93
94The perl library directory, where extensions and cf-specific modules can
95be found. It will be added to C<@INC> automatically.
96
97=item $cf::NOW
98
99The time of the last (current) server tick.
100
101=item $cf::TICK
102
103The interval between server ticks, in seconds.
104
105=item %cf::CFG
106
107Configuration for the server, loaded from C</etc/crossfire/config>, or
108from wherever your confdir points to.
109
110=back
111
112=cut
25 113
26BEGIN { 114BEGIN {
27 *CORE::GLOBAL::warn = sub { 115 *CORE::GLOBAL::warn = sub {
28 my $msg = join "", @_; 116 my $msg = join "", @_;
117 utf8::encode $msg;
118
29 $msg .= "\n" 119 $msg .= "\n"
30 unless $msg =~ /\n$/; 120 unless $msg =~ /\n$/;
31 121
32 print STDERR "cfperl: $msg";
33 LOG llevError, "cfperl: $msg"; 122 LOG llevError, "cfperl: $msg";
34 }; 123 };
35} 124}
36 125
126@safe::cf::global::ISA = @cf::global::ISA = 'cf::attachable';
127@safe::cf::object::ISA = @cf::object::ISA = 'cf::attachable';
128@safe::cf::player::ISA = @cf::player::ISA = 'cf::attachable';
129@safe::cf::client::ISA = @cf::client::ISA = 'cf::attachable';
130@safe::cf::map::ISA = @cf::map::ISA = 'cf::attachable';
37@safe::cf::object::player::ISA = @cf::object::player::ISA = 'cf::object'; 131@safe::cf::object::player::ISA = @cf::object::player::ISA = 'cf::object';
38 132
39# we bless all objects into (empty) derived classes to force a method lookup 133# we bless all objects into (empty) derived classes to force a method lookup
40# within the Safe compartment. 134# within the Safe compartment.
41for my $pkg (qw(cf::object cf::object::player cf::player cf::map cf::party cf::region cf::arch cf::living)) { 135for my $pkg (qw(
136 cf::global cf::attachable
137 cf::object cf::object::player
138 cf::client cf::player
139 cf::arch cf::living
140 cf::map cf::party cf::region
141)) {
42 no strict 'refs'; 142 no strict 'refs';
43 @{"safe::$pkg\::wrap::ISA"} = @{"$pkg\::wrap::ISA"} = $pkg; 143 @{"safe::$pkg\::wrap::ISA"} = @{"$pkg\::wrap::ISA"} = $pkg;
44} 144}
45 145
46$Event::DIED = sub { 146$Event::DIED = sub {
48}; 148};
49 149
50my %ext_pkg; 150my %ext_pkg;
51my @exts; 151my @exts;
52my @hook; 152my @hook;
53my %command;
54my %extcmd;
55 153
56############################################################################# 154=head2 UTILITY FUNCTIONS
57# utility functions 155
156=over 4
157
158=cut
58 159
59use JSON::Syck (); # TODO# replace by JSON::PC once working 160use JSON::Syck (); # TODO# replace by JSON::PC once working
161
162=item $ref = cf::from_json $json
163
164Converts a JSON string into the corresponding perl data structure.
165
166=cut
60 167
61sub from_json($) { 168sub from_json($) {
62 $JSON::Syck::ImplicitUnicode = 1; # work around JSON::Syck bugs 169 $JSON::Syck::ImplicitUnicode = 1; # work around JSON::Syck bugs
63 JSON::Syck::Load $_[0] 170 JSON::Syck::Load $_[0]
64} 171}
65 172
173=item $json = cf::to_json $ref
174
175Converts a perl data structure into its JSON representation.
176
177=cut
178
66sub to_json($) { 179sub to_json($) {
67 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs 180 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs
68 JSON::Syck::Dump $_[0] 181 JSON::Syck::Dump $_[0]
69} 182}
70 183
184=item cf::lock_wait $string
185
186Wait until the given lock is available. See cf::lock_acquire.
187
188=item my $lock = cf::lock_acquire $string
189
190Wait until the given lock is available and then acquires it and returns
191a Coro::guard object. If the guard object gets destroyed (goes out of scope,
192for example when the coroutine gets canceled), the lock is automatically
193returned.
194
195Lock names should begin with a unique identifier (for example, cf::map::find
196uses map_find and cf::map::load uses map_load).
197
198=cut
199
200our %LOCK;
201
202sub lock_wait($) {
203 my ($key) = @_;
204
205 # wait for lock, if any
206 while ($LOCK{$key}) {
207 push @{ $LOCK{$key} }, $Coro::current;
208 Coro::schedule;
209 }
210}
211
212sub lock_acquire($) {
213 my ($key) = @_;
214
215 # wait, to be sure we are not locked
216 lock_wait $key;
217
218 $LOCK{$key} = [];
219
220 Coro::guard {
221 # wake up all waiters, to be on the safe side
222 $_->ready for @{ delete $LOCK{$key} };
223 }
224}
225
226sub freeze_mainloop {
227 return unless $TICK_WATCHER->is_active;
228
229 my $guard = Coro::guard { $TICK_WATCHER->start };
230 $TICK_WATCHER->stop;
231 $guard
232}
233
234=item cf::async { BLOCK }
235
236Currently the same as Coro::async_pool, meaning you cannot use
237C<on_destroy>, C<join> or other gimmicks on these coroutines. The only
238thing you are allowed to do is call C<prio> on it.
239
240=cut
241
242BEGIN { *async = \&Coro::async_pool }
243
244=item cf::sync_job { BLOCK }
245
246The design of crossfire+ requires that the main coro ($Coro::main) is
247always able to handle events or runnable, as crossfire+ is only partly
248reentrant. Thus "blocking" it by e.g. waiting for I/O is not acceptable.
249
250If it must be done, put the blocking parts into C<sync_job>. This will run
251the given BLOCK in another coroutine while waiting for the result. The
252server will be frozen during this time, so the block should either finish
253fast or be very important.
254
255=cut
256
257sub sync_job(&) {
258 my ($job) = @_;
259
260 if ($Coro::current == $Coro::main) {
261 # this is the main coro, too bad, we have to block
262 # till the operation succeeds, freezing the server :/
263
264 # TODO: use suspend/resume instead
265 # (but this is cancel-safe)
266 my $freeze_guard = freeze_mainloop;
267
268 my $busy = 1;
269 my @res;
270
271 (async {
272 @res = eval { $job->() };
273 warn $@ if $@;
274 undef $busy;
275 })->prio (Coro::PRIO_MAX);
276
277 while ($busy) {
278 unless (Coro::cede) {
279 Coro::nready ? Event::one_event 0 : Event::one_event;
280 Coro::cede_notself unless Coro::cede;
281 }
282 }
283
284 wantarray ? @res : $res[0]
285 } else {
286 # we are in another coroutine, how wonderful, everything just works
287
288 $job->()
289 }
290}
291
292=item $coro = cf::async_ext { BLOCK }
293
294Like async, but this coro is automcatially being canceled when the
295extension calling this is being unloaded.
296
297=cut
298
299sub async_ext(&) {
300 my $cb = shift;
301
302 my $coro = &Coro::async ($cb);
303
304 $coro->on_destroy (sub {
305 delete $EXT_CORO{$coro+0};
306 });
307 $EXT_CORO{$coro+0} = $coro;
308
309 $coro
310}
311
312sub write_runtime {
313 my $runtime = cf::localdir . "/runtime";
314
315 my $fh = aio_open "$runtime~", O_WRONLY | O_CREAT, 0644
316 or return;
317
318 my $value = $cf::RUNTIME + 1 + 10; # 10 is the runtime save interval, for a monotonic clock
319 (aio_write $fh, 0, (length $value), $value, 0) <= 0
320 and return;
321
322 aio_fsync $fh
323 and return;
324
325 close $fh
326 or return;
327
328 aio_rename "$runtime~", $runtime
329 and return;
330
331 1
332}
333
334=back
335
336=cut
337
71############################################################################# 338#############################################################################
72# "new" plug-in system
73 339
74=head3 EVENTS AND OBJECT ATTACHMENTS 340package cf::path;
341
342sub new {
343 my ($class, $path, $base) = @_;
344
345 $path = $path->as_string if ref $path;
346
347 my $self = bless { }, $class;
348
349 # {... are special paths that are not touched
350 # ?xxx/... are special absolute paths
351 # ?random/... random maps
352 # /! non-realised random map exit
353 # /... normal maps
354 # ~/... per-player maps without a specific player (DO NOT USE)
355 # ~user/... per-player map of a specific user
356
357 if ($path =~ /^{/) {
358 # fine as it is
359 } elsif ($path =~ s{^\?random/}{}) {
360 Coro::AIO::aio_load "$cf::RANDOM_MAPS/$path.meta", my $data;
361 $self->{random} = cf::from_json $data;
362 } else {
363 if ($path =~ s{^~([^/]+)?}{}) {
364 $self->{user_rel} = 1;
365
366 if (defined $1) {
367 $self->{user} = $1;
368 } elsif ($base =~ m{^~([^/]+)/}) {
369 $self->{user} = $1;
370 } else {
371 warn "cannot resolve user-relative path without user <$path,$base>\n";
372 }
373 } elsif ($path =~ /^\//) {
374 # already absolute
375 } else {
376 $base =~ s{[^/]+/?$}{};
377 return $class->new ("$base/$path");
378 }
379
380 for ($path) {
381 redo if s{/\.?/}{/};
382 redo if s{/[^/]+/\.\./}{/};
383 }
384 }
385
386 $self->{path} = $path;
387
388 $self
389}
390
391# the name / primary key / in-game path
392sub as_string {
393 my ($self) = @_;
394
395 $self->{user_rel} ? "~$self->{user}$self->{path}"
396 : $self->{random} ? "?random/$self->{path}"
397 : $self->{path}
398}
399
400# the displayed name, this is a one way mapping
401sub visible_name {
402 my ($self) = @_;
403
404# if (my $rmp = $self->{random}) {
405# # todo: be more intelligent about this
406# "?random/$rmp->{origin_map}+$rmp->{origin_x}+$rmp->{origin_y}/$rmp->{dungeon_level}"
407# } else {
408 $self->as_string
409# }
410}
411
412# escape the /'s in the path
413sub _escaped_path {
414 # ∕ is U+2215
415 (my $path = $_[0]{path}) =~ s/\//∕/g;
416 $path
417}
418
419# the original (read-only) location
420sub load_path {
421 my ($self) = @_;
422
423 sprintf "%s/%s/%s", cf::datadir, cf::mapdir, $self->{path}
424}
425
426# the temporary/swap location
427sub save_path {
428 my ($self) = @_;
429
430 $self->{user_rel} ? sprintf "%s/%s/%s/%s", cf::localdir, cf::playerdir, $self->{user}, $self->_escaped_path
431 : $self->{random} ? sprintf "%s/%s", $RANDOM_MAPS, $self->{path}
432 : sprintf "%s/%s/%s", cf::localdir, cf::tmpdir, $self->_escaped_path
433}
434
435# the unique path, might be eq to save_path
436sub uniq_path {
437 my ($self) = @_;
438
439 $self->{user_rel} || $self->{random}
440 ? undef
441 : sprintf "%s/%s/%s", cf::localdir, cf::uniquedir, $self->_escaped_path
442}
443
444# return random map parameters, or undef
445sub random_map_params {
446 my ($self) = @_;
447
448 $self->{random}
449}
450
451# this is somewhat ugly, but style maps do need special treatment
452sub is_style_map {
453 $_[0]{path} =~ m{^/styles/}
454}
455
456package cf;
457
458#############################################################################
459
460=head2 ATTACHABLE OBJECTS
461
462Many objects in crossfire are so-called attachable objects. That means you can
463attach callbacks/event handlers (a collection of which is called an "attachment")
464to it. All such attachable objects support the following methods.
465
466In the following description, CLASS can be any of C<global>, C<object>
467C<player>, C<client> or C<map> (i.e. the attachable objects in
468crossfire+).
75 469
76=over 4 470=over 4
77 471
78=item $object->attach ($attachment, key => $value...)
79
80=item $object->detach ($attachment)
81
82Attach/detach a pre-registered attachment to an object.
83
84=item $player->attach ($attachment, key => $value...)
85
86=item $player->detach ($attachment)
87
88Attach/detach a pre-registered attachment to a player.
89
90=item $map->attach ($attachment, key => $value...) 472=item $attachable->attach ($attachment, key => $value...)
91 473
92=item $map->detach ($attachment) 474=item $attachable->detach ($attachment)
93 475
94Attach/detach a pre-registered attachment to a map. 476Attach/detach a pre-registered attachment to a specific object and give it
477the specified key/value pairs as arguments.
95 478
96=item $bool = $object->attached ($name) 479Example, attach a minesweeper attachment to the given object, making it a
48010x10 minesweeper game:
97 481
98=item $bool = $player->attached ($name) 482 $obj->attach (minesweeper => width => 10, height => 10);
99 483
100=item $bool = $map->attached ($name) 484=item $bool = $attachable->attached ($name)
101 485
102Checks wether the named attachment is currently attached to the object. 486Checks wether the named attachment is currently attached to the object.
103 487
104=item cf::attach_global ... 488=item cf::CLASS->attach ...
105 489
106Attach handlers for global events. 490=item cf::CLASS->detach ...
107 491
108This and all following C<attach_*>-functions expect any number of the 492Define an anonymous attachment and attach it to all objects of the given
109following handler/hook descriptions: 493CLASS. See the next function for an explanation of its arguments.
494
495You can attach to global events by using the C<cf::global> class.
496
497Example, log all player logins:
498
499 cf::player->attach (
500 on_login => sub {
501 my ($pl) = @_;
502 ...
503 },
504 );
505
506Example, attach to the jeweler skill:
507
508 cf::object->attach (
509 type => cf::SKILL,
510 subtype => cf::SK_JEWELER,
511 on_use_skill => sub {
512 my ($sk, $ob, $part, $dir, $msg) = @_;
513 ...
514 },
515 );
516
517=item cf::CLASS::attachment $name, ...
518
519Register an attachment by C<$name> through which attachable objects of the
520given CLASS can refer to this attachment.
521
522Some classes such as crossfire maps and objects can specify attachments
523that are attached at load/instantiate time, thus the need for a name.
524
525These calls expect any number of the following handler/hook descriptions:
110 526
111=over 4 527=over 4
112 528
113=item prio => $number 529=item prio => $number
114 530
116by another C<prio> setting). Lower priority handlers get executed 532by another C<prio> setting). Lower priority handlers get executed
117earlier. The default priority is C<0>, and many built-in handlers are 533earlier. The default priority is C<0>, and many built-in handlers are
118registered at priority C<-1000>, so lower priorities should not be used 534registered at priority C<-1000>, so lower priorities should not be used
119unless you know what you are doing. 535unless you know what you are doing.
120 536
537=item type => $type
538
539(Only for C<< cf::object->attach >> calls), limits the attachment to the
540given type of objects only (the additional parameter C<subtype> can be
541used to further limit to the given subtype).
542
121=item on_I<event> => \&cb 543=item on_I<event> => \&cb
122 544
123Call the given code reference whenever the named event happens (event is 545Call the given code reference whenever the named event happens (event is
124something like C<instantiate>, C<apply>, C<use_skill> and so on, and which 546something like C<instantiate>, C<apply>, C<use_skill> and so on, and which
125handlers are recognised generally depends on the type of object these 547handlers are recognised generally depends on the type of object these
134package and register them. Only handlers for eevents supported by the 556package and register them. Only handlers for eevents supported by the
135object/class are recognised. 557object/class are recognised.
136 558
137=back 559=back
138 560
139=item cf::attach_to_type $object_type, $subtype, ... 561Example, define an attachment called "sockpuppet" that calls the given
562event handler when a monster attacks:
140 563
141Attach handlers for a specific object type (e.g. TRANSPORT) and 564 cf::object::attachment sockpuppet =>
142subtype. If C<$subtype> is zero or undef, matches all objects of the given 565 on_skill_attack => sub {
143type. 566 my ($self, $victim) = @_;
144 567 ...
145=item cf::attach_to_objects ...
146
147Attach handlers to all objects. Do not use this except for debugging or
148very rare events, as handlers are (obviously) called for I<all> objects in
149the game.
150
151=item cf::attach_to_players ...
152
153Attach handlers to all players.
154
155=item cf::attach_to_maps ...
156
157Attach handlers to all maps.
158
159=item cf:register_attachment $name, ...
160
161Register an attachment by name through which objects can refer to this
162attachment.
163
164=item cf:register_player_attachment $name, ...
165
166Register an attachment by name through which players can refer to this
167attachment.
168
169=item cf:register_map_attachment $name, ...
170
171Register an attachment by name through which maps can refer to this
172attachment.
173
174=cut
175
176# the following variables are defined in .xs and must not be re-created
177our @CB_GLOBAL = (); # registry for all global events
178our @CB_OBJECT = (); # all objects (should not be used except in emergency)
179our @CB_PLAYER = ();
180our @CB_TYPE = (); # registry for type (cf-object class) based events
181our @CB_MAP = ();
182
183my %attachment;
184
185sub _attach_cb($\%$$$) {
186 my ($registry, $undo, $event, $prio, $cb) = @_;
187
188 use sort 'stable';
189
190 $cb = [$prio, $cb];
191
192 @{$registry->[$event]} = sort
193 { $a->[0] cmp $b->[0] }
194 @{$registry->[$event] || []}, $cb;
195
196 push @{$undo->{cb}}, [$event, $cb];
197}
198
199# attach handles attaching event callbacks
200# the only thing the caller has to do is pass the correct
201# registry (== where the callback attaches to).
202sub _attach(\@$@) {
203 my ($registry, $klass, @arg) = @_;
204
205 my $prio = 0;
206
207 my %undo = (
208 registry => $registry,
209 cb => [],
210 );
211
212 my %cb_id = map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == $klass, 0 .. $#EVENT;
213
214 while (@arg) {
215 my $type = shift @arg;
216
217 if ($type eq "prio") {
218 $prio = shift @arg;
219
220 } elsif ($type eq "package") {
221 my $pkg = shift @arg;
222
223 while (my ($name, $id) = each %cb_id) {
224 if (my $cb = $pkg->can ($name)) {
225 _attach_cb $registry, %undo, $id, $prio, $cb;
226 }
227 } 568 }
228
229 } elsif (exists $cb_id{$type}) {
230 _attach_cb $registry, %undo, $cb_id{$type}, $prio, shift @arg;
231
232 } elsif (ref $type) {
233 warn "attaching objects not supported, ignoring.\n";
234
235 } else {
236 shift @arg;
237 warn "attach argument '$type' not supported, ignoring.\n";
238 }
239 }
240
241 \%undo
242}
243
244sub _attach_attachment {
245 my ($obj, $name, %arg) = @_;
246
247 return if exists $obj->{_attachment}{$name};
248
249 my $res;
250
251 if (my $attach = $attachment{$name}) {
252 my $registry = $obj->registry;
253
254 for (@$attach) {
255 my ($klass, @attach) = @$_;
256 $res = _attach @$registry, $klass, @attach;
257 }
258
259 $obj->{$name} = \%arg;
260 } else {
261 warn "object uses attachment '$name' that is not available, postponing.\n";
262 }
263
264 $obj->{_attachment}{$name} = undef;
265
266 $res->{attachment} = $name;
267 $res
268}
269
270*cf::object::attach =
271*cf::player::attach =
272*cf::map::attach = sub {
273 my ($obj, $name, %arg) = @_;
274
275 _attach_attachment $obj, $name, %arg;
276};
277
278# all those should be optimised
279*cf::object::detach =
280*cf::player::detach =
281*cf::map::detach = sub {
282 my ($obj, $name) = @_;
283
284 delete $obj->{_attachment}{$name};
285 reattach ($obj);
286};
287
288*cf::object::attached =
289*cf::player::attached =
290*cf::map::attached = sub {
291 my ($obj, $name) = @_;
292
293 exists $obj->{_attachment}{$name}
294};
295
296sub attach_global {
297 _attach @CB_GLOBAL, KLASS_GLOBAL, @_
298}
299
300sub attach_to_type {
301 my $type = shift;
302 my $subtype = shift;
303
304 _attach @{$CB_TYPE[$type + $subtype * NUM_SUBTYPES]}, KLASS_OBJECT, @_
305}
306
307sub attach_to_objects {
308 _attach @CB_OBJECT, KLASS_OBJECT, @_
309}
310
311sub attach_to_players {
312 _attach @CB_PLAYER, KLASS_PLAYER, @_
313}
314
315sub attach_to_maps {
316 _attach @CB_MAP, KLASS_MAP, @_
317}
318
319sub register_attachment {
320 my $name = shift;
321
322 $attachment{$name} = [[KLASS_OBJECT, @_]];
323}
324
325sub register_player_attachment {
326 my $name = shift;
327
328 $attachment{$name} = [[KLASS_PLAYER, @_]];
329}
330
331sub register_map_attachment {
332 my $name = shift;
333
334 $attachment{$name} = [[KLASS_MAP, @_]];
335}
336
337our $override;
338our @invoke_results = (); # referenced from .xs code. TODO: play tricks with reify and mortals?
339
340sub override {
341 $override = 1;
342 @invoke_results = ();
343}
344
345sub do_invoke {
346 my $event = shift;
347 my $callbacks = shift;
348
349 @invoke_results = ();
350
351 local $override;
352
353 for (@$callbacks) {
354 eval { &{$_->[1]} };
355
356 if ($@) {
357 warn "$@";
358 warn "... while processing $EVENT[$event][0](@_) event, skipping processing altogether.\n";
359 override;
360 }
361
362 return 1 if $override;
363 }
364
365 0 569 }
366}
367 570
368=item $bool = cf::invoke EVENT_GLOBAL_XXX, ... 571=item $attachable->valid
369
370=item $bool = $object->invoke (EVENT_OBJECT_XXX, ...)
371
372=item $bool = $player->invoke (EVENT_PLAYER_XXX, ...)
373
374=item $bool = $map->invoke (EVENT_MAP_XXX, ...)
375
376Generate a global/object/player/map-specific event with the given arguments.
377
378This API is preliminary (most likely, the EVENT_KLASS_xxx prefix will be
379removed in future versions), and there is no public API to access override
380results (if you must, access C<@cf::invoke_results> directly).
381
382=back
383
384=head2 methods valid for all pointers
385
386=over 4
387
388=item $object->valid
389
390=item $player->valid
391
392=item $map->valid
393 572
394Just because you have a perl object does not mean that the corresponding 573Just because you have a perl object does not mean that the corresponding
395C-level object still exists. If you try to access an object that has no 574C-level object still exists. If you try to access an object that has no
396valid C counterpart anymore you get an exception at runtime. This method 575valid C counterpart anymore you get an exception at runtime. This method
397can be used to test for existence of the C object part without causing an 576can be used to test for existence of the C object part without causing an
398exception. 577exception.
399 578
579=cut
580
581# the following variables are defined in .xs and must not be re-created
582our @CB_GLOBAL = (); # registry for all global events
583our @CB_ATTACHABLE = (); # registry for all attachables
584our @CB_OBJECT = (); # all objects (should not be used except in emergency)
585our @CB_PLAYER = ();
586our @CB_CLIENT = ();
587our @CB_TYPE = (); # registry for type (cf-object class) based events
588our @CB_MAP = ();
589
590my %attachment;
591
592sub _attach_cb($$$$) {
593 my ($registry, $event, $prio, $cb) = @_;
594
595 use sort 'stable';
596
597 $cb = [$prio, $cb];
598
599 @{$registry->[$event]} = sort
600 { $a->[0] cmp $b->[0] }
601 @{$registry->[$event] || []}, $cb;
602}
603
604# hack
605my %attachable_klass = map +($_ => 1), KLASS_OBJECT, KLASS_CLIENT, KLASS_PLAYER, KLASS_MAP;
606
607# attach handles attaching event callbacks
608# the only thing the caller has to do is pass the correct
609# registry (== where the callback attaches to).
610sub _attach {
611 my ($registry, $klass, @arg) = @_;
612
613 my $object_type;
614 my $prio = 0;
615 my %cb_id = map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == $klass, 0 .. $#EVENT;
616
617 #TODO: get rid of this hack
618 if ($attachable_klass{$klass}) {
619 %cb_id = (%cb_id, map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == KLASS_ATTACHABLE, 0 .. $#EVENT);
620 }
621
622 while (@arg) {
623 my $type = shift @arg;
624
625 if ($type eq "prio") {
626 $prio = shift @arg;
627
628 } elsif ($type eq "type") {
629 $object_type = shift @arg;
630 $registry = $CB_TYPE[$object_type] ||= [];
631
632 } elsif ($type eq "subtype") {
633 defined $object_type or Carp::croak "subtype specified without type";
634 my $object_subtype = shift @arg;
635 $registry = $CB_TYPE[$object_type + $object_subtype * NUM_SUBTYPES] ||= [];
636
637 } elsif ($type eq "package") {
638 my $pkg = shift @arg;
639
640 while (my ($name, $id) = each %cb_id) {
641 if (my $cb = $pkg->can ($name)) {
642 _attach_cb $registry, $id, $prio, $cb;
643 }
644 }
645
646 } elsif (exists $cb_id{$type}) {
647 _attach_cb $registry, $cb_id{$type}, $prio, shift @arg;
648
649 } elsif (ref $type) {
650 warn "attaching objects not supported, ignoring.\n";
651
652 } else {
653 shift @arg;
654 warn "attach argument '$type' not supported, ignoring.\n";
655 }
656 }
657}
658
659sub _object_attach {
660 my ($obj, $name, %arg) = @_;
661
662 return if exists $obj->{_attachment}{$name};
663
664 if (my $attach = $attachment{$name}) {
665 my $registry = $obj->registry;
666
667 for (@$attach) {
668 my ($klass, @attach) = @$_;
669 _attach $registry, $klass, @attach;
670 }
671
672 $obj->{$name} = \%arg;
673 } else {
674 warn "object uses attachment '$name' that is not available, postponing.\n";
675 }
676
677 $obj->{_attachment}{$name} = undef;
678}
679
680sub cf::attachable::attach {
681 if (ref $_[0]) {
682 _object_attach @_;
683 } else {
684 _attach shift->_attach_registry, @_;
685 }
686};
687
688# all those should be optimised
689sub cf::attachable::detach {
690 my ($obj, $name) = @_;
691
692 if (ref $obj) {
693 delete $obj->{_attachment}{$name};
694 reattach ($obj);
695 } else {
696 Carp::croak "cannot, currently, detach class attachments";
697 }
698};
699
700sub cf::attachable::attached {
701 my ($obj, $name) = @_;
702
703 exists $obj->{_attachment}{$name}
704}
705
706for my $klass (qw(ATTACHABLE GLOBAL OBJECT PLAYER CLIENT MAP)) {
707 eval "#line " . __LINE__ . " 'cf.pm'
708 sub cf::\L$klass\E::_attach_registry {
709 (\\\@CB_$klass, KLASS_$klass)
710 }
711
712 sub cf::\L$klass\E::attachment {
713 my \$name = shift;
714
715 \$attachment{\$name} = [[KLASS_$klass, \@_]];
716 }
717 ";
718 die if $@;
719}
720
721our $override;
722our @invoke_results = (); # referenced from .xs code. TODO: play tricks with reify and mortals?
723
724sub override {
725 $override = 1;
726 @invoke_results = ();
727}
728
729sub do_invoke {
730 my $event = shift;
731 my $callbacks = shift;
732
733 @invoke_results = ();
734
735 local $override;
736
737 for (@$callbacks) {
738 eval { &{$_->[1]} };
739
740 if ($@) {
741 warn "$@";
742 warn "... while processing $EVENT[$event][0](@_) event, skipping processing altogether.\n";
743 override;
744 }
745
746 return 1 if $override;
747 }
748
749 0
750}
751
752=item $bool = cf::global::invoke (EVENT_CLASS_XXX, ...)
753
754=item $bool = $attachable->invoke (EVENT_CLASS_XXX, ...)
755
756Generate an object-specific event with the given arguments.
757
758This API is preliminary (most likely, the EVENT_CLASS_xxx prefix will be
759removed in future versions), and there is no public API to access override
760results (if you must, access C<@cf::invoke_results> directly).
761
400=back 762=back
401 763
402=cut 764=cut
403
404*cf::object::valid =
405*cf::player::valid =
406*cf::map::valid = \&cf::_valid;
407 765
408############################################################################# 766#############################################################################
409# object support 767# object support
410 768
411sub instantiate {
412 my ($obj, $data) = @_;
413
414 $data = from_json $data;
415
416 for (@$data) {
417 my ($name, $args) = @$_;
418
419 $obj->attach ($name, %{$args || {} });
420 }
421}
422
423# basically do the same as instantiate, without calling instantiate
424sub reattach { 769sub reattach {
770 # basically do the same as instantiate, without calling instantiate
425 my ($obj) = @_; 771 my ($obj) = @_;
772
426 my $registry = $obj->registry; 773 my $registry = $obj->registry;
427 774
428 @$registry = (); 775 @$registry = ();
429 776
430 delete $obj->{_attachment} unless scalar keys %{ $obj->{_attachment} || {} }; 777 delete $obj->{_attachment} unless scalar keys %{ $obj->{_attachment} || {} };
431 778
432 for my $name (keys %{ $obj->{_attachment} || {} }) { 779 for my $name (keys %{ $obj->{_attachment} || {} }) {
433 if (my $attach = $attachment{$name}) { 780 if (my $attach = $attachment{$name}) {
434 for (@$attach) { 781 for (@$attach) {
435 my ($klass, @attach) = @$_; 782 my ($klass, @attach) = @$_;
436 _attach @$registry, $klass, @attach; 783 _attach $registry, $klass, @attach;
437 } 784 }
438 } else { 785 } else {
439 warn "object uses attachment '$name' that is not available, postponing.\n"; 786 warn "object uses attachment '$name' that is not available, postponing.\n";
440 } 787 }
441 } 788 }
442} 789}
443 790
444sub object_freezer_save { 791cf::attachable->attach (
445 my ($filename, $rdata, $objs) = @_;
446
447 if (length $$rdata) {
448 warn sprintf "saving %s (%d,%d)\n",
449 $filename, length $$rdata, scalar @$objs;
450
451 if (open my $fh, ">:raw", "$filename~") {
452 chmod SAVE_MODE, $fh;
453 syswrite $fh, $$rdata;
454 close $fh;
455
456 if (@$objs && open my $fh, ">:raw", "$filename.pst~") {
457 chmod SAVE_MODE, $fh;
458 syswrite $fh, Storable::nfreeze { version => 1, objs => $objs };
459 close $fh;
460 rename "$filename.pst~", "$filename.pst";
461 } else {
462 unlink "$filename.pst";
463 }
464
465 rename "$filename~", $filename;
466 } else {
467 warn "FATAL: $filename~: $!\n";
468 }
469 } else {
470 unlink $filename;
471 unlink "$filename.pst";
472 }
473}
474
475sub object_thawer_load {
476 my ($filename) = @_;
477
478 local $/;
479
480 my $av;
481
482 #TODO: use sysread etc.
483 if (open my $data, "<:raw:perlio", $filename) {
484 $data = <$data>;
485 if (open my $pst, "<:raw:perlio", "$filename.pst") {
486 $av = eval { (Storable::thaw <$pst>)->{objs} };
487 }
488 return ($data, $av);
489 }
490
491 ()
492}
493
494attach_to_objects
495 prio => -1000000, 792 prio => -1000000,
793 on_instantiate => sub {
794 my ($obj, $data) = @_;
795
796 $data = from_json $data;
797
798 for (@$data) {
799 my ($name, $args) = @$_;
800
801 $obj->attach ($name, %{$args || {} });
802 }
803 },
804 on_reattach => \&reattach,
496 on_clone => sub { 805 on_clone => sub {
497 my ($src, $dst) = @_; 806 my ($src, $dst) = @_;
498 807
499 @{$dst->registry} = @{$src->registry}; 808 @{$dst->registry} = @{$src->registry};
500 809
501 %$dst = %$src; 810 %$dst = %$src;
502 811
503 %{$dst->{_attachment}} = %{$src->{_attachment}} 812 %{$dst->{_attachment}} = %{$src->{_attachment}}
504 if exists $src->{_attachment}; 813 if exists $src->{_attachment};
505 }, 814 },
506; 815);
816
817sub object_freezer_save {
818 my ($filename, $rdata, $objs) = @_;
819
820 sync_job {
821 if (length $$rdata) {
822 warn sprintf "saving %s (%d,%d)\n",
823 $filename, length $$rdata, scalar @$objs;
824
825 if (my $fh = aio_open "$filename~", O_WRONLY | O_CREAT, 0600) {
826 chmod SAVE_MODE, $fh;
827 aio_write $fh, 0, (length $$rdata), $$rdata, 0;
828 aio_fsync $fh;
829 close $fh;
830
831 if (@$objs) {
832 if (my $fh = aio_open "$filename.pst~", O_WRONLY | O_CREAT, 0600) {
833 chmod SAVE_MODE, $fh;
834 my $data = Storable::nfreeze { version => 1, objs => $objs };
835 aio_write $fh, 0, (length $data), $data, 0;
836 aio_fsync $fh;
837 close $fh;
838 aio_rename "$filename.pst~", "$filename.pst";
839 }
840 } else {
841 aio_unlink "$filename.pst";
842 }
843
844 aio_rename "$filename~", $filename;
845 } else {
846 warn "FATAL: $filename~: $!\n";
847 }
848 } else {
849 aio_unlink $filename;
850 aio_unlink "$filename.pst";
851 }
852 }
853}
854
855sub object_freezer_as_string {
856 my ($rdata, $objs) = @_;
857
858 use Data::Dumper;
859
860 $$rdata . Dumper $objs
861}
862
863sub object_thawer_load {
864 my ($filename) = @_;
865
866 my ($data, $av);
867
868 (aio_load $filename, $data) >= 0
869 or return;
870
871 unless (aio_stat "$filename.pst") {
872 (aio_load "$filename.pst", $av) >= 0
873 or return;
874 $av = eval { (Storable::thaw $av)->{objs} };
875 }
876
877 warn sprintf "loading %s (%d)\n",
878 $filename, length $data, scalar @{$av || []};#d#
879 return ($data, $av);
880}
507 881
508############################################################################# 882#############################################################################
509# old plug-in events 883# command handling &c
510 884
511sub inject_event { 885=item cf::register_command $name => \&callback($ob,$args);
512 my $extension = shift;
513 my $event_code = shift;
514 886
515 my $cb = $hook[$event_code]{$extension} 887Register a callback for execution when the client sends the user command
516 or return; 888$name.
517 889
518 &$cb 890=cut
519}
520
521sub inject_global_event {
522 my $event = shift;
523
524 my $cb = $hook[$event]
525 or return;
526
527 List::Util::max map &$_, values %$cb
528}
529
530sub inject_command {
531 my ($name, $obj, $params) = @_;
532
533 for my $cmd (@{ $command{$name} }) {
534 $cmd->[1]->($obj, $params);
535 }
536
537 -1
538}
539 891
540sub register_command { 892sub register_command {
541 my ($name, $time, $cb) = @_; 893 my ($name, $cb) = @_;
542 894
543 my $caller = caller; 895 my $caller = caller;
544 #warn "registering command '$name/$time' to '$caller'"; 896 #warn "registering command '$name/$time' to '$caller'";
545 897
546 push @{ $command{$name} }, [$time, $cb, $caller]; 898 push @{ $COMMAND{$name} }, [$caller, $cb];
547 $COMMAND{"$name\000"} = List::Util::max map $_->[0], @{ $command{$name} };
548} 899}
900
901=item cf::register_extcmd $name => \&callback($pl,$packet);
902
903Register a callbackf ro execution when the client sends an extcmd packet.
904
905If the callback returns something, it is sent back as if reply was being
906called.
907
908=cut
549 909
550sub register_extcmd { 910sub register_extcmd {
551 my ($name, $cb) = @_; 911 my ($name, $cb) = @_;
552 912
553 my $caller = caller; 913 my $caller = caller;
554 #warn "registering extcmd '$name' to '$caller'"; 914 #warn "registering extcmd '$name' to '$caller'";
555 915
556 $extcmd{$name} = [$cb, $caller]; 916 $EXTCMD{$name} = [$cb, $caller];
557} 917}
918
919cf::player->attach (
920 on_command => sub {
921 my ($pl, $name, $params) = @_;
922
923 my $cb = $COMMAND{$name}
924 or return;
925
926 for my $cmd (@$cb) {
927 $cmd->[1]->($pl->ob, $params);
928 }
929
930 cf::override;
931 },
932 on_extcmd => sub {
933 my ($pl, $buf) = @_;
934
935 my $msg = eval { from_json $buf };
936
937 if (ref $msg) {
938 if (my $cb = $EXTCMD{$msg->{msgtype}}) {
939 if (my %reply = $cb->[0]->($pl, $msg)) {
940 $pl->ext_reply ($msg->{msgid}, %reply);
941 }
942 }
943 } else {
944 warn "player " . ($pl->ob->name) . " sent unparseable ext message: <$buf>\n";
945 }
946
947 cf::override;
948 },
949);
558 950
559sub register { 951sub register {
560 my ($base, $pkg) = @_; 952 my ($base, $pkg) = @_;
561 953
562 #TODO 954 #TODO
581 . "#line 1 \"$path\"\n{\n" 973 . "#line 1 \"$path\"\n{\n"
582 . (do { local $/; <$fh> }) 974 . (do { local $/; <$fh> })
583 . "\n};\n1"; 975 . "\n};\n1";
584 976
585 eval $source 977 eval $source
586 or die "$path: $@"; 978 or die $@ ? "$path: $@\n"
979 : "extension disabled.\n";
587 980
588 push @exts, $pkg; 981 push @exts, $pkg;
589 $ext_pkg{$base} = $pkg; 982 $ext_pkg{$base} = $pkg;
590 983
591# no strict 'refs'; 984# no strict 'refs';
604# for my $idx (0 .. $#PLUGIN_EVENT) { 997# for my $idx (0 .. $#PLUGIN_EVENT) {
605# delete $hook[$idx]{$pkg}; 998# delete $hook[$idx]{$pkg};
606# } 999# }
607 1000
608 # remove commands 1001 # remove commands
609 for my $name (keys %command) { 1002 for my $name (keys %COMMAND) {
610 my @cb = grep $_->[2] ne $pkg, @{ $command{$name} }; 1003 my @cb = grep $_->[0] ne $pkg, @{ $COMMAND{$name} };
611 1004
612 if (@cb) { 1005 if (@cb) {
613 $command{$name} = \@cb; 1006 $COMMAND{$name} = \@cb;
614 $COMMAND{"$name\000"} = List::Util::max map $_->[0], @cb;
615 } else { 1007 } else {
616 delete $command{$name};
617 delete $COMMAND{"$name\000"}; 1008 delete $COMMAND{$name};
618 } 1009 }
619 } 1010 }
620 1011
621 # remove extcmds 1012 # remove extcmds
622 for my $name (grep $extcmd{$_}[1] eq $pkg, keys %extcmd) { 1013 for my $name (grep $EXTCMD{$_}[1] eq $pkg, keys %EXTCMD) {
623 delete $extcmd{$name}; 1014 delete $EXTCMD{$name};
624 } 1015 }
625 1016
626 if (my $cb = $pkg->can ("unload")) { 1017 if (my $cb = $pkg->can ("unload")) {
627 eval { 1018 eval {
628 $cb->($pkg); 1019 $cb->($pkg);
632 1023
633 Symbol::delete_package $pkg; 1024 Symbol::delete_package $pkg;
634} 1025}
635 1026
636sub load_extensions { 1027sub load_extensions {
637 my $LIBDIR = maps_directory "perl";
638
639 for my $ext (<$LIBDIR/*.ext>) { 1028 for my $ext (<$LIBDIR/*.ext>) {
640 next unless -r $ext; 1029 next unless -r $ext;
641 eval { 1030 eval {
642 load_extension $ext; 1031 load_extension $ext;
643 1 1032 1
644 } or warn "$ext not loaded: $@"; 1033 } or warn "$ext not loaded: $@";
645 } 1034 }
646} 1035}
647 1036
648############################################################################# 1037#############################################################################
649# extcmd framework, basically convert ext <msg>
650# into pkg::->on_extcmd_arg1 (...) while shortcutting a few
651
652attach_to_players
653 on_extcmd => sub {
654 my ($pl, $buf) = @_;
655
656 my $msg = eval { from_json $buf };
657
658 if (ref $msg) {
659 if (my $cb = $extcmd{$msg->{msgtype}}) {
660 if (my %reply = $cb->[0]->($pl, $msg)) {
661 $pl->ext_reply ($msg->{msgid}, %reply);
662 }
663 }
664 } else {
665 warn "player " . ($pl->ob->name) . " sent unparseable ext message: <$buf>\n";
666 }
667
668 cf::override;
669 },
670;
671
672#############################################################################
673# load/save/clean perl data associated with a map 1038# load/save/clean perl data associated with a map
674 1039
675*cf::mapsupport::on_clean = sub { 1040*cf::mapsupport::on_clean = sub {
676 my ($map) = @_; 1041 my ($map) = @_;
677 1042
679 defined $path or return; 1044 defined $path or return;
680 1045
681 unlink "$path.pst"; 1046 unlink "$path.pst";
682}; 1047};
683 1048
684attach_to_maps prio => -10000, package => cf::mapsupport::; 1049cf::map->attach (prio => -10000, package => cf::mapsupport::);
685 1050
686############################################################################# 1051#############################################################################
687# load/save perl data associated with player->ob objects 1052# load/save perl data associated with player->ob objects
688 1053
689sub all_objects(@) { 1054sub all_objects(@) {
690 @_, map all_objects ($_->inv), @_ 1055 @_, map all_objects ($_->inv), @_
691} 1056}
692 1057
693# TODO: compatibility cruft, remove when no longer needed 1058# TODO: compatibility cruft, remove when no longer needed
694attach_to_players 1059cf::player->attach (
695 on_load => sub { 1060 on_load => sub {
696 my ($pl, $path) = @_; 1061 my ($pl, $path) = @_;
697 1062
698 for my $o (all_objects $pl->ob) { 1063 for my $o (all_objects $pl->ob) {
699 if (my $value = $o->get_ob_key_value ("_perl_data")) { 1064 if (my $value = $o->get_ob_key_value ("_perl_data")) {
701 1066
702 %$o = %{ Storable::thaw pack "H*", $value }; 1067 %$o = %{ Storable::thaw pack "H*", $value };
703 } 1068 }
704 } 1069 }
705 }, 1070 },
706; 1071);
707 1072
708############################################################################# 1073#############################################################################
709# core extensions - in perl 1074
1075=head2 CORE EXTENSIONS
1076
1077Functions and methods that extend core crossfire objects.
1078
1079=head3 cf::player
1080
1081=over 4
710 1082
711=item cf::player::exists $login 1083=item cf::player::exists $login
712 1084
713Returns true when the given account exists. 1085Returns true when the given account exists.
714 1086
717sub cf::player::exists($) { 1089sub cf::player::exists($) {
718 cf::player::find $_[0] 1090 cf::player::find $_[0]
719 or -f sprintf "%s/%s/%s/%s.pl", cf::localdir, cf::playerdir, ($_[0]) x 2; 1091 or -f sprintf "%s/%s/%s/%s.pl", cf::localdir, cf::playerdir, ($_[0]) x 2;
720} 1092}
721 1093
1094=item $player->ext_reply ($msgid, $msgtype, %msg)
1095
1096Sends an ext reply to the player.
1097
1098=cut
1099
1100sub cf::player::ext_reply($$$%) {
1101 my ($self, $id, %msg) = @_;
1102
1103 $msg{msgid} = $id;
1104
1105 $self->send ("ext " . to_json \%msg);
1106}
1107
1108=back
1109
1110
1111=head3 cf::map
1112
1113=over 4
1114
1115=cut
1116
1117package cf::map;
1118
1119use Fcntl;
1120use Coro::AIO;
1121
1122our $MAX_RESET = 3600;
1123our $DEFAULT_RESET = 3000;
1124
1125sub generate_random_map {
1126 my ($path, $rmp) = @_;
1127
1128 # mit "rum" bekleckern, nicht
1129 cf::map::_create_random_map
1130 $path,
1131 $rmp->{wallstyle}, $rmp->{wall_name}, $rmp->{floorstyle}, $rmp->{monsterstyle},
1132 $rmp->{treasurestyle}, $rmp->{layoutstyle}, $rmp->{doorstyle}, $rmp->{decorstyle},
1133 $rmp->{origin_map}, $rmp->{final_map}, $rmp->{exitstyle}, $rmp->{this_map},
1134 $rmp->{exit_on_final_map},
1135 $rmp->{xsize}, $rmp->{ysize},
1136 $rmp->{expand2x}, $rmp->{layoutoptions1}, $rmp->{layoutoptions2}, $rmp->{layoutoptions3},
1137 $rmp->{symmetry}, $rmp->{difficulty}, $rmp->{difficulty_given}, $rmp->{difficulty_increase},
1138 $rmp->{dungeon_level}, $rmp->{dungeon_depth}, $rmp->{decoroptions}, $rmp->{orientation},
1139 $rmp->{origin_y}, $rmp->{origin_x}, $rmp->{random_seed}, $rmp->{total_map_hp},
1140 $rmp->{map_layout_style}, $rmp->{treasureoptions}, $rmp->{symmetry_used},
1141 (cf::region::find $rmp->{region})
1142}
1143
1144# and all this just because we cannot iterate over
1145# all maps in C++...
1146sub change_all_map_light {
1147 my ($change) = @_;
1148
1149 $_->change_map_light ($change)
1150 for grep $_->outdoor, values %cf::MAP;
1151}
1152
1153sub try_load_header($) {
1154 my ($path) = @_;
1155
1156 utf8::encode $path;
1157 aio_open $path, O_RDONLY, 0
1158 or return;
1159
1160 my $map = cf::map::new
1161 or return;
1162
1163 # for better error messages only, will be overwritten
1164 $map->path ($path);
1165
1166 $map->load_header ($path)
1167 or return;
1168
1169 $map->{load_path} = $path;
1170
1171 $map
1172}
1173
1174sub find;
1175sub find {
1176 my ($path, $origin) = @_;
1177
1178 #warn "find<$path,$origin>\n";#d#
1179
1180 $path = new cf::path $path, $origin && $origin->path;
1181 my $key = $path->as_string;
1182
1183 cf::lock_wait "map_find:$key";
1184
1185 $cf::MAP{$key} || do {
1186 my $guard = cf::lock_acquire "map_find:$key";
1187
1188 # do it the slow way
1189 my $map = try_load_header $path->save_path;
1190
1191 Coro::cede;
1192
1193 if ($map) {
1194 $map->last_access ((delete $map->{last_access})
1195 || $cf::RUNTIME); #d#
1196 # safety
1197 $map->{instantiate_time} = $cf::RUNTIME
1198 if $map->{instantiate_time} > $cf::RUNTIME;
1199 } else {
1200 if (my $rmp = $path->random_map_params) {
1201 $map = generate_random_map $key, $rmp;
1202 } else {
1203 $map = try_load_header $path->load_path;
1204 }
1205
1206 $map or return;
1207
1208 $map->{load_original} = 1;
1209 $map->{instantiate_time} = $cf::RUNTIME;
1210 $map->last_access ($cf::RUNTIME);
1211 $map->instantiate;
1212
1213 # per-player maps become, after loading, normal maps
1214 $map->per_player (0) if $path->{user_rel};
1215 }
1216
1217 $map->path ($key);
1218 $map->{path} = $path;
1219 $map->{last_save} = $cf::RUNTIME;
1220
1221 Coro::cede;
1222
1223 if ($map->should_reset) {
1224 $map->reset;
1225 undef $guard;
1226 $map = find $path
1227 or return;
1228 }
1229
1230 $cf::MAP{$key} = $map
1231 }
1232}
1233
1234sub load {
1235 my ($self) = @_;
1236
1237 my $path = $self->{path};
1238 my $guard = cf::lock_acquire "map_load:" . $path->as_string;
1239
1240 return if $self->in_memory != cf::MAP_SWAPPED;
1241
1242 $self->in_memory (cf::MAP_LOADING);
1243
1244 $self->alloc;
1245 $self->load_objects ($self->{load_path}, 1)
1246 or return;
1247
1248 $self->set_object_flag (cf::FLAG_OBJ_ORIGINAL, 1)
1249 if delete $self->{load_original};
1250
1251 if (my $uniq = $path->uniq_path) {
1252 utf8::encode $uniq;
1253 if (aio_open $uniq, O_RDONLY, 0) {
1254 $self->clear_unique_items;
1255 $self->load_objects ($uniq, 0);
1256 }
1257 }
1258
1259 Coro::cede;
1260
1261 # now do the right thing for maps
1262 $self->link_multipart_objects;
1263
1264 if ($self->{path}->is_style_map) {
1265 $self->{deny_save} = 1;
1266 $self->{deny_reset} = 1;
1267 } else {
1268 $self->fix_auto_apply;
1269 $self->decay_objects;
1270 $self->update_buttons;
1271 $self->set_darkness_map;
1272 $self->difficulty ($self->estimate_difficulty)
1273 unless $self->difficulty;
1274 $self->activate;
1275 }
1276
1277 Coro::cede;
1278
1279 $self->in_memory (cf::MAP_IN_MEMORY);
1280}
1281
1282sub find_sync {
1283 my ($path, $origin) = @_;
1284
1285 cf::sync_job { cf::map::find $path, $origin }
1286}
1287
1288sub do_load_sync {
1289 my ($map) = @_;
1290
1291 cf::sync_job { $map->load };
1292}
1293
1294sub save {
1295 my ($self) = @_;
1296
1297 my $lock = cf::lock_acquire "map_data:" . $self->path;
1298
1299 $self->{last_save} = $cf::RUNTIME;
1300
1301 return unless $self->dirty;
1302
1303 my $save = $self->{path}->save_path; utf8::encode $save;
1304 my $uniq = $self->{path}->uniq_path; utf8::encode $uniq;
1305
1306 $self->{load_path} = $save;
1307
1308 return if $self->{deny_save};
1309
1310 local $self->{last_access} = $self->last_access;#d#
1311
1312 if ($uniq) {
1313 $self->save_objects ($save, cf::IO_HEADER | cf::IO_OBJECTS);
1314 $self->save_objects ($uniq, cf::IO_UNIQUES);
1315 } else {
1316 $self->save_objects ($save, cf::IO_HEADER | cf::IO_OBJECTS | cf::IO_UNIQUES);
1317 }
1318}
1319
1320sub swap_out {
1321 my ($self) = @_;
1322
1323 # save first because save cedes
1324 $self->save;
1325
1326 my $lock = cf::lock_acquire "map_data:" . $self->path;
1327
1328 return if $self->players;
1329 return if $self->in_memory != cf::MAP_IN_MEMORY;
1330 return if $self->{deny_save};
1331
1332 $self->clear;
1333 $self->in_memory (cf::MAP_SWAPPED);
1334}
1335
1336sub reset_at {
1337 my ($self) = @_;
1338
1339 # TODO: safety, remove and allow resettable per-player maps
1340 return 1e99 if $self->{path}{user_rel};
1341 return 1e99 if $self->{deny_reset};
1342
1343 my $time = $self->fixed_resettime ? $self->{instantiate_time} : $self->last_access;
1344 my $to = List::Util::min $MAX_RESET, $self->reset_timeout || $DEFAULT_RESET;
1345
1346 $time + $to
1347}
1348
1349sub should_reset {
1350 my ($self) = @_;
1351
1352 $self->reset_at <= $cf::RUNTIME
1353}
1354
1355sub unlink_save {
1356 my ($self) = @_;
1357
1358 utf8::encode (my $save = $self->{path}->save_path);
1359 aioreq_pri 3; IO::AIO::aio_unlink $save;
1360 aioreq_pri 3; IO::AIO::aio_unlink "$save.pst";
1361}
1362
1363sub rename {
1364 my ($self, $new_path) = @_;
1365
1366 $self->unlink_save;
1367
1368 delete $cf::MAP{$self->path};
1369 $self->{path} = new cf::path $new_path;
1370 $self->path ($self->{path}->as_string);
1371 $cf::MAP{$self->path} = $self;
1372
1373 $self->save;
1374}
1375
1376sub reset {
1377 my ($self) = @_;
1378
1379 my $lock = cf::lock_acquire "map_data:" . $self->path;
1380
1381 return if $self->players;
1382 return if $self->{path}{user_rel};#d#
1383
1384 warn "resetting map ", $self->path;#d#
1385
1386 delete $cf::MAP{$self->path};
1387
1388 $_->clear_links_to ($self) for values %cf::MAP;
1389
1390 $self->unlink_save;
1391 $self->destroy;
1392}
1393
1394my $nuke_counter = "aaaa";
1395
1396sub nuke {
1397 my ($self) = @_;
1398
1399 $self->{deny_save} = 1;
1400 $self->reset_timeout (1);
1401 $self->rename ("{nuke}/" . ($nuke_counter++));
1402 $self->reset; # polite request, might not happen
1403}
1404
1405sub customise_for {
1406 my ($map, $ob) = @_;
1407
1408 if ($map->per_player) {
1409 return cf::map::find "~" . $ob->name . "/" . $map->{path}{path};
1410 }
1411
1412 $map
1413}
1414
1415sub emergency_save {
1416 my $freeze_guard = cf::freeze_mainloop;
1417
1418 warn "enter emergency map save\n";
1419
1420 cf::sync_job {
1421 warn "begin emergency map save\n";
1422 $_->save for values %cf::MAP;
1423 };
1424
1425 warn "end emergency map save\n";
1426}
1427
1428package cf;
1429
1430=back
1431
1432
1433=head3 cf::object::player
1434
1435=over 4
1436
722=item $player->reply ($npc, $msg[, $flags]) 1437=item $player_object->reply ($npc, $msg[, $flags])
723 1438
724Sends a message to the player, as if the npc C<$npc> replied. C<$npc> 1439Sends a message to the player, as if the npc C<$npc> replied. C<$npc>
725can be C<undef>. Does the right thing when the player is currently in a 1440can be C<undef>. Does the right thing when the player is currently in a
726dialogue with the given NPC character. 1441dialogue with the given NPC character.
727 1442
728=cut 1443=cut
729 1444
730# rough implementation of a future "reply" method that works 1445# rough implementation of a future "reply" method that works
731# with dialog boxes. 1446# with dialog boxes.
1447#TODO: the first argument must go, split into a $npc->reply_to ( method
732sub cf::object::player::reply($$$;$) { 1448sub cf::object::player::reply($$$;$) {
733 my ($self, $npc, $msg, $flags) = @_; 1449 my ($self, $npc, $msg, $flags) = @_;
734 1450
735 $flags = cf::NDI_BROWN | cf::NDI_UNIQUE unless @_ >= 4; 1451 $flags = cf::NDI_BROWN | cf::NDI_UNIQUE unless @_ >= 4;
736 1452
740 $msg = $npc->name . " says: $msg" if $npc; 1456 $msg = $npc->name . " says: $msg" if $npc;
741 $self->message ($msg, $flags); 1457 $self->message ($msg, $flags);
742 } 1458 }
743} 1459}
744 1460
745=item $player->ext_reply ($msgid, $msgtype, %msg) 1461=item $player_object->may ("access")
746 1462
747Sends an ext reply to the player. 1463Returns wether the given player is authorized to access resource "access"
1464(e.g. "command_wizcast").
748 1465
749=cut 1466=cut
750 1467
751sub cf::player::ext_reply($$$%) { 1468sub cf::object::player::may {
1469 my ($self, $access) = @_;
1470
1471 $self->flag (cf::FLAG_WIZ) ||
1472 (ref $cf::CFG{"may_$access"}
1473 ? scalar grep $self->name eq $_, @{$cf::CFG{"may_$access"}}
1474 : $cf::CFG{"may_$access"})
1475}
1476
1477=item $player_object->enter_link
1478
1479Freezes the player and moves him/her to a special map (C<{link}>).
1480
1481The player should be reaosnably safe there for short amounts of time. You
1482I<MUST> call C<leave_link> as soon as possible, though.
1483
1484=item $player_object->leave_link ($map, $x, $y)
1485
1486Moves the player out of the specila link map onto the given map. If the
1487map is not valid (or omitted), the player will be moved back to the
1488location he/she was before the call to C<enter_link>, or, if that fails,
1489to the emergency map position.
1490
1491Might block.
1492
1493=cut
1494
1495sub cf::object::player::enter_link {
1496 my ($self) = @_;
1497
1498 $self->deactivate_recursive;
1499
1500 return if $self->map == $LINK_MAP;
1501
1502 $self->{_link_pos} ||= [$self->map->{path}, $self->x, $self->y]
1503 if $self->map;
1504
1505 $self->enter_map ($LINK_MAP, 20, 20);
1506}
1507
1508sub cf::object::player::leave_link {
1509 my ($self, $map, $x, $y) = @_;
1510
1511 my $link_pos = delete $self->{_link_pos};
1512
1513 unless ($map) {
1514 # restore original map position
1515 ($map, $x, $y) = @{ $link_pos || [] };
1516 $map = cf::map::find $map;
1517
1518 unless ($map) {
1519 ($map, $x, $y) = @$EMERGENCY_POSITION;
1520 $map = cf::map::find $map
1521 or die "FATAL: cannot load emergency map\n";
1522 }
1523 }
1524
1525 ($x, $y) = (-1, -1)
1526 unless (defined $x) && (defined $y);
1527
1528 # use -1 or undef as default coordinates, not 0, 0
1529 ($x, $y) = ($map->enter_x, $map->enter_y)
1530 if $x <=0 && $y <= 0;
1531
1532 $map->load;
1533
1534 $self->activate_recursive;
1535 $self->enter_map ($map, $x, $y);
1536}
1537
1538cf::player->attach (
1539 on_logout => sub {
1540 my ($pl) = @_;
1541
1542 # abort map switching before logout
1543 if ($pl->ob->{_link_pos}) {
1544 cf::sync_job {
1545 $pl->ob->leave_link
1546 };
1547 }
1548 },
1549 on_login => sub {
1550 my ($pl) = @_;
1551
1552 # try to abort aborted map switching on player login :)
1553 # should happen only on crashes
1554 if ($pl->ob->{_link_pos}) {
1555
1556 $pl->ob->enter_link;
1557 (async {
1558 # we need this sleep as the login has a concurrent enter_exit running
1559 # and this sleep increases chances of the player not ending up in scorn
1560 $pl->ob->reply (undef,
1561 "There was an internal problem at your last logout, "
1562 . "the server will try to bring you to your intended destination in a second.",
1563 cf::NDI_RED);
1564 Coro::Timer::sleep 1;
1565 $pl->ob->leave_link;
1566 })->prio (2);
1567 }
1568 },
1569);
1570
1571=item $player_object->goto ($path, $x, $y)
1572
1573=cut
1574
1575sub cf::object::player::goto {
1576 my ($self, $path, $x, $y) = @_;
1577
1578 $self->enter_link;
1579
1580 (async {
1581 $path = new cf::path $path;
1582
1583 my $map = cf::map::find $path->as_string;
1584 $map = $map->customise_for ($self) if $map;
1585
1586# warn "entering ", $map->path, " at ($x, $y)\n"
1587# if $map;
1588
1589 $map or $self->message ("The exit is closed", cf::NDI_UNIQUE | cf::NDI_RED);
1590
1591 $self->leave_link ($map, $x, $y);
1592 })->prio (1);
1593}
1594
1595=item $player_object->enter_exit ($exit_object)
1596
1597=cut
1598
1599sub parse_random_map_params {
1600 my ($spec) = @_;
1601
1602 my $rmp = { # defaults
1603 xsize => 10,
1604 ysize => 10,
1605 };
1606
1607 for (split /\n/, $spec) {
1608 my ($k, $v) = split /\s+/, $_, 2;
1609
1610 $rmp->{lc $k} = $v if (length $k) && (length $v);
1611 }
1612
1613 $rmp
1614}
1615
1616sub prepare_random_map {
1617 my ($exit) = @_;
1618
1619 # all this does is basically replace the /! path by
1620 # a new random map path (?random/...) with a seed
1621 # that depends on the exit object
1622
1623 my $rmp = parse_random_map_params $exit->msg;
1624
1625 if ($exit->map) {
1626 $rmp->{region} = $exit->map->region_name;
1627 $rmp->{origin_map} = $exit->map->path;
1628 $rmp->{origin_x} = $exit->x;
1629 $rmp->{origin_y} = $exit->y;
1630 }
1631
1632 $rmp->{random_seed} ||= $exit->random_seed;
1633
1634 my $data = cf::to_json $rmp;
1635 my $md5 = Digest::MD5::md5_hex $data;
1636
1637 if (my $fh = aio_open "$cf::RANDOM_MAPS/$md5.meta", O_WRONLY | O_CREAT, 0666) {
1638 aio_write $fh, 0, (length $data), $data, 0;
1639
1640 $exit->slaying ("?random/$md5");
1641 $exit->msg (undef);
1642 }
1643}
1644
1645sub cf::object::player::enter_exit {
752 my ($self, $id, %msg) = @_; 1646 my ($self, $exit) = @_;
753 1647
754 $msg{msgid} = $id; 1648 return unless $self->type == cf::PLAYER;
755 1649
756 $self->send ("ext " . to_json \%msg); 1650 $self->enter_link;
757}
758 1651
759############################################################################# 1652 (async {
760# map scripting support 1653 $self->deactivate_recursive; # just to be sure
1654 unless (eval {
1655 prepare_random_map $exit
1656 if $exit->slaying eq "/!";
1657
1658 my $path = new cf::path $exit->slaying, $exit->map && $exit->map->path;
1659 $self->goto ($path, $exit->stats->hp, $exit->stats->sp);
1660
1661 1;
1662 }) {
1663 $self->message ("Something went wrong deep within the crossfire server. "
1664 . "I'll try to bring you back to the map you were before. "
1665 . "Please report this to the dungeon master",
1666 cf::NDI_UNIQUE | cf::NDI_RED);
1667
1668 warn "ERROR in enter_exit: $@";
1669 $self->leave_link;
1670 }
1671 })->prio (1);
1672}
1673
1674=head3 cf::client
1675
1676=over 4
1677
1678=item $client->send_drawinfo ($text, $flags)
1679
1680Sends a drawinfo packet to the client. Circumvents output buffering so
1681should not be used under normal circumstances.
1682
1683=cut
1684
1685sub cf::client::send_drawinfo {
1686 my ($self, $text, $flags) = @_;
1687
1688 utf8::encode $text;
1689 $self->send_packet (sprintf "drawinfo %d %s", $flags, $text);
1690}
1691
1692
1693=item $success = $client->query ($flags, "text", \&cb)
1694
1695Queues a query to the client, calling the given callback with
1696the reply text on a reply. flags can be C<cf::CS_QUERY_YESNO>,
1697C<cf::CS_QUERY_SINGLECHAR> or C<cf::CS_QUERY_HIDEINPUT> or C<0>.
1698
1699Queries can fail, so check the return code. Or don't, as queries will become
1700reliable at some point in the future.
1701
1702=cut
1703
1704sub cf::client::query {
1705 my ($self, $flags, $text, $cb) = @_;
1706
1707 return unless $self->state == ST_PLAYING
1708 || $self->state == ST_SETUP
1709 || $self->state == ST_CUSTOM;
1710
1711 $self->state (ST_CUSTOM);
1712
1713 utf8::encode $text;
1714 push @{ $self->{query_queue} }, [(sprintf "query %d %s", $flags, $text), $cb];
1715
1716 $self->send_packet ($self->{query_queue}[0][0])
1717 if @{ $self->{query_queue} } == 1;
1718}
1719
1720cf::client->attach (
1721 on_reply => sub {
1722 my ($ns, $msg) = @_;
1723
1724 # this weird shuffling is so that direct followup queries
1725 # get handled first
1726 my $queue = delete $ns->{query_queue}
1727 or return; # be conservative, not sure how that can happen, but we saw a crash here
1728
1729 (shift @$queue)->[1]->($msg);
1730
1731 push @{ $ns->{query_queue} }, @$queue;
1732
1733 if (@{ $ns->{query_queue} } == @$queue) {
1734 if (@$queue) {
1735 $ns->send_packet ($ns->{query_queue}[0][0]);
1736 } else {
1737 $ns->state (ST_PLAYING) if $ns->state == ST_CUSTOM;
1738 }
1739 }
1740 },
1741);
1742
1743=item $client->async (\&cb)
1744
1745Create a new coroutine, running the specified callback. The coroutine will
1746be automatically cancelled when the client gets destroyed (e.g. on logout,
1747or loss of connection).
1748
1749=cut
1750
1751sub cf::client::async {
1752 my ($self, $cb) = @_;
1753
1754 my $coro = &Coro::async ($cb);
1755
1756 $coro->on_destroy (sub {
1757 delete $self->{_coro}{$coro+0};
1758 });
1759
1760 $self->{_coro}{$coro+0} = $coro;
1761
1762 $coro
1763}
1764
1765cf::client->attach (
1766 on_destroy => sub {
1767 my ($ns) = @_;
1768
1769 $_->cancel for values %{ (delete $ns->{_coro}) || {} };
1770 },
1771);
1772
1773=back
1774
1775
1776=head2 SAFE SCRIPTING
1777
1778Functions that provide a safe environment to compile and execute
1779snippets of perl code without them endangering the safety of the server
1780itself. Looping constructs, I/O operators and other built-in functionality
1781is not available in the safe scripting environment, and the number of
1782functions and methods that can be called is greatly reduced.
1783
1784=cut
761 1785
762our $safe = new Safe "safe"; 1786our $safe = new Safe "safe";
763our $safe_hole = new Safe::Hole; 1787our $safe_hole = new Safe::Hole;
764 1788
765$SIG{FPE} = 'IGNORE'; 1789$SIG{FPE} = 'IGNORE';
766 1790
767$safe->permit_only (Opcode::opset qw(:base_core :base_mem :base_orig :base_math sort time)); 1791$safe->permit_only (Opcode::opset qw(:base_core :base_mem :base_orig :base_math sort time));
768 1792
769# here we export the classes and methods available to script code 1793# here we export the classes and methods available to script code
770 1794
1795=pod
1796
1797The following fucntions and emthods are available within a safe environment:
1798
1799 cf::object contr pay_amount pay_player map
1800 cf::object::player player
1801 cf::player peaceful
1802 cf::map trigger
1803
1804=cut
1805
771for ( 1806for (
772 ["cf::object" => qw(contr pay_amount pay_player)], 1807 ["cf::object" => qw(contr pay_amount pay_player map)],
773 ["cf::object::player" => qw(player)], 1808 ["cf::object::player" => qw(player)],
774 ["cf::player" => qw(peaceful)], 1809 ["cf::player" => qw(peaceful)],
1810 ["cf::map" => qw(trigger)],
775) { 1811) {
776 no strict 'refs'; 1812 no strict 'refs';
777 my ($pkg, @funs) = @$_; 1813 my ($pkg, @funs) = @$_;
778 *{"safe::$pkg\::$_"} = $safe_hole->wrap (\&{"$pkg\::$_"}) 1814 *{"safe::$pkg\::$_"} = $safe_hole->wrap (\&{"$pkg\::$_"})
779 for @funs; 1815 for @funs;
780} 1816}
1817
1818=over 4
1819
1820=item @retval = safe_eval $code, [var => value, ...]
1821
1822Compiled and executes the given perl code snippet. additional var/value
1823pairs result in temporary local (my) scalar variables of the given name
1824that are available in the code snippet. Example:
1825
1826 my $five = safe_eval '$first + $second', first => 1, second => 4;
1827
1828=cut
781 1829
782sub safe_eval($;@) { 1830sub safe_eval($;@) {
783 my ($code, %vars) = @_; 1831 my ($code, %vars) = @_;
784 1832
785 my $qcode = $code; 1833 my $qcode = $code;
808 } 1856 }
809 1857
810 wantarray ? @res : $res[0] 1858 wantarray ? @res : $res[0]
811} 1859}
812 1860
1861=item cf::register_script_function $function => $cb
1862
1863Register a function that can be called from within map/npc scripts. The
1864function should be reasonably secure and should be put into a package name
1865like the extension.
1866
1867Example: register a function that gets called whenever a map script calls
1868C<rent::overview>, as used by the C<rent> extension.
1869
1870 cf::register_script_function "rent::overview" => sub {
1871 ...
1872 };
1873
1874=cut
1875
813sub register_script_function { 1876sub register_script_function {
814 my ($fun, $cb) = @_; 1877 my ($fun, $cb) = @_;
815 1878
816 no strict 'refs'; 1879 no strict 'refs';
817 *{"safe::$fun"} = $safe_hole->wrap ($cb); 1880 *{"safe::$fun"} = $safe_hole->wrap ($cb);
818} 1881}
1882
1883=back
1884
1885=cut
819 1886
820############################################################################# 1887#############################################################################
821 1888
822=head2 EXTENSION DATABASE SUPPORT 1889=head2 EXTENSION DATABASE SUPPORT
823 1890
858 1925
859Immediately write the database to disk I<if it is dirty>. 1926Immediately write the database to disk I<if it is dirty>.
860 1927
861=cut 1928=cut
862 1929
1930our $DB;
1931
863{ 1932{
864 my $db;
865 my $path = cf::datadir . "/database.pst"; 1933 my $path = cf::localdir . "/database.pst";
866 1934
867 sub db_load() { 1935 sub db_load() {
868 warn "loading database $path\n";#d# remove later
869 $db = stat $path ? Storable::retrieve $path : { }; 1936 $DB = stat $path ? Storable::retrieve $path : { };
870 } 1937 }
871 1938
872 my $pid; 1939 my $pid;
873 1940
874 sub db_save() { 1941 sub db_save() {
875 warn "saving database $path\n";#d# remove later
876 waitpid $pid, 0 if $pid; 1942 waitpid $pid, 0 if $pid;
877 unless ($pid = fork) { 1943 if (0 == ($pid = fork)) {
878 $db->{_meta}{version} = 1; 1944 $DB->{_meta}{version} = 1;
879 Storable::nstore $db, "$path~"; 1945 Storable::nstore $DB, "$path~";
880 rename "$path~", $path; 1946 rename "$path~", $path;
881 cf::_exit 0 if defined $pid; 1947 cf::_exit 0 if defined $pid;
882 } 1948 }
883 } 1949 }
884 1950
887 sub db_sync() { 1953 sub db_sync() {
888 db_save if $dirty; 1954 db_save if $dirty;
889 undef $dirty; 1955 undef $dirty;
890 } 1956 }
891 1957
892 my $idle = Event->idle (min => $TICK * 2.8, max => 10, repeat => 0, cb => sub { 1958 my $idle = Event->idle (min => $TICK * 2.8, max => 10, repeat => 0, data => WF_AUTOCANCEL, cb => sub {
893 db_sync; 1959 db_sync;
894 }); 1960 });
895 1961
896 sub db_dirty() { 1962 sub db_dirty() {
897 $dirty = 1; 1963 $dirty = 1;
898 $idle->start; 1964 $idle->start;
899 } 1965 }
900 1966
901 sub db_get($;$) { 1967 sub db_get($;$) {
902 @_ >= 2 1968 @_ >= 2
903 ? $db->{$_[0]}{$_[1]} 1969 ? $DB->{$_[0]}{$_[1]}
904 : ($db->{$_[0]} ||= { }) 1970 : ($DB->{$_[0]} ||= { })
905 } 1971 }
906 1972
907 sub db_put($$;$) { 1973 sub db_put($$;$) {
908 if (@_ >= 3) { 1974 if (@_ >= 3) {
909 $db->{$_[0]}{$_[1]} = $_[2]; 1975 $DB->{$_[0]}{$_[1]} = $_[2];
910 } else { 1976 } else {
911 $db->{$_[0]} = $_[1]; 1977 $DB->{$_[0]} = $_[1];
912 } 1978 }
913 db_dirty; 1979 db_dirty;
914 } 1980 }
1981
1982 cf::global->attach (
1983 prio => 10000,
1984 on_cleanup => sub {
1985 db_sync;
1986 },
1987 );
915} 1988}
916 1989
917############################################################################# 1990#############################################################################
918# the server's main() 1991# the server's main()
919 1992
1993sub cfg_load {
1994 open my $fh, "<:utf8", cf::confdir . "/config"
1995 or return;
1996
1997 local $/;
1998 *CFG = YAML::Syck::Load <$fh>;
1999
2000 $EMERGENCY_POSITION = $CFG{emergency_position} || ["/world/world_105_115", 5, 37];
2001
2002 $cf::map::MAX_RESET = $CFG{map_max_reset} if exists $CFG{map_max_reset};
2003 $cf::map::DEFAULT_RESET = $CFG{map_default_reset} if exists $CFG{map_default_reset};
2004
2005 if (exists $CFG{mlockall}) {
2006 eval {
2007 $CFG{mlockall} ? &mlockall : &munlockall
2008 and die "WARNING: m(un)lockall failed: $!\n";
2009 };
2010 warn $@ if $@;
2011 }
2012}
2013
920sub main { 2014sub main {
2015 # we must not ever block the main coroutine
2016 local $Coro::idle = sub {
2017 Carp::cluck "FATAL: Coro::idle was called, major BUG, use cf::sync_job!\n";#d#
2018 async { Event::one_event };
2019 };
2020
2021 cfg_load;
921 db_load; 2022 db_load;
922 load_extensions; 2023 load_extensions;
923 Event::loop; 2024 Event::loop;
924} 2025}
925 2026
926############################################################################# 2027#############################################################################
927# initialisation 2028# initialisation
928 2029
929sub _perl_reload(&) { 2030sub reload() {
930 my ($msg) = @_; 2031 # can/must only be called in main
2032 if ($Coro::current != $Coro::main) {
2033 warn "can only reload from main coroutine\n";
2034 return;
2035 }
931 2036
932 $msg->("reloading..."); 2037 warn "reloading...";
2038
2039 my $guard = freeze_mainloop;
2040 cf::emergency_save;
933 2041
934 eval { 2042 eval {
2043 # if anything goes wrong in here, we should simply crash as we already saved
2044
935 # cancel all watchers 2045 # cancel all watchers
936 $_->cancel for Event::all_watchers; 2046 for (Event::all_watchers) {
2047 $_->cancel if $_->data & WF_AUTOCANCEL;
2048 }
2049
2050 # cancel all extension coros
2051 $_->cancel for values %EXT_CORO;
2052 %EXT_CORO = ();
937 2053
938 # unload all extensions 2054 # unload all extensions
939 for (@exts) { 2055 for (@exts) {
940 $msg->("unloading <$_>"); 2056 warn "unloading <$_>";
941 unload_extension $_; 2057 unload_extension $_;
942 } 2058 }
943 2059
944 # unload all modules loaded from $LIBDIR 2060 # unload all modules loaded from $LIBDIR
945 while (my ($k, $v) = each %INC) { 2061 while (my ($k, $v) = each %INC) {
946 next unless $v =~ /^\Q$LIBDIR\E\/.*\.pm$/; 2062 next unless $v =~ /^\Q$LIBDIR\E\/.*\.pm$/;
947 2063
948 $msg->("removing <$k>"); 2064 warn "removing <$k>";
949 delete $INC{$k}; 2065 delete $INC{$k};
950 2066
951 $k =~ s/\.pm$//; 2067 $k =~ s/\.pm$//;
952 $k =~ s/\//::/g; 2068 $k =~ s/\//::/g;
953 2069
958 Symbol::delete_package $k; 2074 Symbol::delete_package $k;
959 } 2075 }
960 2076
961 # sync database to disk 2077 # sync database to disk
962 cf::db_sync; 2078 cf::db_sync;
2079 IO::AIO::flush;
963 2080
964 # get rid of safe::, as good as possible 2081 # get rid of safe::, as good as possible
965 Symbol::delete_package "safe::$_" 2082 Symbol::delete_package "safe::$_"
966 for qw(cf::object cf::object::player cf::player cf::map cf::party cf::region); 2083 for qw(cf::attachable cf::object cf::object::player cf::client cf::player cf::map cf::party cf::region);
967 2084
968 # remove register_script_function callbacks 2085 # remove register_script_function callbacks
969 # TODO 2086 # TODO
970 2087
971 # unload cf.pm "a bit" 2088 # unload cf.pm "a bit"
974 # don't, removes xs symbols, too, 2091 # don't, removes xs symbols, too,
975 # and global variables created in xs 2092 # and global variables created in xs
976 #Symbol::delete_package __PACKAGE__; 2093 #Symbol::delete_package __PACKAGE__;
977 2094
978 # reload cf.pm 2095 # reload cf.pm
979 $msg->("reloading cf.pm"); 2096 warn "reloading cf.pm";
980 require cf; 2097 require cf;
2098 cf::_connect_to_perl; # nominally unnecessary, but cannot hurt
981 2099
982 # load database again 2100 # load config and database again
2101 cf::cfg_load;
983 cf::db_load; 2102 cf::db_load;
984 2103
985 # load extensions 2104 # load extensions
986 $msg->("load extensions"); 2105 warn "load extensions";
987 cf::load_extensions; 2106 cf::load_extensions;
988 2107
989 # reattach attachments to objects 2108 # reattach attachments to objects
990 $msg->("reattach"); 2109 warn "reattach";
991 _global_reattach; 2110 _global_reattach;
992 }; 2111 };
993 $msg->($@) if $@;
994 2112
995 $msg->("reloaded"); 2113 if ($@) {
2114 warn $@;
2115 warn "error while reloading, exiting.";
2116 exit 1;
2117 }
2118
2119 warn "reloaded successfully";
996}; 2120};
997 2121
998sub perl_reload() { 2122#############################################################################
999 _perl_reload {
1000 warn $_[0];
1001 print "$_[0]\n";
1002 };
1003}
1004 2123
2124unless ($LINK_MAP) {
2125 $LINK_MAP = cf::map::new;
2126
2127 $LINK_MAP->width (41);
2128 $LINK_MAP->height (41);
2129 $LINK_MAP->alloc;
2130 $LINK_MAP->path ("{link}");
2131 $LINK_MAP->{path} = bless { path => "{link}" }, "cf::path";
2132 $LINK_MAP->in_memory (MAP_IN_MEMORY);
2133
2134 # dirty hack because... archetypes are not yet loaded
2135 Event->timer (
2136 after => 2,
2137 cb => sub {
2138 $_[0]->w->cancel;
2139
2140 # provide some exits "home"
2141 my $exit = cf::object::new "exit";
2142
2143 $exit->slaying ($EMERGENCY_POSITION->[0]);
2144 $exit->stats->hp ($EMERGENCY_POSITION->[1]);
2145 $exit->stats->sp ($EMERGENCY_POSITION->[2]);
2146
2147 $LINK_MAP->insert ($exit->clone, 19, 19);
2148 $LINK_MAP->insert ($exit->clone, 19, 20);
2149 $LINK_MAP->insert ($exit->clone, 19, 21);
2150 $LINK_MAP->insert ($exit->clone, 20, 19);
2151 $LINK_MAP->insert ($exit->clone, 20, 21);
2152 $LINK_MAP->insert ($exit->clone, 21, 19);
2153 $LINK_MAP->insert ($exit->clone, 21, 20);
2154 $LINK_MAP->insert ($exit->clone, 21, 21);
2155
2156 $exit->destroy;
2157 });
2158
2159 $LINK_MAP->{deny_save} = 1;
2160 $LINK_MAP->{deny_reset} = 1;
2161
2162 $cf::MAP{$LINK_MAP->path} = $LINK_MAP;
2163}
2164
2165register "<global>", __PACKAGE__;
2166
1005register_command "perl-reload", 0, sub { 2167register_command "reload" => sub {
1006 my ($who, $arg) = @_; 2168 my ($who, $arg) = @_;
1007 2169
1008 if ($who->flag (FLAG_WIZ)) { 2170 if ($who->flag (FLAG_WIZ)) {
1009 _perl_reload { 2171 $who->message ("start of reload.");
1010 warn $_[0]; 2172 reload;
1011 $who->message ($_[0]); 2173 $who->message ("end of reload.");
1012 };
1013 } 2174 }
1014}; 2175};
1015 2176
1016register "<global>", __PACKAGE__;
1017
1018unshift @INC, $LIBDIR; 2177unshift @INC, $LIBDIR;
1019 2178
1020$TICK_WATCHER = Event->timer ( 2179$TICK_WATCHER = Event->timer (
2180 reentrant => 0,
1021 prio => 1, 2181 prio => 0,
1022 at => $NEXT_TICK || 1, 2182 at => $NEXT_TICK || $TICK,
2183 data => WF_AUTOCANCEL,
1023 cb => sub { 2184 cb => sub {
1024 cf::server_tick; # one server iteration 2185 cf::server_tick; # one server iteration
1025 2186 $RUNTIME += $TICK;
1026 my $NOW = Event::time;
1027 $NEXT_TICK += $TICK; 2187 $NEXT_TICK += $TICK;
1028 2188
1029 # if we are delayed by four ticks, skip them all 2189 # if we are delayed by four ticks or more, skip them all
1030 $NEXT_TICK = $NOW if $NOW >= $NEXT_TICK + $TICK * 4; 2190 $NEXT_TICK = Event::time if Event::time >= $NEXT_TICK + $TICK * 4;
1031 2191
1032 $TICK_WATCHER->at ($NEXT_TICK); 2192 $TICK_WATCHER->at ($NEXT_TICK);
1033 $TICK_WATCHER->start; 2193 $TICK_WATCHER->start;
1034 }, 2194 },
1035); 2195);
1036 2196
2197IO::AIO::max_poll_time $TICK * 0.2;
2198
2199Event->io (
2200 fd => IO::AIO::poll_fileno,
2201 poll => 'r',
2202 prio => 5,
2203 data => WF_AUTOCANCEL,
2204 cb => \&IO::AIO::poll_cb,
2205);
2206
2207Event->timer (
2208 data => WF_AUTOCANCEL,
2209 after => 0,
2210 interval => 10,
2211 cb => sub {
2212 (Coro::unblock_sub {
2213 write_runtime
2214 or warn "ERROR: unable to write runtime file: $!";
2215 })->();
2216 },
2217);
2218
2219END { cf::emergency_save }
2220
10371 22211
1038 2222

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines