145 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| --[[
 | |
|     https://github.com/overextended/ox_lib
 | |
| 
 | |
|     This file is licensed under LGPL-3.0 or higher <https://www.gnu.org/licenses/lgpl-3.0.en.html>
 | |
| 
 | |
|     Copyright © 2025 Linden <https://github.com/thelindat>
 | |
| ]]
 | |
| 
 | |
| local pendingCallbacks = {}
 | |
| local timers = {}
 | |
| local cbEvent = '__ox_cb_%s'
 | |
| local callbackTimeout = GetConvarInt('ox:callbackTimeout', 300000)
 | |
| 
 | |
| RegisterNetEvent(cbEvent:format(cache.resource), function(key, ...)
 | |
|     if source == '' then return end
 | |
| 
 | |
|     local cb = pendingCallbacks[key]
 | |
| 
 | |
|     if not cb then return end
 | |
| 
 | |
|     pendingCallbacks[key] = nil
 | |
| 
 | |
|     cb(...)
 | |
| end)
 | |
| 
 | |
| ---@param event string
 | |
| ---@param delay? number | false prevent the event from being called for the given time
 | |
| local function eventTimer(event, delay)
 | |
|     if delay and type(delay) == 'number' and delay > 0 then
 | |
|         local time = GetGameTimer()
 | |
| 
 | |
|         if (timers[event] or 0) > time then
 | |
|             return false
 | |
|         end
 | |
| 
 | |
|         timers[event] = time + delay
 | |
|     end
 | |
| 
 | |
|     return true
 | |
| end
 | |
| 
 | |
| ---@param _ any
 | |
| ---@param event string
 | |
| ---@param delay number | false | nil
 | |
| ---@param cb function | false
 | |
| ---@param ... any
 | |
| ---@return ...
 | |
| local function triggerServerCallback(_, event, delay, cb, ...)
 | |
|     if not eventTimer(event, delay) then return end
 | |
| 
 | |
|     local key
 | |
| 
 | |
|     repeat
 | |
|         key = ('%s:%s'):format(event, math.random(0, 100000))
 | |
|     until not pendingCallbacks[key]
 | |
| 
 | |
|     TriggerServerEvent('ox_lib:validateCallback', event, cache.resource, key)
 | |
|     TriggerServerEvent(cbEvent:format(event), cache.resource, key, ...)
 | |
| 
 | |
|     ---@type promise | false
 | |
|     local promise = not cb and promise.new()
 | |
| 
 | |
|     pendingCallbacks[key] = function(response, ...)
 | |
|         if response == 'cb_invalid' then
 | |
|             response = ("callback '%s' does not exist"):format(event)
 | |
| 
 | |
|             return promise and promise:reject(response) or error(response)
 | |
|         end
 | |
| 
 | |
|         response = { response, ... }
 | |
| 
 | |
|         if promise then
 | |
|             return promise:resolve(response)
 | |
|         end
 | |
| 
 | |
|         if cb then
 | |
|             cb(table.unpack(response))
 | |
|         end
 | |
|     end
 | |
| 
 | |
|     if promise then
 | |
|         SetTimeout(callbackTimeout, function() promise:reject(("callback event '%s' timed out"):format(key)) end)
 | |
| 
 | |
|         return table.unpack(Citizen.Await(promise))
 | |
|     end
 | |
| end
 | |
| 
 | |
| ---@overload fun(event: string, delay: number | false, cb: function, ...)
 | |
| lib.callback = setmetatable({}, {
 | |
|     __call = function(_, event, delay, cb, ...)
 | |
|         if not cb then
 | |
|             warn(("callback event '%s' does not have a function to callback to and will instead await\nuse lib.callback.await or a regular event to remove this warning")
 | |
|                 :format(event))
 | |
|         else
 | |
|             local cbType = type(cb)
 | |
| 
 | |
|             if cbType == 'table' and getmetatable(cb)?.__call then
 | |
|                 cbType = 'function'
 | |
|             end
 | |
| 
 | |
|             assert(cbType == 'function', ("expected argument 3 to have type 'function' (received %s)"):format(cbType))
 | |
|         end
 | |
| 
 | |
|         return triggerServerCallback(_, event, delay, cb, ...)
 | |
|     end
 | |
| })
 | |
| 
 | |
| ---@param event string
 | |
| ---@param delay? number | false prevent the event from being called for the given time.
 | |
| ---Sends an event to the server and halts the current thread until a response is returned.
 | |
| ---@diagnostic disable-next-line: duplicate-set-field
 | |
| function lib.callback.await(event, delay, ...)
 | |
|     return triggerServerCallback(nil, event, delay, false, ...)
 | |
| end
 | |
| 
 | |
| local function callbackResponse(success, result, ...)
 | |
|     if not success then
 | |
|         if result then
 | |
|             return print(('^1SCRIPT ERROR: %s^0\n%s'):format(result,
 | |
|                 Citizen.InvokeNative(`FORMAT_STACK_TRACE` & 0xFFFFFFFF, nil, 0, Citizen.ResultAsString()) or ''))
 | |
|         end
 | |
| 
 | |
|         return false
 | |
|     end
 | |
| 
 | |
|     return result, ...
 | |
| end
 | |
| 
 | |
| local pcall = pcall
 | |
| 
 | |
| ---@param name string
 | |
| ---@param cb function
 | |
| ---Registers an event handler and callback function to respond to server requests.
 | |
| ---@diagnostic disable-next-line: duplicate-set-field
 | |
| function lib.callback.register(name, cb)
 | |
|     event = cbEvent:format(name)
 | |
| 
 | |
|     lib.setValidCallback(name, true)
 | |
| 
 | |
|     RegisterNetEvent(event, function(resource, key, ...)
 | |
|         TriggerServerEvent(cbEvent:format(resource), key, callbackResponse(pcall(cb, ...)))
 | |
|     end)
 | |
| end
 | |
| 
 | |
| return lib.callback
 | 
