2022-06-04 18:09:42 +08:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Text ;
using System.Threading.Tasks ;
using Lidgren.Network ;
using RageCoop.Core ;
2022-06-23 09:46:38 +08:00
using RageCoop.Core.Scripting ;
2022-06-24 16:49:59 +08:00
using System.Reflection ;
2022-06-04 18:09:42 +08:00
using System.Net ;
namespace RageCoop.Server.Scripting
{
2022-06-23 14:10:16 +08:00
public class APIEvents
2022-06-04 18:09:42 +08:00
{
2022-07-01 12:22:31 +08:00
private readonly Server Server ;
internal APIEvents ( Server server )
{
Server = server ;
}
2022-06-19 11:12:20 +08:00
#region INTERNAL
2022-06-23 14:10:16 +08:00
internal Dictionary < int , List < Action < CustomEventReceivedArgs > > > CustomEventHandlers = new ( ) ;
2022-06-19 11:12:20 +08:00
#endregion
2022-06-23 14:10:16 +08:00
public event EventHandler < ChatEventArgs > OnChatMessage ;
2022-06-30 09:28:13 +08:00
/// <summary>
/// Will be invoked from main thread before registered handlers
/// </summary>
public event EventHandler < OnCommandEventArgs > OnCommandReceived ;
/// <summary>
/// Will be invoked from main thread when a client is attempting to connect, use <see cref="HandshakeEventArgs.Deny(string)"/> to deny the connection request.
/// </summary>
2022-06-23 14:10:16 +08:00
public event EventHandler < HandshakeEventArgs > OnPlayerHandshake ;
/// <summary>
/// Will be invoked when a player is connected, but this player might not be ready yet(client resources not loaded), using <see cref="OnPlayerReady"/> is recommended.
/// </summary>
public event EventHandler < Client > OnPlayerConnected ;
/// <summary>
/// Will be invoked after the client connected and all resources(if any) have been loaded.
/// </summary>
public event EventHandler < Client > OnPlayerReady ;
public event EventHandler < Client > OnPlayerDisconnected ;
/// <summary>
/// Invoked everytime a player's main ped has been updated
/// </summary>
public event EventHandler < Client > OnPlayerUpdate ;
internal void ClearHandlers ( )
2022-06-04 18:09:42 +08:00
{
2022-06-23 14:10:16 +08:00
OnChatMessage = null ;
OnPlayerHandshake = null ;
OnPlayerConnected = null ;
OnPlayerReady = null ;
OnPlayerDisconnected = null ;
// OnCustomEventReceived=null;
OnCommandReceived = null ;
OnPlayerUpdate = null ;
}
#region INVOKE
2022-06-30 09:28:13 +08:00
internal void InvokePlayerHandshake ( HandshakeEventArgs args )
{ OnPlayerHandshake ? . Invoke ( this , args ) ; }
2022-07-01 12:22:31 +08:00
internal void InvokeOnCommandReceived ( string cmdName , string [ ] cmdArgs , Client sender )
2022-06-30 09:28:13 +08:00
{
var args = new OnCommandEventArgs ( )
{
2022-07-01 12:22:31 +08:00
Name = cmdName ,
Args = cmdArgs ,
2022-06-30 09:28:13 +08:00
Sender = sender
} ;
OnCommandReceived ? . Invoke ( this , args ) ;
if ( args . Cancel )
{
return ;
}
2022-07-01 12:22:31 +08:00
if ( Server . Commands . Any ( x = > x . Key . Name = = cmdName ) )
2022-06-30 09:28:13 +08:00
{
string [ ] argsWithoutCmd = cmdArgs . Skip ( 1 ) . ToArray ( ) ;
CommandContext ctx = new ( )
{
Client = sender ,
Args = argsWithoutCmd
} ;
2022-07-01 12:22:31 +08:00
KeyValuePair < Command , Action < CommandContext > > command = Server . Commands . First ( x = > x . Key . Name = = cmdName ) ;
2022-06-30 09:28:13 +08:00
command . Value . Invoke ( ctx ) ;
}
else
{
2022-07-01 12:22:31 +08:00
Server . SendChatMessage ( "Server" , "Command not found!" , sender . Connection ) ;
2022-06-30 09:28:13 +08:00
}
}
2022-06-23 14:10:16 +08:00
internal void InvokeOnChatMessage ( Packets . ChatMessage p , Client sender )
{
OnChatMessage ? . Invoke ( this , new ChatEventArgs ( )
2022-06-11 18:41:10 +08:00
{
2022-06-21 18:13:30 +08:00
Sender = sender ,
2022-06-04 18:09:42 +08:00
Message = p . Message
2022-06-23 14:10:16 +08:00
} ) ;
}
internal void InvokePlayerConnected ( Client client )
{ OnPlayerConnected ? . Invoke ( this , client ) ; }
internal void InvokePlayerReady ( Client client )
{ OnPlayerReady ? . Invoke ( this , client ) ; }
internal void InvokePlayerDisconnected ( Client client )
{ OnPlayerDisconnected ? . Invoke ( this , client ) ; }
2022-06-21 18:13:30 +08:00
2022-06-23 14:10:16 +08:00
internal void InvokeCustomEventReceived ( Packets . CustomEvent p , Client sender )
{
var args = new CustomEventReceivedArgs ( ) { Hash = p . Hash , Args = p . Args , Sender = sender } ;
List < Action < CustomEventReceivedArgs > > handlers ;
if ( CustomEventHandlers . TryGetValue ( p . Hash , out handlers ) )
2022-06-21 18:13:30 +08:00
{
2022-06-23 14:10:16 +08:00
handlers . ForEach ( ( x ) = > { x . Invoke ( args ) ; } ) ;
2022-06-21 18:13:30 +08:00
}
2022-06-23 14:10:16 +08:00
}
internal void InvokePlayerUpdate ( Client client )
{
OnPlayerUpdate ? . Invoke ( this , client ) ;
}
#endregion
}
public class API
{
private readonly Server Server ;
internal API ( Server server )
{
Server = server ;
2022-07-01 12:22:31 +08:00
Events = new ( server ) ;
2022-06-23 14:10:16 +08:00
}
2022-07-01 12:22:31 +08:00
public readonly APIEvents Events ;
2022-06-04 18:09:42 +08:00
#region FUNCTIONS
2022-06-23 09:46:38 +08:00
/ *
2022-06-04 18:09:42 +08:00
/// <summary>
/// Send a native call (Function.Call) to all players.
/// Keys = int, float, bool, string and lvector3
/// </summary>
/// <param name="hash">The hash (Example: 0x25223CA6B4D20B7F = GET_CLOCK_HOURS)</param>
/// <param name="args">The arguments (Example: string = int, object = 5)</param>
2022-06-23 14:10:16 +08:00
public void SendNativeCallToAll ( GTA . Native . Hash hash , params object [ ] args )
2022-06-04 18:09:42 +08:00
{
try
{
if ( Server . MainNetServer . ConnectionsCount = = 0 )
{
return ;
}
if ( args ! = null & & args . Length = = 0 )
{
2022-06-23 14:10:16 +08:00
Server . Logger ? . Error ( $"[ServerScript->SendNativeCallToAll(ulong hash, params object[] args)]: args is not null!" ) ;
2022-06-04 18:09:42 +08:00
return ;
}
Packets . NativeCall packet = new ( )
{
Hash = ( ulong ) hash ,
Args = new List < object > ( args ) ? ? new List < object > ( )
} ;
NetOutgoingMessage outgoingMessage = Server . MainNetServer . CreateMessage ( ) ;
packet . Pack ( outgoingMessage ) ;
Server . MainNetServer . SendMessage ( outgoingMessage , Server . MainNetServer . Connections , NetDeliveryMethod . ReliableOrdered , ( byte ) ConnectionChannel . Native ) ;
}
catch ( Exception e )
{
2022-06-23 14:10:16 +08:00
Server . Logger ? . Error ( $">> {e.Message} <<>> {e.Source ?? string.Empty} <<>> {e.StackTrace ?? string.Empty} <<" ) ;
2022-06-04 18:09:42 +08:00
}
}
2022-06-23 09:46:38 +08:00
* /
2022-06-04 18:09:42 +08:00
/// <summary>
/// Get a list of all Clients
/// </summary>
/// <returns>All clients as a dictionary indexed by NetID</returns>
2022-06-23 14:10:16 +08:00
public Dictionary < long , Client > GetAllClients ( )
2022-06-04 18:09:42 +08:00
{
2022-06-23 09:46:38 +08:00
return new ( Server . Clients ) ;
2022-06-04 18:09:42 +08:00
}
/// <summary>
/// Get the client by its username
/// </summary>
/// <param name="username">The username to search for (non case-sensitive)</param>
/// <returns>The Client from this user or null</returns>
2022-06-23 14:10:16 +08:00
public Client GetClientByUsername ( string username )
2022-06-04 18:09:42 +08:00
{
2022-06-21 18:13:30 +08:00
return Server . Clients . Values . FirstOrDefault ( x = > x . Username . ToLower ( ) = = username . ToLower ( ) ) ;
2022-06-04 18:09:42 +08:00
}
/// <summary>
/// Send a chat message to all players
/// </summary>
/// <param name="message">The chat message</param>
/// <param name="username">The username which send this message (default = "Server")</param>
2022-06-23 14:10:16 +08:00
public void SendChatMessage ( string message , List < Client > targets = null , string username = "Server" )
2022-06-04 18:09:42 +08:00
{
try
{
if ( Server . MainNetServer . ConnectionsCount = = 0 )
{
return ;
}
2022-06-21 18:13:30 +08:00
targets ? ? = new ( Server . Clients . Values ) ;
foreach ( Client client in targets )
{
Server . SendChatMessage ( username , message , client . Connection ) ;
}
}
catch ( Exception e )
{
2022-06-23 14:10:16 +08:00
Server . Logger ? . Error ( $">> {e.Message} <<>> {e.Source ?? string.Empty} <<>> {e.StackTrace ?? string.Empty} <<" ) ;
2022-06-21 18:13:30 +08:00
}
}
2022-06-23 14:10:16 +08:00
public void SendChatMessage ( string message , Client target , string username = "Server" )
2022-06-21 18:13:30 +08:00
{
try
{
Server . SendChatMessage ( username , message , target . Connection ) ;
2022-06-04 18:09:42 +08:00
}
catch ( Exception e )
{
2022-06-23 14:10:16 +08:00
Server . Logger ? . Error ( $">> {e.Message} <<>> {e.Source ?? string.Empty} <<>> {e.StackTrace ?? string.Empty} <<" ) ;
2022-06-04 18:09:42 +08:00
}
}
/// <summary>
/// Send CleanUpWorld to all players to delete all objects created by the server
/// </summary>
/// <summary>
/// Register a new command chat command (Example: "/test")
/// </summary>
/// <param name="name">The name of the command (Example: "test" for "/test")</param>
/// <param name="usage">How to use this message (argsLength required!)</param>
/// <param name="argsLength">The length of args (Example: "/message USERNAME MESSAGE" = 2) (usage required!)</param>
/// <param name="callback">Create a new function!</param>
2022-06-23 14:10:16 +08:00
public void RegisterCommand ( string name , string usage , short argsLength , Action < CommandContext > callback )
2022-06-04 18:09:42 +08:00
{
Server . RegisterCommand ( name , usage , argsLength , callback ) ;
}
/// <summary>
/// Register a new command chat command (Example: "/test")
/// </summary>
/// <param name="name">The name of the command (Example: "test" for "/test")</param>
/// <param name="callback">Create a new function!</param>
2022-06-23 14:10:16 +08:00
public void RegisterCommand ( string name , Action < CommandContext > callback )
2022-06-04 18:09:42 +08:00
{
Server . RegisterCommand ( name , callback ) ;
}
/// <summary>
2022-06-24 16:49:59 +08:00
/// Register all commands in a static class
2022-06-04 18:09:42 +08:00
/// </summary>
2022-06-24 16:49:59 +08:00
/// <typeparam name="T">Your static class with commands</typeparam>
2022-06-23 14:10:16 +08:00
public void RegisterCommands < T > ( )
2022-06-04 18:09:42 +08:00
{
Server . RegisterCommands < T > ( ) ;
}
2022-06-24 16:49:59 +08:00
/// <summary>
/// Register all commands inside an class instance
/// </summary>
/// <param name="obj">The instance of type containing the commands</param>
public void RegisterCommands ( object obj )
{
IEnumerable < MethodInfo > commands = obj . GetType ( ) . GetMethods ( ) . Where ( method = > method . GetCustomAttributes ( typeof ( Command ) , false ) . Any ( ) ) ;
foreach ( MethodInfo method in commands )
{
Command attribute = method . GetCustomAttribute < Command > ( true ) ;
RegisterCommand ( attribute . Name , attribute . Usage , attribute . ArgsLength ,
( ctx ) = > { method . Invoke ( obj , new object [ ] { ctx } ) ; } ) ;
}
}
2022-06-21 18:13:30 +08:00
/// <summary>
2022-06-23 14:10:16 +08:00
/// Send an event and data to the specified clients. Use <see cref="Client.SendCustomEvent(int, List{object})"/> if you want to send event to individual client.
2022-06-21 18:13:30 +08:00
/// </summary>
/// <param name="eventHash">An unique identifier of the event</param>
/// <param name="args">The objects conataing your data, supported types: byte, short, ushort, int, uint, long, ulong, float, bool, string.</param>
2022-06-23 14:10:16 +08:00
/// <param name="targets">The target clients to send. Leave it null to send to all clients</param>
2022-06-24 10:33:36 +08:00
public void SendCustomEvent ( int eventHash , List < object > args = null , List < Client > targets = null )
2022-06-12 15:39:32 +08:00
{
2022-06-19 11:12:20 +08:00
targets ? ? = new ( Server . Clients . Values ) ;
var p = new Packets . CustomEvent ( )
2022-06-12 15:39:32 +08:00
{
2022-06-19 11:12:20 +08:00
Args = args ,
Hash = eventHash
} ;
foreach ( var c in targets )
2022-06-12 15:39:32 +08:00
{
2022-06-19 11:12:20 +08:00
Server . Send ( p , c , ConnectionChannel . Event , NetDeliveryMethod . ReliableOrdered ) ;
2022-06-12 15:39:32 +08:00
}
2022-06-04 18:09:42 +08:00
}
2022-06-22 14:18:20 +08:00
/// <summary>
/// Register an handler to the specifed event hash, one event can have multiple handlers.
/// </summary>
/// <param name="hash">An unique identifier of the event, you can hash your event name with <see cref="Core.Scripting.CustomEvents.Hash(string)"/></param>
/// <param name="handler">An handler to be invoked when the event is received from the server. This will be invoked from main thread.
2022-06-23 14:10:16 +08:00
public void RegisterCustomEventHandler ( int hash , Action < CustomEventReceivedArgs > handler )
2022-06-21 18:13:30 +08:00
{
2022-06-22 14:18:20 +08:00
List < Action < CustomEventReceivedArgs > > handlers ;
2022-06-23 14:10:16 +08:00
lock ( Events . CustomEventHandlers )
2022-06-21 18:13:30 +08:00
{
2022-06-23 14:10:16 +08:00
if ( ! Events . CustomEventHandlers . TryGetValue ( hash , out handlers ) )
2022-06-21 18:13:30 +08:00
{
2022-06-23 14:10:16 +08:00
Events . CustomEventHandlers . Add ( hash , handlers = new List < Action < CustomEventReceivedArgs > > ( ) ) ;
2022-06-21 18:13:30 +08:00
}
handlers . Add ( handler ) ;
}
}
2022-06-23 14:10:16 +08:00
public void RegisterCustomEventHandler ( string name , Action < CustomEventReceivedArgs > handler )
2022-06-23 09:46:38 +08:00
{
RegisterCustomEventHandler ( CustomEvents . Hash ( name ) , handler ) ;
}
2022-06-23 14:10:16 +08:00
public Logger GetLogger ( )
2022-06-21 18:13:30 +08:00
{
2022-06-23 14:10:16 +08:00
return Server . Logger ;
2022-06-21 18:13:30 +08:00
}
2022-06-04 18:09:42 +08:00
#endregion
}
}