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.69 by root, Mon Sep 18 01:10:35 2006 UTC vs.
Revision 1.77 by root, Sun Nov 5 11:13:01 2006 UTC

5use Storable; 5use Storable;
6use Opcode; 6use Opcode;
7use Safe; 7use Safe;
8use Safe::Hole; 8use Safe::Hole;
9 9
10use IO::AIO ();
11use YAML::Syck ();
10use Time::HiRes; 12use Time::HiRes;
11use Event; 13use Event;
12$Event::Eval = 1; # no idea why this is required, but it is 14$Event::Eval = 1; # no idea why this is required, but it is
13 15
16# work around bug in YAML::Syck - bad news for perl6, will it be as broken wrt. unicode?
17$YAML::Syck::ImplicitUnicode = 1;
18
14use strict; 19use strict;
15 20
16_init_vars; 21_init_vars;
17 22
18our %COMMAND = (); 23our %COMMAND = ();
20our $LIBDIR = maps_directory "perl"; 25our $LIBDIR = maps_directory "perl";
21 26
22our $TICK = MAX_TIME * 1e-6; 27our $TICK = MAX_TIME * 1e-6;
23our $TICK_WATCHER; 28our $TICK_WATCHER;
24our $NEXT_TICK; 29our $NEXT_TICK;
30
31our %CFG;
32
33our $uptime;
34
35$uptime ||= time;
36
37#############################################################################
38
39=head2 GLOBAL VARIABLES
40
41=over 4
42
43=item $cf::LIBDIR
44
45The perl library directory, where extensions and cf-specific modules can
46be found. It will be added to C<@INC> automatically.
47
48=item $cf::TICK
49
50The interval between server ticks, in seconds.
51
52=item %cf::CFG
53
54Configuration for the server, loaded from C</etc/crossfire/config>, or
55from wherever your confdir points to.
56
57=back
58
59=cut
25 60
26BEGIN { 61BEGIN {
27 *CORE::GLOBAL::warn = sub { 62 *CORE::GLOBAL::warn = sub {
28 my $msg = join "", @_; 63 my $msg = join "", @_;
29 $msg .= "\n" 64 $msg .= "\n"
51my @exts; 86my @exts;
52my @hook; 87my @hook;
53my %command; 88my %command;
54my %extcmd; 89my %extcmd;
55 90
56############################################################################# 91=head2 UTILITY FUNCTIONS
57# utility functions 92
93=over 4
94
95=cut
58 96
59use JSON::Syck (); # TODO# replace by JSON::PC once working 97use JSON::Syck (); # TODO# replace by JSON::PC once working
98
99=item $ref = cf::from_json $json
100
101Converts a JSON string into the corresponding perl data structure.
102
103=cut
60 104
61sub from_json($) { 105sub from_json($) {
62 $JSON::Syck::ImplicitUnicode = 1; # work around JSON::Syck bugs 106 $JSON::Syck::ImplicitUnicode = 1; # work around JSON::Syck bugs
63 JSON::Syck::Load $_[0] 107 JSON::Syck::Load $_[0]
64} 108}
65 109
110=item $json = cf::to_json $ref
111
112Converts a perl data structure into its JSON representation.
113
114=cut
115
66sub to_json($) { 116sub to_json($) {
67 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs 117 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs
68 JSON::Syck::Dump $_[0] 118 JSON::Syck::Dump $_[0]
69} 119}
70 120
71############################################################################# 121=back
72# "new" plug-in system
73 122
123=cut
124
125#############################################################################
126
74=head3 EVENTS AND OBJECT ATTACHMENTS 127=head2 EVENTS AND OBJECT ATTACHMENTS
75 128
76=over 4 129=over 4
77 130
78=item $object->attach ($attachment, key => $value...) 131=item $object->attach ($attachment, key => $value...)
79 132
379removed in future versions), and there is no public API to access override 432removed in future versions), and there is no public API to access override
380results (if you must, access C<@cf::invoke_results> directly). 433results (if you must, access C<@cf::invoke_results> directly).
381 434
382=back 435=back
383 436
384=head2 methods valid for all pointers 437=cut
438
439#############################################################################
440
441=head2 METHODS VALID FOR ALL CORE OBJECTS
385 442
386=over 4 443=over 4
387 444
388=item $object->valid 445=item $object->valid, $player->valid, $map->valid
389
390=item $player->valid
391
392=item $map->valid
393 446
394Just because you have a perl object does not mean that the corresponding 447Just 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 448C-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 449valid 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 450can be used to test for existence of the C object part without causing an
704 } 757 }
705 }, 758 },
706; 759;
707 760
708############################################################################# 761#############################################################################
709# core extensions - in perl 762
763=head2 CORE EXTENSIONS
764
765Functions and methods that extend core crossfire objects.
766
767=over 4
710 768
711=item cf::player::exists $login 769=item cf::player::exists $login
712 770
713Returns true when the given account exists. 771Returns true when the given account exists.
714 772
717sub cf::player::exists($) { 775sub cf::player::exists($) {
718 cf::player::find $_[0] 776 cf::player::find $_[0]
719 or -f sprintf "%s/%s/%s/%s.pl", cf::localdir, cf::playerdir, ($_[0]) x 2; 777 or -f sprintf "%s/%s/%s/%s.pl", cf::localdir, cf::playerdir, ($_[0]) x 2;
720} 778}
721 779
722=item $player->reply ($npc, $msg[, $flags]) 780=item $object->reply ($npc, $msg[, $flags])
723 781
724Sends a message to the player, as if the npc C<$npc> replied. C<$npc> 782Sends 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 783can be C<undef>. Does the right thing when the player is currently in a
726dialogue with the given NPC character. 784dialogue with the given NPC character.
727 785
754 $msg{msgid} = $id; 812 $msg{msgid} = $id;
755 813
756 $self->send ("ext " . to_json \%msg); 814 $self->send ("ext " . to_json \%msg);
757} 815}
758 816
817=back
818
819=cut
820
759############################################################################# 821#############################################################################
760# map scripting support 822
823=head2 SAFE SCRIPTING
824
825Functions that provide a safe environment to compile and execute
826snippets of perl code without them endangering the safety of the server
827itself. Looping constructs, I/O operators and other built-in functionality
828is not available in the safe scripting environment, and the number of
829functions and methods that cna be called is greatly reduced.
830
831=cut
761 832
762our $safe = new Safe "safe"; 833our $safe = new Safe "safe";
763our $safe_hole = new Safe::Hole; 834our $safe_hole = new Safe::Hole;
764 835
765$SIG{FPE} = 'IGNORE'; 836$SIG{FPE} = 'IGNORE';
766 837
767$safe->permit_only (Opcode::opset qw(:base_core :base_mem :base_orig :base_math sort time)); 838$safe->permit_only (Opcode::opset qw(:base_core :base_mem :base_orig :base_math sort time));
768 839
769# here we export the classes and methods available to script code 840# here we export the classes and methods available to script code
841
842=pod
843
844The following fucntions and emthods are available within a safe environment:
845
846 cf::object contr pay_amount pay_player
847 cf::object::player player
848 cf::player peaceful
849
850=cut
770 851
771for ( 852for (
772 ["cf::object" => qw(contr pay_amount pay_player)], 853 ["cf::object" => qw(contr pay_amount pay_player)],
773 ["cf::object::player" => qw(player)], 854 ["cf::object::player" => qw(player)],
774 ["cf::player" => qw(peaceful)], 855 ["cf::player" => qw(peaceful)],
777 my ($pkg, @funs) = @$_; 858 my ($pkg, @funs) = @$_;
778 *{"safe::$pkg\::$_"} = $safe_hole->wrap (\&{"$pkg\::$_"}) 859 *{"safe::$pkg\::$_"} = $safe_hole->wrap (\&{"$pkg\::$_"})
779 for @funs; 860 for @funs;
780} 861}
781 862
863=over 4
864
865=item @retval = safe_eval $code, [var => value, ...]
866
867Compiled and executes the given perl code snippet. additional var/value
868pairs result in temporary local (my) scalar variables of the given name
869that are available in the code snippet. Example:
870
871 my $five = safe_eval '$first + $second', first => 1, second => 4;
872
873=cut
874
782sub safe_eval($;@) { 875sub safe_eval($;@) {
783 my ($code, %vars) = @_; 876 my ($code, %vars) = @_;
784 877
785 my $qcode = $code; 878 my $qcode = $code;
786 $qcode =~ s/"/‟/g; # not allowed in #line filenames 879 $qcode =~ s/"/‟/g; # not allowed in #line filenames
829 my ($fun, $cb) = @_; 922 my ($fun, $cb) = @_;
830 923
831 no strict 'refs'; 924 no strict 'refs';
832 *{"safe::$fun"} = $safe_hole->wrap ($cb); 925 *{"safe::$fun"} = $safe_hole->wrap ($cb);
833} 926}
927
928=back
929
930=cut
834 931
835############################################################################# 932#############################################################################
836 933
837=head2 EXTENSION DATABASE SUPPORT 934=head2 EXTENSION DATABASE SUPPORT
838 935
937} 1034}
938 1035
939############################################################################# 1036#############################################################################
940# the server's main() 1037# the server's main()
941 1038
1039sub cfg_load {
1040 open my $fh, "<:utf8", cf::confdir . "/config"
1041 or return;
1042
1043 local $/;
1044 *CFG = YAML::Syck::Load <$fh>;
1045}
1046
942sub main { 1047sub main {
1048 cfg_load;
943 db_load; 1049 db_load;
944 load_extensions; 1050 load_extensions;
945 Event::loop; 1051 Event::loop;
946} 1052}
947 1053
999 1105
1000 # reload cf.pm 1106 # reload cf.pm
1001 $msg->("reloading cf.pm"); 1107 $msg->("reloading cf.pm");
1002 require cf; 1108 require cf;
1003 1109
1004 # load database again 1110 # load config and database again
1111 cf::cfg_load;
1005 cf::db_load; 1112 cf::db_load;
1006 1113
1007 # load extensions 1114 # load extensions
1008 $msg->("load extensions"); 1115 $msg->("load extensions");
1009 cf::load_extensions; 1116 cf::load_extensions;
1038register "<global>", __PACKAGE__; 1145register "<global>", __PACKAGE__;
1039 1146
1040unshift @INC, $LIBDIR; 1147unshift @INC, $LIBDIR;
1041 1148
1042$TICK_WATCHER = Event->timer ( 1149$TICK_WATCHER = Event->timer (
1043 prio => 1, 1150 prio => 1,
1151 async => 1,
1044 at => $NEXT_TICK || 1, 1152 at => $NEXT_TICK || 1,
1045 cb => sub { 1153 cb => sub {
1046 cf::server_tick; # one server iteration 1154 cf::server_tick; # one server iteration
1047 1155
1048 my $NOW = Event::time; 1156 my $NOW = Event::time;
1049 $NEXT_TICK += $TICK; 1157 $NEXT_TICK += $TICK;
1050 1158
1054 $TICK_WATCHER->at ($NEXT_TICK); 1162 $TICK_WATCHER->at ($NEXT_TICK);
1055 $TICK_WATCHER->start; 1163 $TICK_WATCHER->start;
1056 }, 1164 },
1057); 1165);
1058 1166
1167IO::AIO::max_poll_time $TICK * 0.2;
1168
1169Event->io (fd => IO::AIO::poll_fileno,
1170 poll => 'r',
1171 prio => 5,
1172 cb => \&IO::AIO::poll_cb);
1173
10591 11741
1060 1175

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines