1 | #! perl # mandatory |
1 | #! perl # mandatory |
2 | |
2 | |
3 | use Coro::AIO; |
3 | use Coro::AIO; |
|
|
4 | |
|
|
5 | CONF META_TIMEOUT : map_random_meta_timeout = 86400 * 7; |
4 | |
6 | |
5 | cf::map->register (qr{^\?random/([0-9a-f]{32})}); |
7 | cf::map->register (qr{^\?random/([0-9a-f]{32})}); |
6 | |
8 | |
7 | sub init { |
9 | sub init { |
8 | my ($self) = @_; |
10 | my ($self) = @_; |
… | |
… | |
11 | |
13 | |
12 | if (0 < Coro::AIO::aio_load "$cf::RANDOMDIR/$self->{random_id}.meta", my $data) { |
14 | if (0 < Coro::AIO::aio_load "$cf::RANDOMDIR/$self->{random_id}.meta", my $data) { |
13 | $self->{random} = cf::decode_json $data; |
15 | $self->{random} = cf::decode_json $data; |
14 | $self->{random}{custom} ||= "$self->{random}{origin_map}+$self->{random}{origin_x}+$self->{random}{origin_y}"; |
16 | $self->{random}{custom} ||= "$self->{random}{origin_map}+$self->{random}{origin_x}+$self->{random}{origin_y}"; |
15 | } else { |
17 | } else { |
16 | warn "unable to read meta file for $self->{random_id}\n"; |
18 | cf::info "unable to read meta file for $self->{random_id}\n"; |
17 | return 0; |
19 | return 0; |
18 | } |
20 | } |
19 | |
21 | |
20 | 1 |
22 | 1 |
21 | } |
23 | } |
… | |
… | |
54 | $self->activate; |
56 | $self->activate; |
55 | |
57 | |
56 | 1 |
58 | 1 |
57 | } |
59 | } |
58 | |
60 | |
59 | sub clean_random_maps { |
61 | sub select_random_map { |
60 | my $files = Coro::AIO::aio_readdirx $cf::RANDOMDIR, IO::AIO::READDIR_STAT_ORDER |
62 | my ($maps, $difficulty) = @_; |
61 | or return; |
|
|
62 | |
63 | |
63 | my $META_TIMEOUT = $cf::CFG{map_random_meta_timeout} || 86400 * 7; |
64 | # because I am lazy, I ignore the weighting |
64 | |
65 | |
65 | for my $file (@$files) { |
66 | my @maps = keys %$maps; |
66 | next unless $file =~ /\.meta$/; |
|
|
67 | |
67 | |
68 | Coro::AIO::aio_stat "$cf::RANDOMDIR/$file" |
68 | cf::map::find "/styles/$maps[cf::rmg_rndm scalar @maps]" |
69 | and next; |
|
|
70 | |
|
|
71 | my $age = $cf::NOW - (stat _)[8]; |
|
|
72 | |
|
|
73 | if ($age > $META_TIMEOUT) { |
|
|
74 | warn "resetting random meta data for $file"; |
|
|
75 | IO::AIO::aio_unlink "$cf::RANDOMDIR/$file"; |
|
|
76 | } |
|
|
77 | } |
|
|
78 | } |
69 | } |
79 | |
70 | |
80 | # called by the random map generator |
71 | # called by the random map generator |
81 | sub find_style_; |
72 | sub find_style_; |
82 | sub find_style_($$) { |
73 | sub find_style_($$$) { |
83 | my ($path, $difficulty) = @_; |
74 | my ($path, $difficulty, $recurse) = @_; |
84 | |
75 | |
85 | my $map; |
76 | my $map; |
|
|
77 | |
|
|
78 | # see if there is a metafile |
|
|
79 | if (0 < aio_load "$cf::MAPDIR/$path.rmg", my $meta) { |
|
|
80 | $meta = cf::decode_json $meta; |
|
|
81 | |
|
|
82 | # only "maps" is supported |
|
|
83 | if ($meta->{maps}) { |
|
|
84 | $map = select_random_map $meta->{maps}, $difficulty; |
|
|
85 | } |
|
|
86 | } |
86 | |
87 | |
87 | $map = cf::map::find $path |
88 | $map = cf::map::find $path |
88 | unless aio_stat "$cf::MAPDIR/$path.map"; |
89 | unless aio_stat "$cf::MAPDIR/$path.map"; |
89 | |
90 | |
90 | unless ($map) { |
91 | unless ($map) { |
91 | # search files and/or dirs |
92 | # search files and/or dirs |
92 | if (my ($dirs, $nondirs) = aio_scandir "$cf::MAPDIR/$path/", 1) { |
93 | if (my ($dirs, $nondirs) = aio_scandir "$cf::MAPDIR/$path/", 1) { |
93 | my @entries = sort grep s/\.map$//, @$nondirs; |
94 | my @entries = sort grep s/\.(?:map|rmg)$//, @$nondirs; |
94 | |
95 | |
95 | if ($difficulty < 0) { |
96 | if ($difficulty < 0) { |
96 | # pick a fully random map, but only a map, do not recurse |
97 | # pick a fully random map, but only a map, do not recurse |
97 | $map = cf::map::find "$path/$entries[cf::rmg_rndm scalar @entries]" |
98 | $map = cf::map::find "$path/$entries[cf::rmg_rndm scalar @entries]" |
98 | if @entries; |
99 | if @entries; |
99 | } else { |
100 | } else { |
100 | # pick a map with nearest difficulty value ("mapname_<difficulty>.map") |
101 | # pick a map with nearest difficulty value ("mapname_<difficulty>.map") |
101 | @entries = sort @$dirs |
102 | @entries = sort @$dirs |
102 | unless @entries; |
103 | unless @entries || !$recurse; |
103 | |
104 | |
104 | my $min_diff = 1e99; |
105 | my $min_diff = 1e99; |
105 | |
106 | |
106 | for my $name (@entries) { |
107 | for my $name (@entries) { |
107 | if ($name =~ /_(\d+)$/) { |
108 | if ($name =~ /_(\d+)$/) { |
108 | my $diff = abs $difficulty - $1 + 0.5; # prefer the more difficult version |
109 | my $diff = abs $difficulty - $1 + 0.25 + 0.25 * cf::rmg_rndm; # prefer the more difficult version |
109 | ($map, $min_diff) = ($name, $diff) if $diff < $min_diff; |
110 | ($map, $min_diff) = ($name, $diff) if $diff < $min_diff; |
110 | } |
111 | } |
111 | } |
112 | } |
112 | |
113 | |
113 | unless ($map) { |
114 | unless ($map) { |
… | |
… | |
122 | } |
123 | } |
123 | |
124 | |
124 | $map |
125 | $map |
125 | } |
126 | } |
126 | |
127 | |
127 | sub find_style($$$) { |
128 | sub find_style($$$$) { |
128 | my ($dir, $name, $difficulty) = @_; |
129 | my ($dir, $name, $difficulty, $recurse) = @_; |
|
|
130 | |
|
|
131 | cf::cede_to_tick; |
|
|
132 | |
|
|
133 | my $map; |
129 | |
134 | |
130 | cf::cede_to_tick; |
135 | if ($name) { |
131 | |
|
|
132 | my $map = find_style_ $name ? "$dir/$name" : $dir, $difficulty; |
136 | $map = find_style_ "$dir/$name", $difficulty, $recurse; |
|
|
137 | } else { |
|
|
138 | $map = (find_style_ "$dir/default", $difficulty, $recurse) |
|
|
139 | || (find_style_ $dir, $difficulty, $recurse); |
|
|
140 | } |
133 | |
141 | |
134 | if ($map) { |
142 | if ($map) { |
135 | $map->load; |
143 | $map->load; |
136 | $map->deactivate; |
144 | $map->deactivate; |
137 | } |
145 | } |
138 | |
146 | |
139 | #warn "return $dir,$name,$difficulty => $map\n" if $difficulty >= 0;#d# |
147 | #warn "return $dir,$name,$difficulty => $map\n" if $difficulty >= 0;#d# |
140 | $map |
148 | $map |
141 | } |
149 | } |
142 | |
150 | |
143 | # clean up old temp maps regularly |
151 | cf::async_ext { |
144 | our $CLEAN_RANDOM_MAPS = cf::periodic 3600, Coro::unblock_sub { |
152 | local $Coro::current->{desc} = "random map meta file cleaner"; |
145 | clean_random_maps; |
153 | $Coro::current->nice (1); |
|
|
154 | |
|
|
155 | while () { |
|
|
156 | Coro::AnyEvent::idle_upto $META_TIMEOUT / 10 * 2; |
|
|
157 | |
|
|
158 | my ($files) = Coro::AIO::aio_readdirx $cf::RANDOMDIR, IO::AIO::READDIR_STAT_ORDER |
|
|
159 | or return; |
|
|
160 | |
|
|
161 | for my $file (@$files) { |
|
|
162 | next unless $file =~ /\.meta$/; |
|
|
163 | |
|
|
164 | Coro::AIO::aio_stat "$cf::RANDOMDIR/$file" |
|
|
165 | and next; |
|
|
166 | |
|
|
167 | my $age = $cf::NOW - (stat _)[8]; |
|
|
168 | |
|
|
169 | if ($age > $META_TIMEOUT) { |
|
|
170 | cf::trace "resetting random meta data for $file"; |
|
|
171 | IO::AIO::aio_unlink "$cf::RANDOMDIR/$file"; |
|
|
172 | } |
|
|
173 | } |
|
|
174 | |
|
|
175 | Coro::AnyEvent::sleep $META_TIMEOUT / 10; |
|
|
176 | } |
146 | }; |
177 | }; |
147 | |
178 | |
148 | # map generator stresstest, NEVER enable under normal circumstances |
179 | # map generator stresstest, NEVER enable under normal circumstances |
149 | if ($ENV{STRESSTEST}) { |
180 | if ($ENV{STRESSTEST}) { |
150 | cf::async { |
181 | cf::async { |
… | |
… | |
152 | while () { |
183 | while () { |
153 | my $map = cf::map::new; |
184 | my $map = cf::map::new; |
154 | $map->generate_random_map ({ |
185 | $map->generate_random_map ({ |
155 | region => "scorn", |
186 | region => "scorn", |
156 | random_seed => $seed++, |
187 | random_seed => $seed++, |
157 | xsize => (int rand 100) + 1, |
188 | xsize => (cf::rndm 1, 100), |
158 | ysize => (int rand 100) + 1, |
189 | ysize => (cf::rndm 1, 100), |
159 | }); |
190 | }); |
160 | warn sprintf "%d: %dx%d o# %d\n", $seed, $map->width, $map->height, &cf::object::objects_size;#d# |
191 | warn sprintf "%d: %dx%d o# %d\n", $seed, $map->width, $map->height, &cf::object::objects_size;#d# |
161 | $map->destroy; |
192 | $map->destroy; |
162 | } |
193 | } |
163 | }; |
194 | }; |
… | |
… | |
177 | } |
208 | } |
178 | |
209 | |
179 | # save test |
210 | # save test |
180 | if (0) { |
211 | if (0) { |
181 | cf::async { |
212 | cf::async { |
182 | # 0.58 |
213 | # 0.080 |
183 | Coro::Timer::sleep 2; |
214 | Coro::Timer::sleep 2; |
184 | my $map = cf::map::find "/mlab/citydeclouds2"; |
215 | my $map = cf::map::find "/mlab/citydeclouds2"; |
185 | $map->load_header; |
216 | $map->load_header; |
186 | $map->load; |
217 | $map->load; |
187 | $map->post_load_original; |
218 | $map->post_load_original; |
188 | my $m=100; |
219 | my $m=100; |
189 | for (1..50) { |
220 | for (1..50) { |
190 | my $t=EV::time; |
221 | my $t=AE::time; |
191 | $map->_save_objects ("/tmp/x", cf::IO_HEADER | cf::IO_OBJECTS | cf::IO_UNIQUES); |
222 | $map->_save_objects ("/tmp/x", cf::IO_HEADER | cf::IO_OBJECTS | cf::IO_UNIQUES); |
192 | $t = EV::time-$t; |
223 | $t = AE::time-$t; |
193 | $m=$t if $m>$t; |
224 | $m=$t if $m>$t; |
194 | warn $m; |
225 | warn $m; |
195 | } |
226 | } |
196 | }; |
227 | }; |
197 | } |
228 | } |
198 | |
229 | |
199 | 1 |
|
|
200 | |
|
|