ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/DC.pm
Revision: 1.139
Committed: Sun Apr 1 15:08:14 2007 UTC (17 years, 1 month ago) by root
Branch: MAIN
Changes since 1.138: +2 -2 lines
Log Message:
*** empty log message ***

File Contents

# User Rev Content
1 root 1.1 =head1 NAME
2    
3 root 1.108 CFPlus - undocumented utility garbage for our crossfire client
4 root 1.1
5     =head1 SYNOPSIS
6    
7 root 1.108 use CFPlus;
8 root 1.1
9     =head1 DESCRIPTION
10    
11     =over 4
12    
13     =cut
14    
15 root 1.108 package CFPlus;
16 root 1.1
17 root 1.121 use Carp ();
18    
19 root 1.1 BEGIN {
20 root 1.130 $VERSION = '0.97';
21 root 1.1
22 root 1.2 use XSLoader;
23 root 1.108 XSLoader::load "CFPlus", $VERSION;
24 root 1.1 }
25    
26 root 1.62 use utf8;
27    
28 root 1.52 use AnyEvent ();
29 root 1.34 use BerkeleyDB;
30 root 1.89 use Pod::POM ();
31 root 1.135 use File::Path ();
32 root 1.89 use Storable (); # finally
33    
34 root 1.127 BEGIN {
35     use Crossfire::Protocol::Base ();
36     *to_json = \&Crossfire::Protocol::Base::to_json;
37     *from_json = \&Crossfire::Protocol::Base::from_json;
38     }
39    
40 root 1.103 =item guard { BLOCK }
41    
42     Returns an object that executes the given block as soon as it is destroyed.
43    
44     =cut
45    
46     sub guard(&) {
47 root 1.108 bless \(my $cb = $_[0]), "CFPlus::Guard"
48 root 1.103 }
49    
50 root 1.108 sub CFPlus::Guard::DESTROY {
51 root 1.103 ${$_[0]}->()
52     }
53    
54 root 1.133 =item shorten $string[, $maxlength]
55    
56     =cut
57    
58     sub shorten($;$) {
59     my ($str, $len) = @_;
60     substr $str, $len, (length $str), "..." if $len + 3 <= length $str;
61     $str
62     }
63    
64 root 1.105 sub asxml($) {
65     local $_ = $_[0];
66 root 1.89
67 root 1.105 s/&/&amp;/g;
68     s/>/&gt;/g;
69     s/</&lt;/g;
70 root 1.89
71 root 1.105 $_
72 root 1.89 }
73    
74 root 1.123 sub socketpipe() {
75     socketpair my $fh1, my $fh2, Socket::AF_UNIX, Socket::SOCK_STREAM, Socket::PF_UNSPEC
76     or die "cannot establish bidiretcional pipe: $!\n";
77    
78     ($fh1, $fh2)
79     }
80    
81 root 1.127 sub background(&;&) {
82     my ($bg, $cb) = @_;
83 root 1.123
84     my ($fh_r, $fh_w) = CFPlus::socketpipe;
85    
86     my $pid = fork;
87    
88     if (defined $pid && !$pid) {
89 root 1.124 local $SIG{__DIE__};
90 root 1.123
91     open STDOUT, ">&", $fh_w;
92     open STDERR, ">&", $fh_w;
93     close $fh_r;
94     close $fh_w;
95    
96     $| = 1;
97    
98 root 1.127 eval { $bg->() };
99 root 1.124
100     if ($@) {
101     my $msg = $@;
102     $msg =~ s/\n+/\n/;
103     warn "FATAL: $msg";
104     CFPlus::_exit 1;
105     }
106 root 1.123
107     # win32 is fucked up, of course. exit will clean stuff up,
108     # which destroys our database etc. _exit will exit ALL
109     # forked processes, because of the dreaded fork emulation.
110 root 1.124 CFPlus::_exit 0;
111 root 1.123 }
112    
113     close $fh_w;
114    
115     my $buffer;
116    
117 root 1.126 my $w; $w = AnyEvent->io (fh => $fh_r, poll => 'r', cb => sub {
118 root 1.123 unless (sysread $fh_r, $buffer, 4096, length $buffer) {
119 root 1.126 undef $w;
120 root 1.127 $cb->();
121     return;
122 root 1.123 }
123    
124     while ($buffer =~ s/^(.*)\n//) {
125     my $line = $1;
126 root 1.124 $line =~ s/\s+$//;
127 root 1.123 utf8::decode $line;
128 root 1.127 if ($line =~ /^\x{e877}json_msg (.*)$/s) {
129 root 1.139 $cb->(JSON::XS->new->allow_nonref->decode ($1));
130 root 1.127 } else {
131     ::message ({
132     markup => "background($pid): " . CFPlus::asxml $line,
133     });
134     }
135 root 1.123 }
136     });
137     }
138    
139 root 1.127 sub background_msg {
140     my ($msg) = @_;
141    
142 root 1.139 $msg = "\x{e877}json_msg " . JSON::XS->new->allow_nonref->encode ($msg);
143 root 1.127 $msg =~ s/\n//g;
144     utf8::encode $msg;
145     print $msg, "\n";
146     }
147    
148 root 1.108 package CFPlus::Database;
149 root 1.89
150     our @ISA = BerkeleyDB::Btree::;
151    
152     sub get($$) {
153     my $data;
154    
155     $_[0]->db_get ($_[1], $data) == 0
156     ? $data
157     : ()
158     }
159    
160     my %DB_SYNC;
161    
162     sub put($$$) {
163     my ($db, $key, $data) = @_;
164    
165 root 1.117 my $hkey = $db + 0;
166 root 1.137 CFPlus::weaken $db;
167 root 1.138 $DB_SYNC{$hkey} ||= AnyEvent->timer (after => 30, cb => sub {
168 root 1.117 delete $DB_SYNC{$hkey};
169 root 1.116 $db->db_sync if $db;
170     });
171 root 1.89
172     $db->db_put ($key => $data)
173     }
174    
175 root 1.108 package CFPlus;
176 root 1.52
177 root 1.5 sub find_rcfile($) {
178     my $path;
179    
180 root 1.46 for (grep !ref, @INC) {
181 root 1.108 $path = "$_/CFPlus/resources/$_[0]";
182 root 1.5 return $path if -r $path;
183     }
184    
185     die "FATAL: can't find required file $_[0]\n";
186     }
187    
188     sub read_cfg {
189     my ($file) = @_;
190    
191 root 1.107 open my $fh, $file
192 root 1.5 or return;
193    
194     local $/;
195 root 1.107 my $CFG = <$fh>;
196 root 1.5
197 root 1.108 if ($CFG =~ /^---/) { ## TODO compatibility cruft, remove
198     require YAML;
199     utf8::decode $CFG;
200     $::CFG = YAML::Load ($CFG);
201     } elsif ($CFG =~ /^\{/) {
202     $::CFG = from_json $CFG;
203 root 1.107 } else {
204 root 1.108 $::CFG = eval $CFG; ## todo comaptibility cruft
205 root 1.107 }
206 root 1.5 }
207    
208     sub write_cfg {
209     my ($file) = @_;
210    
211 root 1.107 $::CFG->{VERSION} = $::VERSION;
212    
213     open my $fh, ">:utf8", $file
214 root 1.5 or return;
215 root 1.108 print $fh to_json $::CFG;
216 root 1.5 }
217    
218 root 1.122 sub http_proxy {
219     my @proxy = win32_proxy_info;
220    
221     if (@proxy) {
222     "http://" . (@proxy < 2 ? "" : @proxy < 3 ? "$proxy[1]\@" : "$proxy[1]:$proxy[2]\@") . $proxy[0]
223     } elsif (exists $ENV{http_proxy}) {
224     $ENV{http_proxy}
225     } else {
226     ()
227     }
228     }
229    
230     sub set_proxy {
231     my $proxy = http_proxy
232     or return;
233    
234     $ENV{http_proxy} = $proxy;
235     }
236    
237 root 1.127 sub lwp_useragent {
238     require LWP::UserAgent;
239    
240     CFPlus::set_proxy;
241    
242     my $ua = LWP::UserAgent->new (
243     agent => "cfplus $VERSION",
244     keep_alive => 1,
245     env_proxy => 1,
246     timeout => 30,
247     );
248     }
249    
250     sub lwp_check($) {
251     my ($res) = @_;
252    
253     $res->is_error
254     and die $res->status_line;
255    
256     $res
257     }
258    
259 root 1.77 our $DB_ENV;
260 root 1.121 our $DB_STATE;
261    
262     sub db_table($) {
263     my ($table) = @_;
264    
265     $table =~ s/([^a-zA-Z0-9_\-])/sprintf "=%x=", ord $1/ge;
266    
267     new CFPlus::Database
268     -Env => $DB_ENV,
269     -Filename => $table,
270     # -Filename => "database",
271     # -Subname => $table,
272     -Property => DB_CHKSUM,
273     -Flags => DB_CREATE | DB_UPGRADE,
274     or die "unable to create/open database table $_[0]: $BerkeleyDB::Error"
275     }
276 root 1.77
277 root 1.135 our $DB_HOME = "$Crossfire::VARDIR/cfplus";
278    
279     sub open_db {
280 root 1.76 use strict;
281    
282 root 1.135 mkdir $DB_HOME, 0777;
283 root 1.77 my $recover = $BerkeleyDB::db_version >= 4.4
284     ? eval "DB_REGISTER | DB_RECOVER"
285     : 0;
286    
287     $DB_ENV = new BerkeleyDB::Env
288 root 1.135 -Home => $DB_HOME,
289 root 1.138 -Cachesize => 8_000_000,
290 root 1.135 -ErrFile => "$DB_HOME/errorlog.txt",
291 root 1.39 # -ErrPrefix => "DATABASE",
292 root 1.76 -Verbose => 1,
293 root 1.77 -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN | $recover,
294 root 1.78 -SetFlags => DB_AUTO_COMMIT | DB_LOG_AUTOREMOVE,
295 root 1.135 or die "unable to create/open database home $DB_HOME: $BerkeleyDB::Error";
296 root 1.76
297 root 1.121 $DB_STATE = db_table "state";
298 root 1.135
299     1
300     }
301    
302     unless (eval { open_db }) {
303     File::Path::rmtree $DB_HOME;
304     open_db;
305 root 1.34 }
306    
307 root 1.108 package CFPlus::Layout;
308 root 1.97
309 root 1.108 $CFPlus::OpenGL::SHUTDOWN_HOOK{"CFPlus::Layout"} = sub {
310 root 1.98 reset_glyph_cache;
311 root 1.97 };
312    
313 root 1.1 1;
314    
315     =back
316    
317     =head1 AUTHOR
318    
319     Marc Lehmann <schmorp@schmorp.de>
320     http://home.schmorp.de/
321    
322     =cut
323