Skip to content

Cryotheus/entity_proxy

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

26 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Networking Entity Proxy Module

Magical code that makes networking entities (more) reliable. Workshop link.
Both realms can create proxies, so you can just to hold a reliable reference to an invalid entity. Anything marked with INTERNAL shouldn't be used unless you know what you're doing.

Why?

You may want this so you can:

  • Reliably network entities by fixing race conditions between entity networking and net messages.
  • Maintain information about an entity after it has been removed.
  • Hold references to entities that will never exist (eg. server side entities on client).

Module

The module name is entity_proxy and for the most part, you should only need to use entity_proxy.Read, entity_proxy.Write, and entity_proxy.Destroy. The original aim of this module was to easily fix existing projects that network entities over net messages improperly. This aim is still valid, but this module can also be used to better manage invalid entities.

Module Fields

Safe to read, not so safe to write.

Field Type Description
Proxies table INTERNAL Contains all the namespaces' tables of live proxies
WaitingProxies table INTERNAL Contains all the namespaces' tables of proxies waiting for an entity to be received

Module Functions

It is safe to assume that any:entity_index can be a number, Entity, or EntityProxy.
The default timeout is 8 (seconds) on the client, and false on the server meaning the server won't wait for the entity to be created.

Function Parameters Returns Description
Clear string:namespace Calls DestroyInternal on all proxies in the namespace
Create string:namespace any:entity_index number/boolean:timeout EntityProxy Calls CreateInternal after converting entity_indexto a number
CreateInternal string:namespace number:entity_index number/boolean:timeout EntityProxy INTERNAL Creates and register a proxy (timeout will use default if nil)
Destroy string:namespace any:entity_index Calls DestroyInternal after converting entity_indexto a number
DestroyInternal string:namespace number:entity_index INTERNAL Destroys the entity proxy dropping it from the Proxies table
Get string:namespace any:entity_index number/boolean:timeout EntityProxy Returns an existing proxy or creates one
GetExisting string:namespace any:entity_index EntityProxy Returns an existing proxy or nil if it doesn't exist
Read string:namespace number/boolean:timeout EntityProxy Reads 13 bits and calls Get
Register string:namespace Creates the tables required to use this module with the given namespace
ToString EntityProxy string Creates a string with useful debug information about the proxy
Write any:entity_index Writes a 13-bit unsigned integer after converting entity_index to a number

Example

--don't use require as clients can exploit that to load binary modules instead
--it is safe to include the module across multiple files as it only loads once
include("includes/entity_proxy.lua")
util.AddNetworkString("SomeNetMessage")

if SERVER then
	--grabbing a random entity for this example
	local entities = ents.GetAll()
	local some_entity = entities[math.random(#entities)]
	
	net.Start("SomeNetMessage")
	entity_proxy.Write(some_entity) --use this instead of net.WriteEntity
	net.Broadcast()
else --the CLIENT
	net.Receive("SomeNetMessage", function()
		local entity_proxy = entity_proxy.Read() --use this instead of net.ReadEntity
		
		print("received an entity proxy", entity_proxy)
		
		if entity_proxy:IsValid() then
			print("since the entity was valid when it was received, we can immediately use it!")
			print(entity_proxy:EntIndex(), entity_proxy:GetPos(), entity_proxy:GetClass())
		else
			print("the entity was not valid immediately, thus the module will wait for it to be created")
			
			--if you want to do something immediately after the entity is created, you can use the OnEntityProxyReceived method
			function entity_proxy:OnEntityProxyReceived(entity)
				--please note that this will most likely not be called in this example
				--this is more useful for entities created on the server at a similar time to when the net message is sent
				--but in this example we are just grabbing random entities, which are most likely server side entities if they are not valid on the client
				print("the entity was created, it is now useful!")
				print("the proxy is `self`", self)
				print("and the unproxied Entity is `entity`", entity)
			end
			
			function entity_proxy:OnEntityProxyTimeout()
				--if you don't want the entity to timeout,
				print("the entity was not created in time, so the proxy timed out!", self)
			end
		end
	end)
end

EntityProxy

For the most part an proxy like an entity, but will become valid once the entity is created instead of staying invalid. Internally they are tables with a few useful methods and metamethods; this means isentity/IsEntity will return false, and istable will return true. Any fields you set on the proxy while the proxied entity is invalid will not be lost, but saved in a local table. The fields you set in this table will be moved over to the entity when they become valid. The opposite will happen when the entity is destroyed, the fields you previously set on the entity when it was invalid will be moved back to the proxy. If you want to ensure this behavior happens on a specific field, you can call EntityProxy:SaveEntityProxyField. If you don't want the proxy to save a field, you can call EntityProxy:DropEntityProxyField. You can also call EntityProxy:ClearEntityProxyFields to remove all fields from the proxy.
The meta table for proxies is not in the registry as each meta table is unique to the proxy (for upvalue access).

Fields

No need to set these, they are automatically set for you. Reading from them is safe.

Field Type Description
EntityProxyTimeout number:curtime Set to the curtime that the proxy timed out at (nil otherwise)
ProxyReceivedEntity boolean Set to true if the entity was received after creation (nil otherwise)
__name string Always set to EntityProxy

Methods

Method Parameters Returns Description
ClearEntityProxyFields Same as calling EntityProxy:DropEntityProxyField on every field
ClearProxiedEntityDetours Removed all existing entity method detours
DecrementEntityProxyReferenceCount number boolean:destroyed Subtract the reference_count local by number (defaults to 1) and calls entity_proxy.DestroyInternal at 0 or below
Destroy Calls entity_proxy.DestroyInternal on the proxy
DropEntityProxyField any:key The opposite of EntityProxy:SaveEntityProxyField, don't save this field when the entity becomes invalid
EntIndex number Sames as Entity:EntIndex but also works invalid entities
GetEntityProxyDetours table Returns the proxy_detours local table used by the proxy
GetEntityProxyNamespace string Returns the namespace the entity proxy belongs to (the one it was created with)
GetEntityProxyReferenceCount number Gets the reference_count local (0 on creation)
GetProxiedEntity Entity Self explanatory
IncrementEntityProxyReferenceCount number Add the reference_count local by number (defaults to 1)
IsEntityProxyAlive boolean Returns true if the proxy is still registered under the namespace (false otherwise)
OnEntityProxyDestroyed Called when the proxy is destroyed (with entity_proxy.DestroyInternal)
OnEntityProxyReceived entity Called when an entity with the matching index has been created (called after EntityProxy:SetProxiedEntity)
OnEntityProxyTimeout boolean Set to true or return true to call entity_proxy.DestroyInternal
OnProxiedEntityRemove boolean Set to true or return true to call entity_proxy.DestroyInternal, called when the entity is removed
RefreshEntityProxyTimer Restarts the timeout timer
SaveEntityProxyField any:key Force the proxy to save this field from the entity when it is removed, and does nothing if the entity is invalid
SetEntityProxyReferenceCount number boolean:destroyed Set the reference_count local to number and call entity_proxy.DestroyInternal at 0 or below
SetProxiedEntity entity INTERNAL Sets the entity upvalue, moves null safety values, and sets up removal callback
__index any:key any INTERNAL
__newindex any:key any:value INTERNAL Sets the value on the entity if it's valid, otherwise caches them until received
__tostring string Formats useful data about the entity proxy into a string for debugging

Safety

When the proxy is no longer needed, you should destroy it with entity_proxy.Destroy.
More advanced garbage collection can be performed using reference counting. For every instance that you start using the proxy, call EntityProxy:IncrementEntityProxyReferenceCount to indicate that the proxy is in use. When that instance no longer needs the proxy, call EntityProxy:DecrementEntityProxyReferenceCount to automatically call entity_proxy.DestroyInternal when the reference_count local reaches 0.

Memory Leak Consequences

  • The proxy will remain in the entity_proxy.Proxies[namespace] table until the server restarts.
    This is to ensure you get the same table reference every time you call entity_proxy.Get or entity_proxy.Read.
  • You may unintentionally reuse the proxy after the entity it was created for has been removed, which will cause the received entity to be invalid.
  • Other horrifying undefined behavior I have yet to discover.

About

Magical code that makes networking entities (more) reliable.

Topics

Resources

License

Stars

Watchers

Forks