146 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
		
		
			
		
	
	
			146 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 |