aboutsummaryrefslogtreecommitdiff
path: root/assets/manual.lyx
blob: 3255f682434dd73ed03444589ef5cd6299a5bbb0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
#LyX 2.2 created this file. For more info see http://www.lyx.org/
\lyxformat 508
\begin_document
\begin_header
\save_transient_properties true
\origin unavailable
\textclass paper
\use_default_options true
\maintain_unincluded_children false
\language english
\language_package default
\inputencoding auto
\fontencoding global
\font_roman "default" "default"
\font_sans "default" "default"
\font_typewriter "default" "default"
\font_math "auto" "auto"
\font_default_family default
\use_non_tex_fonts false
\font_sc false
\font_osf false
\font_sf_scale 100 100
\font_tt_scale 100 100
\graphics default
\default_output_format default
\output_sync 0
\bibtex_command default
\index_command default
\paperfontsize default
\spacing single
\use_hyperref false
\papersize default
\use_geometry true
\use_package amsmath 1
\use_package amssymb 1
\use_package cancel 1
\use_package esint 1
\use_package mathdots 1
\use_package mathtools 1
\use_package mhchem 1
\use_package stackrel 1
\use_package stmaryrd 1
\use_package undertilde 1
\cite_engine basic
\cite_engine_type default
\biblio_style plain
\use_bibtopic false
\use_indices false
\paperorientation portrait
\suppress_date false
\justification true
\use_refstyle 1
\index Index
\shortcut idx
\color #008000
\end_index
\leftmargin 1cm
\topmargin 1cm
\rightmargin 1cm
\bottommargin 1cm
\secnumdepth 3
\tocdepth 3
\paragraph_separation indent
\paragraph_indentation default
\quotes_language english
\papercolumns 1
\papersides 1
\paperpagestyle default
\tracking_changes false
\output_changes false
\html_math_output 0
\html_css_as_file 0
\html_be_strict false
\end_header

\begin_body

\begin_layout Part*
Advanced Trains 
\end_layout

\begin_layout Standard
This mod aims to provide realistic, good-looking and functional trains by
 introducing a revolutionary rail placement system.
 It features several wagons that can be coupled together.
\end_layout

\begin_layout Standard
This mod is not finished.
 If you miss features, suggest them, but do not denounce this mod just because
 they are not yet implemented.
 They will be.
\end_layout

\begin_layout Subsection*
Placing Rails
\end_layout

\begin_layout Standard
Minetest's in-house rail system features rails that turn at an angle of
 90 degrees – totally impractical for the use with realistic trains.
 So we have our own rails.
 Remember: Carts can't drive on the rails provided by this mod, as do trains
 not drive on minetest's default rails because of their different track
 widths.
\end_layout

\begin_layout Standard
First, craft some rails.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-43-29.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
Now, place one at any position and another one right next to it: you have
 made your first railway track!
\end_layout

\begin_layout Standard
To learn how to make turns have a look at the following examples.
 A rail node has been placed only at the red-marked places.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-04-12.png
	width 5cm

\end_inset


\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-04-57.png
	width 4cm

\end_inset


\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-05-51.png
	width 5cm

\end_inset


\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-07-13.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
As shown in the illustrations above, the 30-degree angled rails use a knight's
 move (2 ahead, 1 aside) for placement.
 For the rails to look realistic, I encourage you not to build turns that
 are too narrow.
 IMO the angles you can build with this are still way to narrow, but this
 is the best compromise I can find.
 
\end_layout

\begin_layout Subsection*
Switches
\end_layout

\begin_layout Standard
To create switches we need the trackworker tool.
 ATM it looks like a Doctor Who Sonic Screwdriver.
 Aside from turning rails into switches, it is also capable of rotating
 everything (rails, bumpers, signals) in this mod.
 Due to internal mechanics, nothing can be rotated using the default screwdriver.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-56-34.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
Place some rails.
 Then left-click 1-2 times on one of these rails, until you see a switch.
 Use right-click to rotate it how you need it.
 You can change the switch direction by right-clicking the switch or by
 powering it with mesecons.
\end_layout

\begin_layout Standard
Unfortunately tracks that are placed next to switches don't always automatically
 connect to them.
 You need to correct manually using the Trackworker.
 One day I will implement proper handling for these.
 When you are finished it could look like this:
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-08-02.png
	width 5cm

\end_inset


\end_layout

\begin_layout Subsection*
Rail crosses
\end_layout

\begin_layout Standard
There are no real cross-rail nodes.
 However you can create crossing rails by being creative and using the knight's
 move or by placing opposing 45-degree rails.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-09-01.png
	width 5cm

\end_inset


\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_10-10-15.png
	width 5cm

\end_inset


\end_layout

\begin_layout Subsection*
Height differences 
\end_layout

\begin_layout Standard
To master height differences you can craft slope nodes:
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-45-38.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
To place them, you have to prepare the base, then stand in the right direction
 and point to the slope start point, then place it.
 A slope will be constructed in the direction you are facing (45 degree
 steps) leaned against the next solid node.
 The right number of slopes is subtracted from the item stack if you are
 in survival.
\end_layout

\begin_layout Subsection*
Bumpers, platforms, signals and detector rails
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-48-54.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
Bumpers are objects that are usually placed at the end of a track to prevent
 trains rolling off it.
 After placed, they can be rotated using the Trackworker.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-50-27.png
	width 5cm

\end_inset


\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-51-02.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
These are a regular analog signal and an electric signal.
 Like everything, you can rotate them using the Trackworker.
 Right-click or power with mesecons to signal trains that they can pass
 or have to stop.
 The signals do not have any effect on trains, they can only signal the
 driver.
 A more advanced signalling system (with distant signals/signal combinations)
 is planned.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-58-39.png
	width 5cm

\end_inset


\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2016-09-17_09-58-20.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
These are some platform nodes.
 I suggest using the left one, it's only half height and looks better.
 These nodes also have a sandstone variant, craft with sandstone bricks
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2017-03-09_11-33-09.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
These detector rails turn adjacent mesecons on when a train is standing/driving
 over them.
\end_layout

\begin_layout Standard
Notice: Detector rails and bumpers currently aren't aligned to the regular
 tracks.
 This will be fixed soon.
 Meanwhile, you need to rotate them manually.
\end_layout

\begin_layout Subsection*
Trains
\end_layout

\begin_layout Standard
There are some wagons included in this modpack, however community members
 (namely mbb and Andrey) have made some more wagons that can be downloaded
 and enabled separately.
 Visit the forum topic (
\begin_inset Flex URL
status collapsed

\begin_layout Plain Layout

https://forum.minetest.net/viewtopic.php?f=11&t=14726
\end_layout

\end_inset

) to download them.
\end_layout

\begin_layout Standard
To see what's included, look up in a craft guide or consult the creative
 mode inventory.
\end_layout

\begin_layout Standard
To place wagons simply craft and click a track.
 To remove a wagon, punch it.
 Only the person who placed the wagon can do this.
 In survival if you destroy trains you get only some of your steel back,
 so you will be asked to confirm if you really want to destroy a wagon.
\end_layout

\begin_layout Subsection*
Driving trains
\end_layout

\begin_layout Standard
Right-click any wagon to get on.
 This will attach you to the wagon and register you as passenger.
 Depending on how the wagon is set up, you are either in a passenger seat
 or inside a driver stand.
 Right-clicking again will show your possibilities on what you can do in/with
 the wagon.
\end_layout

\begin_layout Standard
Example:
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/Bildschirmfoto_2017-03-09_11-42-49.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
When entering a subway wagon, you are formally inside the passenger area.
 You can see this by the fact that there's no head-up display.
 Right-clicking brings up this form.
\end_layout

\begin_layout Standard
The first button will make you move to the Driver stand, so you can drive
 the train.
\end_layout

\begin_layout Standard
The second button should say 
\begin_inset Quotes eld
\end_inset

Wagon properties
\begin_inset Quotes erd
\end_inset

 and appears only for the wagon owner.
 See 
\begin_inset Quotes eld
\end_inset

Wagon Properties
\begin_inset Quotes erd
\end_inset

.
\end_layout

\begin_layout Standard
The last button tells that the doors are closed, so you can't get off at
 this time.
 If the doors are open or the wagon has no doors, this button says 
\begin_inset Quotes eld
\end_inset

Get off
\begin_inset Quotes erd
\end_inset

.
\end_layout

\begin_layout Standard
It is always possible to bypass closed doors and get off by holding the
 Sneak key and right-clicking the wagon or by holding Sneak and Use at the
 same time.
 Remember that this may result in your death when the train is travelling
 fast.
\end_layout

\begin_layout Standard
The Japanese train and the Subway train support automatic getting on by
 just walking into the wagon.
 As soon as you stand on a platform and walk towards a door, you will automatica
lly get on the wagon.
 On these, pressing W or S while inside the Passenger Area will also make
 you get off.
\end_layout

\begin_layout Subsection*
Train controls
\end_layout

\begin_layout Standard
If you are inside a driver stand you are presented with a head-up display:
\end_layout

\begin_layout Standard
The upper bar shows your current speed and the lower bar shows what speed
 you ordered the train to hold.
 Assuming you have the default controls (WASD, Shift for sneak, Space for
 jump), the following key bindings apply: 
\end_layout

\begin_layout Itemize
W - faster
\end_layout

\begin_layout Itemize
S - slower / change direction
\end_layout

\begin_layout Itemize
A / D – open/close doors
\end_layout

\begin_layout Itemize
Space: brake (shown by =B=, target speed will be decreased automatically)
\end_layout

\begin_layout Itemize
Sneak+S: set speed to 0 (train rolls out, brake to stop!)
\end_layout

\begin_layout Itemize
Sneak+W: Set full speed
\end_layout

\begin_layout Itemize
Sneak+A: Set speed to 4 (~40km/h)
\end_layout

\begin_layout Itemize
Sneak+D: Set speed to 8 (~100km/h)
\end_layout

\begin_layout Itemize
Sneak+Space: toggle brake (the brake will not release when releasing the
 keys, shown by =^B=)
\end_layout

\begin_layout Subsection*
Coupling wagons
\end_layout

\begin_layout Standard
You just learned how to drive an engine.
 Now place a wagon anywhere and drive your engine slowly towards that wagon.
 As soon as they collided your engine will stop.
 Now get off and right-click the green icon that appeared between the engine
 and the train.
 You have coupled the wagon to the engine.
\end_layout

\begin_layout Standard
\begin_inset Graphics
	filename manual_img/screenshot_20161203_231622.png
	width 5cm

\end_inset


\end_layout

\begin_layout Standard
To discouple a wagon, punch the red icon between the wagons you want to
 discouple while the train is standing.
\end_layout

\begin_layout Subsection*
Automatic Train Control (ATC)
\end_layout

\begin_layout Standard
ATC rails allow you to automate train operation.
 There are two types of ATC rails:
\end_layout

\begin_layout Subsubsection*
Regular ATC
\end_layout

\begin_layout Standard
The ATC rail does not have a crafting recipe.
 When placed, you can set a command and it will be sent to any train driving
 over the controller.
 
\end_layout

\begin_layout Standard
Only the static mode is implemented, changing the mode has no effect.
\end_layout

\begin_layout Standard
For a detailed explanation how ATC commands work and their syntax see atc_comman
d.txt
\end_layout

\begin_layout Standard
Note: to rotate ATC rails, you need to bypass the formspec that is set for
 the node.
 To do this, hold Sneak when right-clicking the rail with the trackworker
 tool.
\end_layout

\begin_layout Subsubsection*
LUA ATC
\end_layout

\begin_layout Standard
The LUA ATC suite is part of the mod advtrains_luaautomation.
 The LUA ATC components are quite similar to Mesecons Luacontrollers and
 allow to create all kinds of automation systems.
 This tool is not intended for beginners or regular players, but for server
 admins who wish to create a heavily automated subway system.
\end_layout

\begin_layout Standard
More information on those can be found inside the mod directory of advtrains_lua
automation.
\end_layout

\end_body
\end_document
l kwa">return; } // length of sha1 must be exactly 20 (160 bits), else ignore the file if (sha1.size() != 20) { errorstream << "Client: ignoring illegal SHA1 sent by server: " << hex_encode(sha1) << " \"" << name << "\"" << std::endl; return; } FileStatus *filestatus = new FileStatus(); filestatus->received = false; filestatus->sha1 = sha1; filestatus->current_remote = -1; m_files.insert(std::make_pair(name, filestatus)); } void ClientMediaDownloader::addRemoteServer(const std::string &baseurl) { assert(!m_initial_step_done); // pre-condition #ifdef USE_CURL if (g_settings->getBool("enable_remote_media_server")) { infostream << "Client: Adding remote server \"" << baseurl << "\" for media download" << std::endl; RemoteServerStatus *remote = new RemoteServerStatus(); remote->baseurl = baseurl; remote->active_count = 0; m_remotes.push_back(remote); } #else infostream << "Client: Ignoring remote server \"" << baseurl << "\" because cURL support is not compiled in" << std::endl; #endif } void ClientMediaDownloader::step(Client *client) { if (!m_initial_step_done) { initialStep(client); m_initial_step_done = true; } // Remote media: check for completion of fetches if (m_httpfetch_active) { bool fetched_something = false; HTTPFetchResult fetch_result; while (httpfetch_async_get(m_httpfetch_caller, fetch_result)) { m_httpfetch_active--; fetched_something = true; // Is this a hashset (index.mth) or a media file? if (fetch_result.request_id < m_remotes.size()) remoteHashSetReceived(fetch_result); else remoteMediaReceived(fetch_result, client); } if (fetched_something) startRemoteMediaTransfers(); // Did all remote transfers end and no new ones can be started? // If so, request still missing files from the minetest server // (Or report that we have all files.) if (m_httpfetch_active == 0) { if (m_uncached_received_count < m_uncached_count) { infostream << "Client: Failed to remote-fetch " << (m_uncached_count-m_uncached_received_count) << " files. Requesting them" << " the usual way." << std::endl; } startConventionalTransfers(client); } } } void ClientMediaDownloader::initialStep(Client *client) { // Check media cache m_uncached_count = m_files.size(); for (auto &file_it : m_files) { const std::string &name = file_it.first; FileStatus *filestatus = file_it.second; const std::string &sha1 = filestatus->sha1; if (tryLoadFromCache(name, sha1, client)) { filestatus->received = true; m_uncached_count--; } } assert(m_uncached_received_count == 0); // Create the media cache dir if we are likely to write to it if (m_uncached_count != 0) createCacheDirs(); // If we found all files in the cache, report this fact to the server. // If the server reported no remote servers, immediately start // conventional transfers. Note: if cURL support is not compiled in, // m_remotes is always empty, so "!USE_CURL" is redundant but may // reduce the size of the compiled code if (!USE_CURL || m_uncached_count == 0 || m_remotes.empty()) { startConventionalTransfers(client); } else { // Otherwise start off by requesting each server's sha1 set // This is the first time we use httpfetch, so alloc a caller ID m_httpfetch_caller = httpfetch_caller_alloc(); // Set the active fetch limit to curl_parallel_limit or 84, // whichever is greater. This gives us some leeway so that // inefficiencies in communicating with the httpfetch thread // don't slow down fetches too much. (We still want some limit // so that when the first remote server returns its hash set, // not all files are requested from that server immediately.) // One such inefficiency is that ClientMediaDownloader::step() // is only called a couple times per second, while httpfetch // might return responses much faster than that. // Note that httpfetch strictly enforces curl_parallel_limit // but at no inter-thread communication cost. This however // doesn't help with the aforementioned inefficiencies. // The signifance of 84 is that it is 2*6*9 in base 13. m_httpfetch_active_limit = g_settings->getS32("curl_parallel_limit"); m_httpfetch_active_limit = MYMAX(m_httpfetch_active_limit, 84); // Write a list of hashes that we need. This will be POSTed // to the server using Content-Type: application/octet-stream std::string required_hash_set = serializeRequiredHashSet(); // minor fixme: this loop ignores m_httpfetch_active_limit // another minor fixme, unlikely to matter in normal usage: // these index.mth fetches do (however) count against // m_httpfetch_active_limit when starting actual media file // requests, so if there are lots of remote servers that are // not responding, those will stall new media file transfers. for (u32 i = 0; i < m_remotes.size(); ++i) { assert(m_httpfetch_next_id == i); RemoteServerStatus *remote = m_remotes[i]; actionstream << "Client: Contacting remote server \"" << remote->baseurl << "\"" << std::endl; HTTPFetchRequest fetch_request; fetch_request.url = remote->baseurl + MTHASHSET_FILE_NAME; fetch_request.caller = m_httpfetch_caller; fetch_request.request_id = m_httpfetch_next_id; // == i fetch_request.method = HTTP_POST; fetch_request.raw_data = required_hash_set; fetch_request.extra_headers.emplace_back( "Content-Type: application/octet-stream"); // Encapsulate possible IPv6 plain address in [] std::string addr = client->getAddressName(); if (addr.find(':', 0) != std::string::npos) addr = '[' + addr + ']'; fetch_request.extra_headers.emplace_back( std::string("Referer: minetest://") + addr + ":" + std::to_string(client->getServerAddress().getPort())); httpfetch_async(fetch_request); m_httpfetch_active++; m_httpfetch_next_id++; m_outstanding_hash_sets++; } } } void ClientMediaDownloader::remoteHashSetReceived( const HTTPFetchResult &fetch_result) { u32 remote_id = fetch_result.request_id; assert(remote_id < m_remotes.size()); RemoteServerStatus *remote = m_remotes[remote_id]; m_outstanding_hash_sets--; if (fetch_result.succeeded) { try { // Server sent a list of file hashes that are // available on it, try to parse the list std::set<std::string> sha1_set; deSerializeHashSet(fetch_result.data, sha1_set); // Parsing succeeded: For every file that is // available on this server, add this server // to the available_remotes array for(auto it = m_files.upper_bound(m_name_bound); it != m_files.end(); ++it) { FileStatus *f = it->second; if (!f->received && sha1_set.count(f->sha1)) f->available_remotes.push_back(remote_id); } } catch (SerializationError &e) { infostream << "Client: Remote server \"" << remote->baseurl << "\" sent invalid hash set: " << e.what() << std::endl; } } } void ClientMediaDownloader::remoteMediaReceived( const HTTPFetchResult &fetch_result, Client *client) { // Some remote server sent us a file. // -> decrement number of active fetches // -> mark file as received if fetch succeeded // -> try to load media std::string name; { auto it = m_remote_file_transfers.find(fetch_result.request_id); assert(it != m_remote_file_transfers.end()); name = it->second; m_remote_file_transfers.erase(it); } sanity_check(m_files.count(name) != 0); FileStatus *filestatus = m_files[name]; sanity_check(!filestatus->received); sanity_check(filestatus->current_remote >= 0); RemoteServerStatus *remote = m_remotes[filestatus->current_remote]; filestatus->current_remote = -1; remote->active_count--; // If fetch succeeded, try to load media file if (fetch_result.succeeded) { bool success = checkAndLoad(name, filestatus->sha1, fetch_result.data, false, client); if (success) { filestatus->received = true; assert(m_uncached_received_count < m_uncached_count); m_uncached_received_count++; } } } s32 ClientMediaDownloader::selectRemoteServer(FileStatus *filestatus) { // Pre-conditions assert(filestatus != NULL); assert(!filestatus->received); assert(filestatus->current_remote < 0); if (filestatus->available_remotes.empty()) return -1; // Of all servers that claim to provide the file (and haven't // been unsuccessfully tried before), find the one with the // smallest number of currently active transfers s32 best = 0; s32 best_remote_id = filestatus->available_remotes[best]; s32 best_active_count = m_remotes[best_remote_id]->active_count; for (u32 i = 1; i < filestatus->available_remotes.size(); ++i) { s32 remote_id = filestatus->available_remotes[i]; s32 active_count = m_remotes[remote_id]->active_count; if (active_count < best_active_count) { best = i; best_remote_id = remote_id; best_active_count = active_count; } } filestatus->available_remotes.erase( filestatus->available_remotes.begin() + best); return best_remote_id; } void ClientMediaDownloader::startRemoteMediaTransfers() { bool changing_name_bound = true; for (auto files_iter = m_files.upper_bound(m_name_bound); files_iter != m_files.end(); ++files_iter) { // Abort if active fetch limit is exceeded if (m_httpfetch_active >= m_httpfetch_active_limit) break; const std::string &name = files_iter->first; FileStatus *filestatus = files_iter->second; if (!filestatus->received && filestatus->current_remote < 0) { // File has not been received yet and is not currently // being transferred. Choose a server for it. s32 remote_id = selectRemoteServer(filestatus); if (remote_id >= 0) { // Found a server, so start fetching RemoteServerStatus *remote = m_remotes[remote_id]; std::string url = remote->baseurl + hex_encode(filestatus->sha1); verbosestream << "Client: " << "Requesting remote media file " << "\"" << name << "\" " << "\"" << url << "\"" << std::endl; HTTPFetchRequest fetch_request; fetch_request.url = url; fetch_request.caller = m_httpfetch_caller; fetch_request.request_id = m_httpfetch_next_id; fetch_request.timeout = g_settings->getS32("curl_file_download_timeout"); httpfetch_async(fetch_request); m_remote_file_transfers.insert(std::make_pair( m_httpfetch_next_id, name)); filestatus->current_remote = remote_id; remote->active_count++; m_httpfetch_active++; m_httpfetch_next_id++; } } if (filestatus->received || (filestatus->current_remote < 0 && !m_outstanding_hash_sets)) { // If we arrive here, we conclusively know that we // won't fetch this file from a remote server in the // future. So update the name bound if possible. if (changing_name_bound) m_name_bound = name; } else changing_name_bound = false; } } void ClientMediaDownloader::startConventionalTransfers(Client *client) { assert(m_httpfetch_active == 0); // pre-condition if (m_uncached_received_count != m_uncached_count) { // Some media files have not been received yet, use the // conventional slow method (minetest protocol) to get them std::vector<std::string> file_requests; for (auto &file : m_files) { if (!file.second->received) file_requests.push_back(file.first); } assert((s32) file_requests.size() == m_uncached_count - m_uncached_received_count); client->request_media(file_requests); } } bool ClientMediaDownloader::conventionalTransferDone( const std::string &name, const std::string &data, Client *client) { // Check that file was announced auto file_iter = m_files.find(name); if (file_iter == m_files.end()) { errorstream << "Client: server sent media file that was" << "not announced, ignoring it: \"" << name << "\"" << std::endl; return false; } FileStatus *filestatus = file_iter->second; assert(filestatus != NULL); // Check that file hasn't already been received if (filestatus->received) { errorstream << "Client: server sent media file that we already" << "received, ignoring it: \"" << name << "\"" << std::endl; return true; } // Mark file as received, regardless of whether loading it works and // whether the checksum matches (because at this point there is no // other server that could send a replacement) filestatus->received = true; assert(m_uncached_received_count < m_uncached_count); m_uncached_received_count++; // Check that received file matches announced checksum // If so, load it checkAndLoad(name, filestatus->sha1, data, false, client); return true; } /* IClientMediaDownloader */ IClientMediaDownloader::IClientMediaDownloader(): m_media_cache(getMediaCacheDir()), m_write_to_cache(true) { } void IClientMediaDownloader::createCacheDirs() { if (!m_write_to_cache) return; std::string path = getMediaCacheDir(); if (!fs::CreateAllDirs(path)) { errorstream << "Client: Could not create media cache directory: " << path << std::endl; } } bool IClientMediaDownloader::tryLoadFromCache(const std::string &name, const std::string &sha1, Client *client) { std::ostringstream tmp_os(std::ios_base::binary); bool found_in_cache = m_media_cache.load(hex_encode(sha1), tmp_os); // If found in cache, try to load it from there if (found_in_cache) return checkAndLoad(name, sha1, tmp_os.str(), true, client); return false; } bool IClientMediaDownloader::checkAndLoad( const std::string &name, const std::string &sha1, const std::string &data, bool is_from_cache, Client *client) { const char *cached_or_received = is_from_cache ? "cached" : "received"; const char *cached_or_received_uc = is_from_cache ? "Cached" : "Received"; std::string sha1_hex = hex_encode(sha1); // Compute actual checksum of data std::string data_sha1; { SHA1 data_sha1_calculator; data_sha1_calculator.addBytes(data.c_str(), data.size()); unsigned char *data_tmpdigest = data_sha1_calculator.getDigest(); data_sha1.assign((char*) data_tmpdigest, 20); free(data_tmpdigest); } // Check that received file matches announced checksum if (data_sha1 != sha1) { std::string data_sha1_hex = hex_encode(data_sha1); infostream << "Client: " << cached_or_received_uc << " media file " << sha1_hex << " \"" << name << "\" " << "mismatches actual checksum " << data_sha1_hex << std::endl; return false; } // Checksum is ok, try loading the file bool success = loadMedia(client, data, name); if (!success) { infostream << "Client: " << "Failed to load " << cached_or_received << " media: " << sha1_hex << " \"" << name << "\"" << std::endl; return false; } verbosestream << "Client: " << "Loaded " << cached_or_received << " media: " << sha1_hex << " \"" << name << "\"" << std::endl; // Update cache (unless we just loaded the file from the cache) if (!is_from_cache && m_write_to_cache) m_media_cache.update(sha1_hex, data); return true; } /* Minetest Hashset File Format All values are stored in big-endian byte order. [u32] signature: 'MTHS' [u16] version: 1 For each hash in set: [u8*20] SHA1 hash Version changes: 1 - Initial version */ std::string ClientMediaDownloader::serializeRequiredHashSet() { std::ostringstream os(std::ios::binary); writeU32(os, MTHASHSET_FILE_SIGNATURE); // signature writeU16(os, 1); // version // Write list of hashes of files that have not been // received (found in cache) yet for (const auto &it : m_files) { if (!it.second->received) { FATAL_ERROR_IF(it.second->sha1.size() != 20, "Invalid SHA1 size"); os << it.second->sha1; } } return os.str(); } void ClientMediaDownloader::deSerializeHashSet(const std::string &data, std::set<std::string> &result) { if (data.size() < 6 || data.size() % 20 != 6) { throw SerializationError( "ClientMediaDownloader::deSerializeHashSet: " "invalid hash set file size"); } const u8 *data_cstr = (const u8*) data.c_str(); u32 signature = readU32(&data_cstr[0]); if (signature != MTHASHSET_FILE_SIGNATURE) { throw SerializationError( "ClientMediaDownloader::deSerializeHashSet: " "invalid hash set file signature"); } u16 version = readU16(&data_cstr[4]); if (version != 1) { throw SerializationError( "ClientMediaDownloader::deSerializeHashSet: " "unsupported hash set file version"); } for (u32 pos = 6; pos < data.size(); pos += 20) { result.insert(data.substr(pos, 20)); }