mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2025-12-11 10:11:09 +00:00
Done using this command sed -Ei 's/(\s*\S+)\s*\t+/\1 /g' code/**/*.dm We have countless examples in the codebase with this style gone wrong, and defines and such being on hideously different levels of indentation. Fixing this to keep the alignment involves tainting the blames of code your PR doesn't need to be touching at all. And ultimately, it's hideous. There are some files that this sed makes uglier. I can fix these when they are pointed out, but I believe this is ultimately for the greater good of readability. I'm more concerned with if any strings relied on this. Hi codeowners! Co-authored-by: Jared-Fogle <35135081+Jared-Fogle@users.noreply.github.com>
524 lines
20 KiB
Plaintext
524 lines
20 KiB
Plaintext
SUBSYSTEM_DEF(networks)
|
|
name = "Networks"
|
|
priority = FIRE_PRIORITY_NETWORKS
|
|
wait = 5
|
|
flags = SS_KEEP_TIMING
|
|
init_order = INIT_ORDER_NETWORKS
|
|
|
|
var/list/relays = list()
|
|
/// Legacy ntnet lookup for software. Should be changed latter so don't rely on this
|
|
/// being here.
|
|
var/datum/ntnet/station_root/station_network
|
|
var/datum/ntnet/station_root/syndie_network
|
|
var/list/network_initialize_queue = list()
|
|
/// all interfaces by their hardware address.
|
|
/// Do NOT use to verify a reciver_id is valid, use the network.root_devices for that
|
|
var/list/interfaces_by_hardware_id = list()
|
|
/// Area to network, network to area list
|
|
/// This is an associated list to quickly find an area using either its id or network
|
|
/// Used mainly to make sure all area id's are unique even if a mapper uses the same
|
|
/// area many times
|
|
var/list/area_network_lookup = list()
|
|
|
|
/// List of networks using their fully qualified network name. Used for quick lookups
|
|
/// of networks for sending packets
|
|
var/list/networks = list()
|
|
/// List of the root networks starting at their root names. Used to find and/or build
|
|
/// network tress
|
|
var/list/root_networks = list()
|
|
|
|
|
|
|
|
// Why not list? Because its a Copy() every time we add a packet, and thats stupid.
|
|
var/datum/netdata/first = null // start of the queue. Pulled off in fire.
|
|
var/datum/netdata/last = null // end of the queue. pushed on by transmit
|
|
var/packet_count = 0
|
|
// packet stats
|
|
var/count_broadcasts_packets = 0 // count of broadcast packets sent
|
|
var/count_failed_packets = 0 // count of message fails
|
|
var/count_good_packets = 0
|
|
// Logs moved here
|
|
// Amount of logs the system tries to keep in memory. Keep below 999 to prevent byond from acting weirdly.
|
|
// High values make displaying logs much laggier.
|
|
var/setting_maxlogcount = 100
|
|
var/list/logs = list()
|
|
|
|
/// Random name search to make sure we have unique names.
|
|
/// DO NOT REMOVE NAMES HERE UNLESS YOU KNOW WHAT YOUR DOING
|
|
var/list/used_names = list()
|
|
|
|
|
|
/// You shouldn't need to do this. But mapping is async and there is no guarantee that Initialize
|
|
/// will run before these networks are dynamically created. So its here.
|
|
/datum/controller/subsystem/networks/PreInit()
|
|
/// Limbo network needs to be made at boot up for all error devices
|
|
new/datum/ntnet(LIMBO_NETWORK_ROOT)
|
|
station_network = new(STATION_NETWORK_ROOT)
|
|
syndie_network = new(SYNDICATE_NETWORK_ROOT)
|
|
/// As well as the station network incase something funny goes during startup
|
|
new/datum/ntnet(CENTCOM_NETWORK_ROOT)
|
|
|
|
|
|
/datum/controller/subsystem/networks/stat_entry(msg)
|
|
msg = "NET: QUEUE([packet_count]) FAILS([count_failed_packets]) BROADCAST([count_broadcasts_packets])"
|
|
return ..()
|
|
|
|
/datum/controller/subsystem/networks/Initialize()
|
|
station_network.register_map_supremecy() // sigh
|
|
assign_areas_root_ids(GLOB.sortedAreas) // setup area names before Initialize
|
|
station_network.build_software_lists()
|
|
syndie_network.build_software_lists()
|
|
|
|
// At round start, fix the network_id's so the station root is on them
|
|
initialized = TRUE
|
|
// Now when the objects Initialize they will join the right network
|
|
return ..()
|
|
|
|
/*
|
|
* Process incoming queued packet and return NAK/ACK signals
|
|
*
|
|
* This should only be called when you want the target object to process the NAK/ACK signal, usually
|
|
* during fire. At this point data.receiver_id has already been converted if it was a broadcast but
|
|
* is undefined in this function.
|
|
* Arguments:
|
|
* * receiver_id - text hardware id for the target device
|
|
* * data - packet to be sent
|
|
*/
|
|
|
|
/datum/controller/subsystem/networks/proc/_process_packet(receiver_id, datum/netdata/data)
|
|
/// Used only for sending NAK/ACK and error reply's
|
|
var/datum/component/ntnet_interface/sending_interface = interfaces_by_hardware_id[data.sender_id]
|
|
|
|
/// Check if the network_id is valid and if not send an error and return
|
|
var/datum/ntnet/target_network = networks[data.network_id]
|
|
if(!target_network)
|
|
count_failed_packets++
|
|
add_log("Bad target network '[data.network_id]'", null, data.sender_id)
|
|
if(!QDELETED(sending_interface))
|
|
SEND_SIGNAL(sending_interface.parent, COMSIG_COMPONENT_NTNET_NAK, data , NETWORK_ERROR_BAD_NETWORK)
|
|
return
|
|
|
|
/// Check if the receiver_id is in the network. If not send an error and return
|
|
var/datum/component/ntnet_interface/target_interface = target_network.root_devices[receiver_id]
|
|
if(QDELETED(target_interface))
|
|
count_failed_packets++
|
|
add_log("Bad target device '[receiver_id]'", target_network, data.sender_id)
|
|
if(!QDELETED(sending_interface))
|
|
SEND_SIGNAL(sending_interface.parent, COMSIG_COMPONENT_NTNET_NAK, data, NETWORK_ERROR_BAD_RECEIVER_ID)
|
|
return
|
|
|
|
// Check if we care about permissions. If we do check if we are allowed the message to be processed
|
|
if(data.passkey) // got to check permissions
|
|
var/obj/O = target_interface.parent
|
|
if(O)
|
|
if(!O.check_access_ntnet(data.passkey))
|
|
count_failed_packets++
|
|
add_log("Access denied to ([receiver_id]) from ([data.network_id])", target_network, data.sender_id)
|
|
if(!QDELETED(sending_interface))
|
|
SEND_SIGNAL(sending_interface.parent, COMSIG_COMPONENT_NTNET_NAK, data, NETWORK_ERROR_UNAUTHORIZED)
|
|
return
|
|
else
|
|
add_log("A access key message was sent to a non-device", target_network, data.sender_id)
|
|
if(!QDELETED(sending_interface))
|
|
SEND_SIGNAL(sending_interface.parent, COMSIG_COMPONENT_NTNET_NAK, data, NETWORK_ERROR_UNAUTHORIZED)
|
|
|
|
|
|
SEND_SIGNAL(target_interface.parent, COMSIG_COMPONENT_NTNET_RECEIVE, data)
|
|
// All is good, send the packet then send an ACK to the sender
|
|
if(!QDELETED(sending_interface))
|
|
SEND_SIGNAL(sending_interface.parent, COMSIG_COMPONENT_NTNET_ACK, data)
|
|
count_good_packets++
|
|
|
|
/// Helper define to make sure we pop the packet and qdel it
|
|
#define POP_PACKET(CURRENT) first = CURRENT.next; packet_count--; if(!first) { last = null; packet_count = 0; }; qdel(CURRENT);
|
|
|
|
/datum/controller/subsystem/networks/fire(resumed = 0)
|
|
var/datum/netdata/current
|
|
var/datum/component/ntnet_interface/target_interface
|
|
while(first)
|
|
current = first
|
|
/// Check if we are a list. If so process the list
|
|
if(islist(current.receiver_id)) // are we a broadcast list
|
|
var/list/receivers = current.receiver_id
|
|
var/receiver_id = receivers[receivers.len--] // pop it
|
|
_process_packet(receiver_id, current)
|
|
if(receivers.len == 0) // pop it if done
|
|
count_broadcasts_packets++
|
|
POP_PACKET(current)
|
|
else // else set up a broadcast or send a single targete
|
|
// check if we are sending to a network or to a single target
|
|
target_interface = interfaces_by_hardware_id[current.receiver_id]
|
|
if(target_interface) // a single sender id
|
|
_process_packet(current.receiver_id, current) // single target
|
|
POP_PACKET(current)
|
|
else // ok so lets find the network to send it too
|
|
var/datum/ntnet/net = networks[current.network_id] // get the sending network
|
|
net = net?.networks[current.receiver_id] // find the target network to broadcast
|
|
if(net) // we found it
|
|
current.receiver_id = net.collect_interfaces() // make a list of all the sending targets
|
|
else
|
|
// We got an error, the network is bad so send a NAK
|
|
target_interface = interfaces_by_hardware_id[current.sender_id]
|
|
if(!QDELETED(target_interface))
|
|
SEND_SIGNAL(target_interface.parent, COMSIG_COMPONENT_NTNET_NAK, current , NETWORK_ERROR_BAD_NETWORK)
|
|
POP_PACKET(current) // and get rid of it
|
|
if (MC_TICK_CHECK)
|
|
return
|
|
|
|
#undef POP_PACKET
|
|
|
|
/*
|
|
* Main function to queue a packet. As long as we have valid receiver_id and network_id we will take it
|
|
*
|
|
* Main queuing function for any message sent. if the data.receiver_id is null, then it will be broadcasted
|
|
* error checking is only done during the process this just throws it on the queue.
|
|
* Arguments:
|
|
* * data - packet to be sent
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/transmit(datum/netdata/data)
|
|
data.next = null // sanity check
|
|
|
|
if(!last)
|
|
first = last = data
|
|
else
|
|
last.next = data
|
|
last = data
|
|
packet_count++
|
|
// We do error checking when the packet is sent
|
|
return NETWORK_ERROR_OK
|
|
|
|
|
|
/datum/controller/subsystem/networks/proc/check_relay_operation(zlevel=0) //can be expanded later but right now it's true/false.
|
|
for(var/i in relays)
|
|
var/obj/machinery/ntnet_relay/n = i
|
|
if(zlevel && n.z != zlevel)
|
|
continue
|
|
if(n.is_operational)
|
|
return TRUE
|
|
return FALSE
|
|
|
|
/datum/controller/subsystem/networks/proc/log_data_transfer( datum/netdata/data)
|
|
logs += "[station_time_timestamp()] - [data.generate_netlog()]"
|
|
if(logs.len > setting_maxlogcount)
|
|
logs = logs.Copy(logs.len - setting_maxlogcount, 0)
|
|
|
|
/**
|
|
* Records a message into the station logging system for the network
|
|
*
|
|
* This CAN be read in station by personal so do not use it for game debugging
|
|
* during fire. At this point data.receiver_id has already been converted if it was a broadcast but
|
|
* is undefined in this function. It is also dumped to normal logs but remember players can read/intercept
|
|
* these messages
|
|
* Arguments:
|
|
* * log_string - message to log
|
|
* * network - optional, It can be a ntnet or just the text equivalent
|
|
* * hardware_id = optional, text, will look it up and return with the parent.name as well
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/add_log(log_string, network = null , hardware_id = null)
|
|
set waitfor = FALSE // so process keeps running
|
|
var/list/log_text = list()
|
|
log_text += "\[[station_time_timestamp()]\]"
|
|
if(network)
|
|
var/datum/ntnet/net = network
|
|
if(!istype(net))
|
|
net = networks[network]
|
|
if(net) // bad network?
|
|
log_text += "{[net.network_id]}"
|
|
else // bad network?
|
|
log_text += "{[network] *BAD*}"
|
|
|
|
if(hardware_id)
|
|
var/datum/component/ntnet_interface/conn = interfaces_by_hardware_id[hardware_id]
|
|
if(conn)
|
|
log_text += " ([hardware_id])[conn.parent]"
|
|
else
|
|
log_text += " ([hardware_id])*BAD ID*"
|
|
else
|
|
log_text += "*SYSTEM*"
|
|
log_text += " - "
|
|
log_text += log_string
|
|
log_string = log_text.Join()
|
|
|
|
logs.Add(log_string)
|
|
//log_telecomms("NetLog: [log_string]") // causes runtime on startup humm
|
|
|
|
// We have too many logs, remove the oldest entries until we get into the limit
|
|
if(logs.len > setting_maxlogcount)
|
|
logs = logs.Copy(logs.len-setting_maxlogcount,0)
|
|
|
|
|
|
/**
|
|
* Removes all station logs for the current game
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/purge_logs()
|
|
logs = list()
|
|
add_log("-!- LOGS DELETED BY SYSTEM OPERATOR -!-")
|
|
|
|
|
|
|
|
/**
|
|
* Updates the maximum amount of logs and purges those that go beyond that number
|
|
*
|
|
* Shouldn't been needed to be run by players but maybe admins need it?
|
|
* Arguments:
|
|
* * lognumber - new setting_maxlogcount count
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/update_max_log_count(lognumber)
|
|
if(!lognumber)
|
|
return FALSE
|
|
// Trim the value if necessary
|
|
lognumber = max(MIN_NTNET_LOGS, min(lognumber, MAX_NTNET_LOGS))
|
|
setting_maxlogcount = lognumber
|
|
add_log("Configuration Updated. Now keeping [setting_maxlogcount] logs in system memory.")
|
|
|
|
|
|
|
|
/**
|
|
* Gives an area a root and a network_area_id
|
|
*
|
|
* When a device is added to the network on map load, it needs to know where it is.
|
|
* So that it is added to that ruins/base's network instead of the general station network
|
|
* This way people on the station cannot just hack Charlie's doors and visa versa. All area's
|
|
* "should" have this information and if not one is created from existing map tags or
|
|
* ruin template id's. This SHOULD run before the Initialize of a atom, or the root will not
|
|
* be put in the object.area
|
|
*
|
|
* An example on what the area.network_root_id does/
|
|
* Before Init: obj.network_id = "ATMOS.SCRUBBER" area.network_root_id="SS13_STATION" area.network_area_id = "BRIDGE"
|
|
* After Init: obj.network_id = "SS13_STATION.ATMOS.SCRUBBER" also obj.network_id = "SS13_STATION.AREA.BRIDGE"
|
|
*
|
|
* Arguments:
|
|
* * area - Area to modify the root id.
|
|
* * template - optional, map_template of that area
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/lookup_area_root_id(area/A, datum/map_template/M=null)
|
|
/// Check if the area is valid and if it doesn't have a network root id.
|
|
if(!istype(A) || A.network_root_id != null)
|
|
return
|
|
|
|
/// If we are a ruin or a shuttle, we get our own network
|
|
if(M)
|
|
/// if we have a template, try to get the network id from the template
|
|
if(M.station_id && M.station_id != LIMBO_NETWORK_ROOT) // check if the template specifies it
|
|
A.network_root_id = simple_network_name_fix(M.station_id)
|
|
else if(istype(M, /datum/map_template/shuttle)) // if not, then check if its a shuttle type
|
|
var/datum/map_template/shuttle/T = M // we are a shuttle so use shuttle id
|
|
A.network_root_id = simple_network_name_fix(T.shuttle_id)
|
|
else if(istype(M,/datum/map_template/ruin)) // if not again, check if its a ruin type
|
|
var/datum/map_template/ruin/R = M
|
|
A.network_root_id = simple_network_name_fix(R.id)
|
|
|
|
if(!A.network_root_id) // not assigned? Then lets use some defaults
|
|
// Anything in Centcom is completely isolated
|
|
// Special case for holodecks.
|
|
if(istype(A,/area/holodeck))
|
|
A.network_root_id = "HOLODECK" // isolated from the station network
|
|
else if(SSmapping.level_trait(A.z, ZTRAIT_CENTCOM))
|
|
A.network_root_id = CENTCOM_NETWORK_ROOT
|
|
// Otherwise the default is the station
|
|
else
|
|
A.network_root_id = STATION_NETWORK_ROOT
|
|
|
|
/datum/controller/subsystem/networks/proc/assign_area_network_id(area/A, datum/map_template/M=null)
|
|
if(!istype(A))
|
|
return
|
|
if(!A.network_root_id)
|
|
lookup_area_root_id(A, M)
|
|
// finally set the network area id, bit copy paste from area Initialize
|
|
// This is done in case we have more than one area type, each area instance has its own network name
|
|
if(!A.network_area_id)
|
|
A.network_area_id = A.network_root_id + ".AREA." + simple_network_name_fix(A.name) // Make the string
|
|
if(!(A.area_flags & UNIQUE_AREA)) // if we aren't a unique area, make sure our name is different
|
|
A.network_area_id = SSnetworks.assign_random_name(5, A.network_area_id + "_") // tack on some garbage incase there are two area types
|
|
|
|
/datum/controller/subsystem/networks/proc/assign_areas_root_ids(list/areas, datum/map_template/M=null)
|
|
for(var/area/A in areas)
|
|
assign_area_network_id(A, M)
|
|
|
|
/**
|
|
* Converts a list of string's into a full network_id
|
|
*
|
|
* Converts a list of individual branches into a proper network id. Validates
|
|
* individual parts to make sure they are clean.
|
|
*
|
|
* ex. list("A","B","C") -> A.B.C
|
|
*
|
|
* Arguments:
|
|
* * tree - List of strings
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/network_list_to_string(list/tree)
|
|
#ifdef DEBUG_NETWORKS
|
|
ASSERT(tree && tree.len > 0) // this should be obvious but JUST in case.
|
|
for(var/part in tree)
|
|
if(!verify_network_name(part) || findtext(name,".")!=0) // and no stray dots
|
|
stack_trace("network_list_to_string: Cannot create network with ([part]) of ([tree.Join(".")])")
|
|
break
|
|
#endif
|
|
return tree.Join(".")
|
|
|
|
/**
|
|
* Converts string into a list of network branches
|
|
*
|
|
* Converts a a proper network id into a list of the individual branches
|
|
*
|
|
* ex. A.B.C -> list("A","B","C")
|
|
*
|
|
* Arguments:
|
|
* * tree - List of strings
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/network_string_to_list(name)
|
|
#ifdef DEBUG_NETWORKS
|
|
if(!verify_network_name(name))
|
|
stack_trace("network_string_to_list: [name] IS INVALID")
|
|
#endif
|
|
return splittext(name,".") // should we do a splittext_char? I doubt we really need unicode in network names
|
|
|
|
|
|
/**
|
|
* Hard creates a network. Helper function for create_network_simple and create_network
|
|
*
|
|
* Hard creates a using a list of branches and returns. No error checking as it should
|
|
* of been done before this call
|
|
*
|
|
* Arguments:
|
|
* * network_tree - list,text List of branches of network
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/_hard_create_network(list/network_tree)
|
|
var/network_name_part = network_tree[1]
|
|
var/network_id = network_name_part
|
|
var/datum/ntnet/parent = root_networks[network_name_part]
|
|
var/datum/ntnet/network
|
|
if(!parent) // we have no network root? Must be mapload of a ruin or such
|
|
parent = new(network_name_part)
|
|
|
|
// go up the branches, creating nodes
|
|
for(var/i in 2 to network_tree.len)
|
|
network_name_part = network_tree[i]
|
|
network = parent.children[network_name_part]
|
|
network_id += "." + network_name_part
|
|
if(!network)
|
|
network = new(network_id, network_name_part, parent)
|
|
|
|
parent = network
|
|
return network
|
|
|
|
|
|
/**
|
|
* Creates or finds a network anywhere in the world using a fully qualified name
|
|
*
|
|
* This is the simple case finding of a network in the world. It must take a full
|
|
* qualified network name and it will either return an existing network or build
|
|
* a new one from scratch. We must be able to create names on the fly as there is
|
|
* no way for the map loader to tell us ahead of time what networks to create or
|
|
* use for any maps or templates. So this thing will throw silent mapping errors
|
|
* and log them, but will always return a network for something.
|
|
*
|
|
* Arguments:
|
|
* * network_id - text, Fully qualified network name
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/create_network_simple(network_id)
|
|
|
|
var/datum/ntnet/network = networks[network_id]
|
|
if(network!=null)
|
|
return network // don't worry about it
|
|
|
|
/// Checks to make sure the network is valid. We log BOTH to mapping and telecoms
|
|
/// so if your checking for network errors you can find it in mapping to (because its their fault!)
|
|
if(!verify_network_name(network_id))
|
|
log_mapping("create_network_simple: [network_id] IS INVALID, replacing with LIMBO")
|
|
log_telecomms("create_network_simple: [network_id] IS INVALID, replacing with LIMBO")
|
|
return networks[LIMBO_NETWORK_ROOT]
|
|
|
|
var/list/network_tree = network_string_to_list(network_id)
|
|
if(!network_tree || network_tree.len == 0)
|
|
log_mapping("create_network_simple: [network_id] IS INVALID, replacing with LIMBO")
|
|
log_telecomms("create_network_simple: [network_id] IS INVALID, replacing with LIMBO")
|
|
return networks[LIMBO_NETWORK_ROOT]
|
|
|
|
network = _hard_create_network(network_tree)
|
|
#ifdef DEBUG_NETWORKS
|
|
if(!network)
|
|
CRASH("NETWORK CANNOT BE NULL")
|
|
#endif
|
|
log_telecomms("create_network_simple: created final [network.network_id]")
|
|
return network // and we are done!
|
|
|
|
|
|
/**
|
|
* Creates or finds a network anywhere in the world using bits of text
|
|
*
|
|
* This works the same as create_network_simple however it allows the addition
|
|
* of qualified network names. So you can call it with a root_id and a sub
|
|
* network. However this function WILL return null if it cannot be created
|
|
* so it should be used with error checking is involved.
|
|
*
|
|
* ex. create_network("ROOT_NETWORK", "ATMOS.SCRUBBERS") -> ROOT_NETWORK.ATMOS.SCRUBBERS
|
|
*
|
|
* Arguments:
|
|
* * tree - List of string
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/create_network(...)
|
|
var/list/network_tree = list()
|
|
|
|
for(var/i in 1 to args.len)
|
|
var/part = args[i]
|
|
#ifdef DEBUG_NETWORKS
|
|
if(!part || !istext(part))
|
|
/// stack trace here because this is a bad error
|
|
stack_trace("create_network: We only take text on [part] index [i]")
|
|
return null
|
|
#endif
|
|
network_tree += network_string_to_list(part)
|
|
|
|
var/datum/ntnet/network = _hard_create_network(network_tree)
|
|
log_telecomms("create_network: created final [network.network_id]")
|
|
return network
|
|
|
|
/**
|
|
* Generate a hardware id for devices.
|
|
*
|
|
* Creates a 32 bit hardware id for network devices. This is random so masking
|
|
* the number won't make routing "easier" (Think Ethernet) It does check if
|
|
* an existing device has the number but will NOT assign it as thats
|
|
* up to the collar
|
|
*
|
|
* Returns (string) The generated name
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/get_next_HID()
|
|
do
|
|
var/string = md5("[num2text(rand(HID_RESTRICTED_END, 999999999), 12)]")
|
|
if(!string)
|
|
log_runtime("Could not generagea m5 hash from address, problem with md5?")
|
|
return //errored
|
|
. = "[copytext_char(string, 1, 9)]" //16 ^ 8 possibilities I think.
|
|
while(interfaces_by_hardware_id[.])
|
|
|
|
/**
|
|
* Generate a name devices
|
|
*
|
|
* Creates a randomly generated tag or name for devices or anything really
|
|
* it keeps track of a special list that makes sure no name is used more than
|
|
* once
|
|
*
|
|
* args:
|
|
* * len (int)(Optional) Default=5 The length of the name
|
|
* * prefix (string)(Optional) static text in front of the random name
|
|
* * postfix (string)(Optional) static text in back of the random name
|
|
* Returns (string) The generated name
|
|
*/
|
|
/datum/controller/subsystem/networks/proc/assign_random_name(len=5, prefix="", postfix="")
|
|
var/static/valid_chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
|
|
var/list/new_name = list()
|
|
var/text
|
|
// machine id's should be fun random chars hinting at a larger world
|
|
do
|
|
new_name.Cut()
|
|
new_name += prefix
|
|
for(var/i = 1 to len)
|
|
new_name += valid_chars[rand(1,length(valid_chars))]
|
|
new_name += postfix
|
|
text = new_name.Join()
|
|
while(used_names[text])
|
|
used_names[text] = TRUE
|
|
return text
|