MG Mud User | 88f1247 | 2016-06-24 23:31:02 +0200 | [diff] [blame] | 1 | /** Doku: |
| 2 | public int AddRoute(string transporter, string* l_stops) |
| 3 | Beschreibung: |
| 4 | Setzt die aktuelle Route eines Transporters. |
| 5 | Die Route hier ist allerdings nur die Liste von Haltepunkten - nicht der |
| 6 | Rest, der in den Routen in den Transporter abgelegt ist. |
| 7 | Eine neue Route ersetzt immer die bisherige Route |
| 8 | Wird von /std/transport gerufen, wenn P_NO_TRAVELING nicht gesetzt ist. |
| 9 | Rueckgabewerte: |
| 10 | - 1: Transporter und Haltepunkt Eingetragen. |
| 11 | - 0: Transporter oder Haltepunkt kein string|string*. |
| 12 | - -1: Es wurde versucht einen Transporter in /p/ oder /players/ |
| 13 | einzutragen |
| 14 | und der Eintragende ist kein EM. |
| 15 | |
| 16 | public int RemoveTransporter(string|object transporter) |
| 17 | public int RemoveStop(string|object stop) |
| 18 | Beschreibung: |
| 19 | Entfernt einen kompletten Transporter oder einen Haltepunkt aus dem |
| 20 | Daemon. |
| 21 | Rueckgabewerte: |
| 22 | - 1: Erfolg. |
| 23 | - 0: Transporter|Haltepunkt existiert nicht. |
| 24 | - -1: Keine Berechtigung. |
| 25 | |
| 26 | public varargs int|object* HasTransporter(object stop, string transporter_id) |
| 27 | Beschreibung: |
| 28 | Gibt eine Liste der Transporter aus, den den Haltepunkt anlaufen, oder |
| 29 | prueft auf einen bestimmten Transporter. |
| 30 | Rueckgabewerte: |
| 31 | - 0: Haltepunkt ist kein Objekt, dem Haltepunkt sind keine Transporter |
| 32 | zugeordnet, |
| 33 | die Transporter sind nicht geladen |
| 34 | oder, bei angabe von <transporter>: <transporter> faehrt den Hafen |
| 35 | nicht an. |
| 36 | - object*: Liste mit einem oder mehreren Transportern. |
| 37 | Bemerkung: |
| 38 | Auch bei der Angabe von <transporter_id> koennen mehrere Objekte |
| 39 | zurueckgegeben werden, es wird mittels id() gefiltert. |
| 40 | |
| 41 | public mixed RouteExists(object transporter, string dest) |
| 42 | Beschreibung: |
| 43 | Prueft auf korrekte Parameter und leitet die Anfrage an |
| 44 | /std/transport weiter. |
| 45 | Rueckgabewerte: |
| 46 | - 0: <transporter> ist kein Objekt oder <dest> ist kein String. |
| 47 | - Das Ergebnis von transporter->HasRoute(dest); |
| 48 | |
| 49 | public mapping QueryTransporters() |
| 50 | public mapping QueryAllStops() |
| 51 | public mapping QueryTransporterByStop(string stop) |
| 52 | public mapping QueryStopsByTransporter(string transporter) |
| 53 | Beschreibung: |
| 54 | Gibt das jewilige gefilterte oder ungefilterte Mapping aus. |
| 55 | Rueckgabewerte: |
| 56 | Mapping mit dem Ergebnis. |
| 57 | */ |
| 58 | |
| 59 | #pragma strong_types,save_types,rtt_checks |
| 60 | #pragma pedantic,range_check,warn_deprecated |
| 61 | #pragma warn_empty_casts,warn_missing_return,warn_function_inconsistent |
| 62 | #pragma no_clone,no_shadow |
| 63 | |
| 64 | #include <transport.h> |
| 65 | #include <wizlevels.h> |
| 66 | #include <daemon.h> |
| 67 | |
| 68 | #define TRANSPORTER 0 |
| 69 | #define STOP 1 |
| 70 | |
| 71 | #define MEMORY "/secure/memory" |
| 72 | |
Zesstra | 4e7b332 | 2019-07-19 17:23:39 +0200 | [diff] [blame] | 73 | // Propertyname zum Speichern der Reloadversuchszaehler |
| 74 | #define RELOAD_TRIES "p_lib_reload_try_counter" |
| 75 | |
MG Mud User | 88f1247 | 2016-06-24 23:31:02 +0200 | [diff] [blame] | 76 | // Key: string Schiff. |
| 77 | // Value0: string* Haefen, die von <Schiff> angefahren werden. |
| 78 | mapping transporters = ([]); |
| 79 | // Key: string Hafen |
| 80 | // Value0: string* Schiffe, die <Hafen> anfahren. |
| 81 | mapping stops = ([]); |
| 82 | |
| 83 | protected void create() |
| 84 | { |
| 85 | // Vom MEMORY die Daten abholen. |
| 86 | // Wenn keine da, war vermutlich Reboot, dann muessen wir warten, bis die |
| 87 | // Transporter sich wieder melden. Aber die Mappings muessen ins Memory |
| 88 | // geschrieben werden. |
| 89 | mapping tmp = MEMORY->Load("stops"); |
| 90 | if (mappingp(tmp)) |
| 91 | { |
| 92 | stops = tmp; |
| 93 | } |
| 94 | else |
| 95 | { |
| 96 | if (MEMORY->Save("stops",stops) != 1) |
| 97 | raise_error("Could not save memory to /secure/memory."); |
| 98 | } |
| 99 | tmp = MEMORY->Load("transporters"); |
| 100 | if (mappingp(tmp)) |
| 101 | { |
| 102 | transporters = tmp; |
| 103 | } |
| 104 | else |
| 105 | { |
| 106 | if (MEMORY->Save("transporters",transporters) != 1) |
| 107 | raise_error("Could not save memory to /secure/memory."); |
| 108 | } |
Zesstra | 4e7b332 | 2019-07-19 17:23:39 +0200 | [diff] [blame] | 109 | set_next_reset(86400); |
| 110 | } |
| 111 | |
| 112 | protected void reload_transporters(object *ships) |
| 113 | { |
| 114 | if (!sizeof(ships)) |
| 115 | return; |
| 116 | else if (sizeof(ships) > 1) |
| 117 | call_out(#'reload_transporters, 45, ships[1..]); |
| 118 | |
| 119 | object s = ships[0]; |
| 120 | if (s) |
| 121 | { |
| 122 | object *inv = all_inventory(s); |
| 123 | if (sizeof(inv)) |
| 124 | { |
| 125 | // Wenn Spieler drin sind, muss auf jeden Fall gewartet werden. |
| 126 | // Bei sonstigem Krempel nur dann, wenn wir das noch nicht oft genug |
| 127 | // erfolglos versucht haben. |
| 128 | // Bemerkung: eigentlich muesste man hier pruefen, was davon per |
| 129 | // AddItem() erzeugt wurde. Das ist mir gerade aber zu aufwendig. |
| 130 | if (sizeof(filter(inv, #'query_once_interactive)) |
| 131 | || s->QueryProp(RELOAD_TRIES) < 3) |
| 132 | { |
| 133 | s->SetProp(RELOAD_TRIES, s->QueryProp(RELOAD_TRIES) + 1); |
| 134 | // naechsten reset vorziehen |
| 135 | set_next_reset(10800); |
| 136 | return; |
| 137 | } |
| 138 | // ansonsten ist uns das Inventar jetzt egal und bald Schrott... |
| 139 | } |
| 140 | string sname = object_name(s); |
| 141 | s->remove(1); |
| 142 | load_object(sname); |
| 143 | // Und hoffentlich starten Transporter selber. Wenn nicht, muss hier evtl. |
| 144 | // noch nen Continue hin. |
| 145 | } |
Zesstra | c1f496a | 2017-09-21 22:56:52 +0200 | [diff] [blame] | 146 | } |
| 147 | |
| 148 | public void reset() |
| 149 | { |
Zesstra | 4e7b332 | 2019-07-19 17:23:39 +0200 | [diff] [blame] | 150 | set_next_reset(86400); |
| 151 | // Zeit des Programms vom Standardtransporter herausfinden. Wenn der nicht |
| 152 | // geladen ist, muss nix gemacht werden. |
| 153 | object std_transport = find_object("/std/transport"); |
| 154 | if (!std_transport) |
| 155 | return; |
| 156 | int std_time = program_time(std_transport); |
Zesstra | f15383d | 2019-07-19 16:35:06 +0200 | [diff] [blame] | 157 | |
Zesstra | 4e7b332 | 2019-07-19 17:23:39 +0200 | [diff] [blame] | 158 | // ueber alle Transporter laufen. Wenn der Transport aelter ist als |
| 159 | // /std/transport, wird er neu gelauden. |
| 160 | // Das kann Nebenwirkungen haben. Aber in diesem Fall sollte man besser |
| 161 | // den Transporter so bauen, dass es OK ist. |
| 162 | // Aber speziell bei VC-Transportern ist fraglich, wie gut das |
| 163 | // funktioniert... |
| 164 | // Und es funktioniert nur fuer Blueprints. |
| 165 | object *old_ships = map(transporters, function object (string sname) |
| 166 | { |
| 167 | object ship = find_object(sname); |
| 168 | if (ship && !clonep(ship) && program_time(ship) < std_time) |
| 169 | return ship; |
| 170 | return 0; |
| 171 | } ); |
| 172 | old_ships -= ({0}); |
| 173 | if (sizeof(old_ships)) |
| 174 | reload_transporters(old_ships); |
MG Mud User | 88f1247 | 2016-06-24 23:31:02 +0200 | [diff] [blame] | 175 | } |
| 176 | |
| 177 | varargs int remove(int s) |
| 178 | { |
| 179 | destruct(this_object()); |
| 180 | return 1; |
| 181 | } |
| 182 | |
| 183 | // Loeschfunktion |
| 184 | private int _remove_data(string|object key, int what) |
| 185 | { |
| 186 | mapping first,second; |
| 187 | // Erstmal die Arbeitsmappings befuellen. |
| 188 | if(what==TRANSPORTER) |
| 189 | { |
| 190 | first=transporters; |
| 191 | second=stops; |
| 192 | } |
| 193 | else |
| 194 | { |
| 195 | first=stops; |
| 196 | second=transporters; |
| 197 | } |
| 198 | |
| 199 | // Key auf einen String normalisieren. |
| 200 | if(objectp(key)) |
| 201 | { |
| 202 | key=object_name(key); |
| 203 | } |
| 204 | |
| 205 | if(!member(first,key)) |
| 206 | { |
| 207 | return 0; |
| 208 | } |
| 209 | |
| 210 | // Erstmal aus dem jeweils anderen Mapping austragen. |
| 211 | foreach(string s : first[key]) |
| 212 | { |
| 213 | // Nicht existent oder nicht zugeordnet -> weiter. |
| 214 | if(!member(second,s) || member(second[s],key)==-1) |
| 215 | { |
| 216 | continue; |
| 217 | } |
| 218 | second[s]-=({key}); |
| 219 | if (!sizeof(second[s])) |
| 220 | m_delete(second,s); |
| 221 | } |
| 222 | // Jetzt noch aus dem eigenen Mapping austragen. |
| 223 | m_delete(first,key); |
| 224 | return 1; |
| 225 | } |
| 226 | |
| 227 | // Ein komplettes Schiff entfernen. |
| 228 | public int RemoveTransporter(object|string transporter) |
| 229 | { |
Zesstra@Morgengrauen | f8f1b75 | 2016-11-01 13:33:38 +0100 | [diff] [blame] | 230 | if (extern_call() |
| 231 | && previous_object() != transporter |
| 232 | && !IS_ELDER(getuid(previous_object()))) |
| 233 | return -1; |
MG Mud User | 88f1247 | 2016-06-24 23:31:02 +0200 | [diff] [blame] | 234 | return _remove_data(transporter,TRANSPORTER); |
| 235 | } |
| 236 | |
| 237 | // Entfernt einen kompletten Hafen aus dem Daemon |
| 238 | public int RemoveStop(object|string stop) |
| 239 | { |
Zesstra@Morgengrauen | f8f1b75 | 2016-11-01 13:33:38 +0100 | [diff] [blame] | 240 | if (extern_call() |
| 241 | && previous_object() != stop |
| 242 | && !IS_ELDER(getuid(previous_object()))) |
| 243 | return -1; |
MG Mud User | 88f1247 | 2016-06-24 23:31:02 +0200 | [diff] [blame] | 244 | return _remove_data(stop,STOP); |
| 245 | } |
| 246 | |
| 247 | // Setzt die aktuelle Route eines Transporters. |
| 248 | // Die Route hier ist allerdings nur die Liste von Haltepunkten - nicht der |
| 249 | // Rest, der in den Routen in den Transporter abgelegt ist. |
| 250 | // Wird von /std/transport gerufen, wenn P_NO_TRAVELING nicht gesetzt ist. |
| 251 | public int AddRoute(string transporter, string* l_stops) |
| 252 | { |
| 253 | if (!stringp(transporter) || !pointerp(l_stops) || !sizeof(l_stops)) |
| 254 | return 0; |
| 255 | |
| 256 | if ((transporter[0..2] == "/p/" || transporter[0..8] == "/players/") |
| 257 | && !IS_ARCH(getuid(previous_object()))) |
| 258 | { |
| 259 | return -1; |
| 260 | } |
| 261 | |
| 262 | // Damit sind alle Abfragen auf find_object(transporter|stop) ueberfluessig, |
| 263 | // im Zweifelsfall knallt es naemlich hier. ;-) |
| 264 | transporter = object_name(load_object(transporter)); |
| 265 | |
| 266 | // Wenn die route bereits existiert, austragen. Dies ist noetig, weil die |
| 267 | // Haltepunkte ja auch bereinigt werden muessen. |
| 268 | if (member(transporters, transporter)) |
| 269 | RemoveTransporter(transporter); |
| 270 | |
| 271 | // Transporter eintragen, die Route wird dabei immer ueberschrieben, wenn |
| 272 | // der Transporter schon bekannt ist. Ziel: hier ist immer die aktuell |
| 273 | // konfigurierte Route drin. |
| 274 | m_add(transporters, transporter, l_stops); |
| 275 | |
| 276 | // Nach obigen Schema, nur diesmal Haltepunkte und zugehoerige Schiffe |
| 277 | // eintragen. |
| 278 | foreach(string stop : l_stops) |
| 279 | { |
| 280 | if (!member(stops,stop)) |
| 281 | { |
| 282 | m_add(stops, stop, ({transporter})); |
| 283 | } |
| 284 | else if (member(stops[stop],transporter) == -1) |
| 285 | { |
| 286 | stops[stop] += ({ transporter }); |
| 287 | } |
| 288 | } |
| 289 | return 1; |
| 290 | |
| 291 | } |
| 292 | |
| 293 | // Abfrage ob ein Schiff einen Hafen anfaehrt. |
| 294 | public varargs int|object* HasTransporter(object stop, |
| 295 | string transporter_id) |
| 296 | { |
| 297 | if (!objectp(stop)) return 0; |
| 298 | |
| 299 | <string|object>* trans=stops[object_name(stop)]; |
| 300 | if (!pointerp(trans)) return 0; |
| 301 | |
| 302 | // Eigentlich sollten sich Transporter ordentlich abmelden... Aber zur |
| 303 | // Sicherheit die nicht gefundenen Objekte ausfiltern. |
| 304 | trans = map(trans, #'find_object) - ({0}); |
| 305 | |
| 306 | // Wenn ne bestimmte ID gesucht wird, nur diese Transporter liefern. |
| 307 | if (stringp(transporter_id)) |
| 308 | trans = filter_objects(trans, "id", transporter_id); |
| 309 | |
| 310 | return sizeof(trans) ? trans : 0; |
| 311 | } |
| 312 | |
| 313 | public mixed RouteExists(object transporter, string dest) |
| 314 | { |
| 315 | if (!objectp(transporter) || !stringp(dest)) |
| 316 | { |
| 317 | return 0; |
| 318 | } |
| 319 | return transporter->HasRoute(dest); |
| 320 | } |
| 321 | |
| 322 | public mapping QueryTransporters() |
| 323 | { |
| 324 | return deep_copy(transporters); |
| 325 | } |
| 326 | |
| 327 | public mapping QueryAllStops() |
| 328 | { |
| 329 | return deep_copy(stops); |
| 330 | } |
| 331 | |
| 332 | // Liefert alle Transporter mit ihren Haltepunkten zurueck, die diesen Halt ansteuern. |
| 333 | public mapping QueryTransportByStop(string stop) |
| 334 | { |
| 335 | mapping res=m_allocate(4); |
| 336 | foreach(string s, string* arr : transporters) |
| 337 | { |
| 338 | if(member(arr,stop)!=-1) |
| 339 | { |
| 340 | m_add(res,s,arr); |
| 341 | } |
| 342 | } |
| 343 | return res; |
| 344 | } |
| 345 | |
| 346 | // Liefert alle Haltepunkte mit ihren Transportern zurueck, die dieser |
| 347 | // transporter ansteuert. |
| 348 | public mapping QueryStopsByTransporter(string transporter) |
| 349 | { |
| 350 | mapping res=m_allocate(4); |
| 351 | foreach(string s, string* arr : stops) |
| 352 | { |
| 353 | if(member(arr,transporter)!=-1) |
| 354 | { |
| 355 | m_add(res,s,arr); |
| 356 | } |
| 357 | } |
| 358 | return res; |
| 359 | } |
| 360 | |