mirror of
https://github.com/Facepunch/Facepunch.Steamworks.git
synced 2025-01-26 05:28:07 +03:00
GameServer boilerplate
This commit is contained in:
parent
c954ec0ffe
commit
458355b546
@ -4,6 +4,8 @@ using System.Linq;
|
|||||||
|
|
||||||
namespace Facepunch.Steamworks.Test
|
namespace Facepunch.Steamworks.Test
|
||||||
{
|
{
|
||||||
|
[DeploymentItem( "FacepunchSteamworksApi.dll" )]
|
||||||
|
[DeploymentItem( "steam_appid.txt" )]
|
||||||
[TestClass]
|
[TestClass]
|
||||||
public class Friends
|
public class Friends
|
||||||
{
|
{
|
||||||
|
@ -92,6 +92,7 @@
|
|||||||
<Compile Include="Client\Networking.cs" />
|
<Compile Include="Client\Networking.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
<Compile Include="Client\Friends.cs" />
|
<Compile Include="Client\Friends.cs" />
|
||||||
|
<Compile Include="Server\Server.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\Facepunch.Steamworks\Facepunch.Steamworks.csproj">
|
<ProjectReference Include="..\Facepunch.Steamworks\Facepunch.Steamworks.csproj">
|
||||||
|
23
Facepunch.Steamworks.Test/Server/Server.cs
Normal file
23
Facepunch.Steamworks.Test/Server/Server.cs
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
using System;
|
||||||
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||||
|
|
||||||
|
namespace Facepunch.Steamworks.Test
|
||||||
|
{
|
||||||
|
[DeploymentItem( "FacepunchSteamworksApi.dll" )]
|
||||||
|
[DeploymentItem( "steam_appid.txt" )]
|
||||||
|
[DeploymentItem( "tier0_s.dll" )]
|
||||||
|
[DeploymentItem( "vstdlib_s.dll" )]
|
||||||
|
[DeploymentItem( "steamclient.dll" )]
|
||||||
|
[TestClass]
|
||||||
|
public class Server
|
||||||
|
{
|
||||||
|
[TestMethod]
|
||||||
|
public void Init()
|
||||||
|
{
|
||||||
|
using ( var server = new Facepunch.Steamworks.Server( 252490, 0, 20216, 20816, 20817, false, "VersionString" ) )
|
||||||
|
{
|
||||||
|
Assert.IsTrue( server.Valid );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
BIN
Facepunch.Steamworks.Test/bin/Debug/steam_api.dll
Normal file
BIN
Facepunch.Steamworks.Test/bin/Debug/steam_api.dll
Normal file
Binary file not shown.
BIN
Facepunch.Steamworks.Test/bin/Debug/steamclient.dll
Normal file
BIN
Facepunch.Steamworks.Test/bin/Debug/steamclient.dll
Normal file
Binary file not shown.
BIN
Facepunch.Steamworks.Test/bin/Debug/tier0_s.dll
Normal file
BIN
Facepunch.Steamworks.Test/bin/Debug/tier0_s.dll
Normal file
Binary file not shown.
BIN
Facepunch.Steamworks.Test/bin/Debug/vstdlib_s.dll
Normal file
BIN
Facepunch.Steamworks.Test/bin/Debug/vstdlib_s.dll
Normal file
Binary file not shown.
@ -7,8 +7,6 @@ namespace Facepunch.Steamworks
|
|||||||
{
|
{
|
||||||
public partial class Client : IDisposable
|
public partial class Client : IDisposable
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
internal class Internal : IDisposable
|
internal class Internal : IDisposable
|
||||||
{
|
{
|
||||||
private uint _hpipe;
|
private uint _hpipe;
|
||||||
|
@ -131,8 +131,11 @@
|
|||||||
<Compile Include="Interop\ServerRules.cs" />
|
<Compile Include="Interop\ServerRules.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
<Compile Include="Interop\steam_api_interop.cs" />
|
<Compile Include="Interop\steam_api_interop.cs" />
|
||||||
|
<Compile Include="Server.cs" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<Folder Include="Server\" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup />
|
|
||||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<PostBuildEvent>
|
<PostBuildEvent>
|
||||||
|
@ -15,6 +15,28 @@ namespace Valve.Interop
|
|||||||
|
|
||||||
internal class NativeEntrypoints
|
internal class NativeEntrypoints
|
||||||
{
|
{
|
||||||
|
internal class Extended
|
||||||
|
{
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl )]
|
||||||
|
public static extern bool SteamInternal_GameServer_Init( uint unIP, ushort usSteamPort, ushort usGamePort, ushort usQueryPort, int eServerMode, string pchVersionString );
|
||||||
|
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl)]
|
||||||
|
internal static extern IntPtr SteamInternal_CreateInterface( string ver );
|
||||||
|
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl )]
|
||||||
|
internal static extern uint SteamGameServer_GetHSteamUser();
|
||||||
|
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl )]
|
||||||
|
internal static extern uint SteamGameServer_GetHSteamPipe();
|
||||||
|
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl )]
|
||||||
|
internal static extern void SteamGameServer_Shutdown();
|
||||||
|
|
||||||
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl )]
|
||||||
|
internal static extern void SteamGameServer_RunCallbacks();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SteamAPI_ISteamClient_CreateSteamPipe" )]
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SteamAPI_ISteamClient_CreateSteamPipe" )]
|
||||||
internal static extern uint SteamAPI_ISteamClient_CreateSteamPipe( IntPtr instancePtr );
|
internal static extern uint SteamAPI_ISteamClient_CreateSteamPipe( IntPtr instancePtr );
|
||||||
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SteamAPI_ISteamClient_BReleaseSteamPipe" )]
|
[DllImportAttribute( "FacepunchSteamworksApi", CallingConvention = CallingConvention.Cdecl, EntryPoint = "SteamAPI_ISteamClient_BReleaseSteamPipe" )]
|
||||||
@ -2416,7 +2438,7 @@ namespace Valve.Steamworks
|
|||||||
{
|
{
|
||||||
CheckIfUsable();
|
CheckIfUsable();
|
||||||
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamGameServer(m_pSteamClient,hSteamUser,hSteamPipe,pchVersion);
|
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamGameServer(m_pSteamClient,hSteamUser,hSteamPipe,pchVersion);
|
||||||
return (ISteamGameServer)Marshal.PtrToStructure( result, typeof( ISteamGameServer ) );
|
return new CSteamGameServer( result );
|
||||||
}
|
}
|
||||||
internal override void SetLocalIPBinding( uint unIP, char usPort )
|
internal override void SetLocalIPBinding( uint unIP, char usPort )
|
||||||
{
|
{
|
||||||
@ -2463,7 +2485,7 @@ namespace Valve.Steamworks
|
|||||||
{
|
{
|
||||||
CheckIfUsable();
|
CheckIfUsable();
|
||||||
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamGameServerStats(m_pSteamClient,hSteamuser,hSteamPipe,pchVersion);
|
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamGameServerStats(m_pSteamClient,hSteamuser,hSteamPipe,pchVersion);
|
||||||
return (ISteamGameServerStats)Marshal.PtrToStructure( result, typeof( ISteamGameServerStats ) );
|
return new CSteamGameServerStats( result );
|
||||||
}
|
}
|
||||||
internal override ISteamApps GetISteamApps( uint hSteamUser, uint hSteamPipe, string pchVersion )
|
internal override ISteamApps GetISteamApps( uint hSteamUser, uint hSteamPipe, string pchVersion )
|
||||||
{
|
{
|
||||||
@ -2510,7 +2532,7 @@ namespace Valve.Steamworks
|
|||||||
{
|
{
|
||||||
CheckIfUsable();
|
CheckIfUsable();
|
||||||
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamHTTP(m_pSteamClient,hSteamuser,hSteamPipe,pchVersion);
|
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamHTTP(m_pSteamClient,hSteamuser,hSteamPipe,pchVersion);
|
||||||
return (ISteamHTTP)Marshal.PtrToStructure( result, typeof( ISteamHTTP ) );
|
return new CSteamHTTP( result );
|
||||||
}
|
}
|
||||||
internal override ISteamUnifiedMessages GetISteamUnifiedMessages( uint hSteamuser, uint hSteamPipe, string pchVersion )
|
internal override ISteamUnifiedMessages GetISteamUnifiedMessages( uint hSteamuser, uint hSteamPipe, string pchVersion )
|
||||||
{
|
{
|
||||||
@ -2528,7 +2550,7 @@ namespace Valve.Steamworks
|
|||||||
{
|
{
|
||||||
CheckIfUsable();
|
CheckIfUsable();
|
||||||
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamUGC(m_pSteamClient,hSteamUser,hSteamPipe,pchVersion);
|
IntPtr result = NativeEntrypoints.SteamAPI_ISteamClient_GetISteamUGC(m_pSteamClient,hSteamUser,hSteamPipe,pchVersion);
|
||||||
return (ISteamUGC)Marshal.PtrToStructure( result, typeof( ISteamUGC ) );
|
return new CSteamUGC( result );
|
||||||
}
|
}
|
||||||
internal override ISteamAppList GetISteamAppList( uint hSteamUser, uint hSteamPipe, string pchVersion )
|
internal override ISteamAppList GetISteamAppList( uint hSteamUser, uint hSteamPipe, string pchVersion )
|
||||||
{
|
{
|
||||||
|
239
Facepunch.Steamworks/Server.cs
Normal file
239
Facepunch.Steamworks/Server.cs
Normal file
@ -0,0 +1,239 @@
|
|||||||
|
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
|
namespace Facepunch.Steamworks
|
||||||
|
{
|
||||||
|
public partial class Server : IDisposable
|
||||||
|
{
|
||||||
|
internal class Internal : IDisposable
|
||||||
|
{
|
||||||
|
internal Valve.Steamworks.ISteamClient client;
|
||||||
|
internal Valve.Steamworks.ISteamGameServer gameServer;
|
||||||
|
internal Valve.Steamworks.ISteamUtils utils;
|
||||||
|
internal Valve.Steamworks.ISteamNetworking networking;
|
||||||
|
internal Valve.Steamworks.ISteamGameServerStats stats;
|
||||||
|
internal Valve.Steamworks.ISteamHTTP http;
|
||||||
|
internal Valve.Steamworks.ISteamInventory inventory;
|
||||||
|
internal Valve.Steamworks.ISteamUGC ugc;
|
||||||
|
internal Valve.Steamworks.ISteamApps apps;
|
||||||
|
|
||||||
|
internal bool Init()
|
||||||
|
{
|
||||||
|
var user = Valve.Interop.NativeEntrypoints.Extended.SteamGameServer_GetHSteamUser();
|
||||||
|
var pipe = Valve.Interop.NativeEntrypoints.Extended.SteamGameServer_GetHSteamPipe();
|
||||||
|
if ( pipe == 0 )
|
||||||
|
return false;
|
||||||
|
|
||||||
|
var clientPtr = Valve.Interop.NativeEntrypoints.Extended.SteamInternal_CreateInterface( "SteamClient017" );
|
||||||
|
if ( clientPtr == IntPtr.Zero )
|
||||||
|
{
|
||||||
|
throw new System.Exception( "Steam Server: Couldn't load SteamClient017" );
|
||||||
|
}
|
||||||
|
|
||||||
|
client = new Valve.Steamworks.CSteamClient( clientPtr );
|
||||||
|
|
||||||
|
|
||||||
|
gameServer = client.GetISteamGameServer( user, pipe, "SteamGameServer012" );
|
||||||
|
|
||||||
|
if ( gameServer.GetIntPtr() == IntPtr.Zero )
|
||||||
|
{
|
||||||
|
gameServer = null;
|
||||||
|
throw new System.Exception( "Steam Server: Couldn't load SteamGameServer012" );
|
||||||
|
}
|
||||||
|
|
||||||
|
utils = client.GetISteamUtils( pipe, "SteamUtils008" );
|
||||||
|
networking = client.GetISteamNetworking( user, pipe, "SteamNetworking005" );
|
||||||
|
stats = client.GetISteamGameServerStats( user, pipe, "SteamGameServerStats001" );
|
||||||
|
http = client.GetISteamHTTP( user, pipe, "STEAMHTTP_INTERFACE_VERSION002" );
|
||||||
|
inventory = client.GetISteamInventory( user, pipe, "STEAMINVENTORY_INTERFACE_V001" );
|
||||||
|
ugc = client.GetISteamUGC( user, pipe, "STEAMUGC_INTERFACE_VERSION008" );
|
||||||
|
|
||||||
|
if ( ugc.GetIntPtr() == IntPtr.Zero )
|
||||||
|
throw new System.Exception( "Steam Server: Couldn't load STEAMUGC_INTERFACE_VERSION008" );
|
||||||
|
|
||||||
|
apps = client.GetISteamApps( user, pipe, "STEAMAPPS_INTERFACE_VERSION008" );
|
||||||
|
|
||||||
|
if ( apps.GetIntPtr() == IntPtr.Zero )
|
||||||
|
throw new System.Exception( "Steam Server: Couldn't load STEAMAPPS_INTERFACE_VERSION008" );
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if ( client != null )
|
||||||
|
{
|
||||||
|
client.BShutdownIfAllPipesClosed();
|
||||||
|
client = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal Internal native;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Current running program's AppId
|
||||||
|
/// </summary>
|
||||||
|
public uint AppId { get; private set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Current user's Username
|
||||||
|
/// </summary>
|
||||||
|
public string Username { get; private set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Current user's SteamId
|
||||||
|
/// </summary>
|
||||||
|
public ulong SteamId { get; private set; }
|
||||||
|
|
||||||
|
public enum MessageType : int
|
||||||
|
{
|
||||||
|
Message = 0,
|
||||||
|
Warning = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Called with a message from Steam
|
||||||
|
/// </summary>
|
||||||
|
public Action<MessageType, string> OnMessage;
|
||||||
|
|
||||||
|
public Server( uint appId, uint IpAddress, ushort SteamPort, ushort GamePort, ushort QueryPort, bool Secure, string VersionString )
|
||||||
|
{
|
||||||
|
if ( !Valve.Interop.NativeEntrypoints.Extended.SteamInternal_GameServer_Init( IpAddress, SteamPort, GamePort, QueryPort, Secure ? 3 : 2 , VersionString ) )
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Valve.Steamworks.SteamAPI.Init( appId );
|
||||||
|
|
||||||
|
native = new Internal();
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get other interfaces
|
||||||
|
//
|
||||||
|
if ( !native.Init() )
|
||||||
|
{
|
||||||
|
native.Dispose();
|
||||||
|
native = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Set up warning hook callback
|
||||||
|
//
|
||||||
|
SteamAPIWarningMessageHook ptr = InternalOnWarning;
|
||||||
|
// native.client.SetWarningMessageHook( Marshal.GetFunctionPointerForDelegate( ptr ) );
|
||||||
|
|
||||||
|
//
|
||||||
|
// Cache common, unchanging info
|
||||||
|
//
|
||||||
|
AppId = appId;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Initial settings
|
||||||
|
//
|
||||||
|
native.gameServer.EnableHeartbeats( true );
|
||||||
|
MaxPlayers = 32;
|
||||||
|
BotCount = 0;
|
||||||
|
MapName = "unset";
|
||||||
|
|
||||||
|
//
|
||||||
|
// Run update, first call does some initialization
|
||||||
|
//
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if ( native != null)
|
||||||
|
{
|
||||||
|
native.Dispose();
|
||||||
|
native = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[UnmanagedFunctionPointer( CallingConvention.Cdecl )]
|
||||||
|
public delegate void SteamAPIWarningMessageHook( int nSeverity, System.Text.StringBuilder pchDebugText );
|
||||||
|
|
||||||
|
private void InternalOnWarning( int nSeverity, System.Text.StringBuilder text )
|
||||||
|
{
|
||||||
|
if ( OnMessage != null )
|
||||||
|
{
|
||||||
|
OnMessage( ( MessageType)nSeverity, text.ToString() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal event Action OnUpdate;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Should be called at least once every frame
|
||||||
|
/// </summary>
|
||||||
|
public void Update()
|
||||||
|
{
|
||||||
|
if ( native == null )
|
||||||
|
return;
|
||||||
|
|
||||||
|
Valve.Interop.NativeEntrypoints.Extended.SteamGameServer_RunCallbacks();
|
||||||
|
// Voice.Update();
|
||||||
|
// Inventory.Update();
|
||||||
|
// Networking.Update();
|
||||||
|
|
||||||
|
if ( OnUpdate != null )
|
||||||
|
OnUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool Valid
|
||||||
|
{
|
||||||
|
get { return native != null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
internal Action InstallCallback( int type, Delegate action )
|
||||||
|
{
|
||||||
|
var del = Marshal.GetFunctionPointerForDelegate( action );
|
||||||
|
|
||||||
|
// var ptr = Marshal.GetFunctionPointerForDelegate( action );
|
||||||
|
// Valve.Steamworks.SteamAPI.RegisterCallback( del, type );
|
||||||
|
|
||||||
|
// Valve.Steamworks.SteamAPI.UnregisterCallback( del );
|
||||||
|
|
||||||
|
//return () => Valve.Steamworks.SteamAPI.UnregisterCallback( ptr );
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the current MaxPlayers.
|
||||||
|
/// This doesn't enforce any kind of limit, it just updates the master server.
|
||||||
|
/// </summary>
|
||||||
|
public int MaxPlayers
|
||||||
|
{
|
||||||
|
get { return _maxplayers; }
|
||||||
|
set { if ( _maxplayers == value ) return; native.gameServer.SetMaxPlayerCount( value ); _maxplayers = value; }
|
||||||
|
}
|
||||||
|
private int _maxplayers = 0;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the current BotCount.
|
||||||
|
/// This doesn't enforce any kind of limit, it just updates the master server.
|
||||||
|
/// </summary>
|
||||||
|
public int BotCount
|
||||||
|
{
|
||||||
|
get { return _botcount; }
|
||||||
|
set { if ( _botcount == value ) return; native.gameServer.SetBotPlayerCount( value ); _botcount = value; }
|
||||||
|
}
|
||||||
|
private int _botcount = 0;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the current Map Name.
|
||||||
|
/// </summary>
|
||||||
|
public string MapName
|
||||||
|
{
|
||||||
|
get { return _mapname; }
|
||||||
|
set { if ( _mapname == value ) return; native.gameServer.SetMapName( value ); _mapname = value; }
|
||||||
|
}
|
||||||
|
private string _mapname;
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user