SDK Wrapper (Beta)¶
Keep in mind this project is a Work In Progress, with many features and improvements to come.
Please leave your feedback.
SDK Wrapper is not endorsed or sponsored by Valve.
Introduction¶
As you know from the previous article, there are some essential differences between GOG and other platforms, including Steam. With this tool we aim to decrease the time needed to implement additional features on GOG if you already have a working Steam build (with other platforms planned for future development).
If you already have a Steam version of your product (SDK Wrapper is intended to be used only after you have created your game’s Steam build), you can get it up and running on our platform within minutes, not hours. No changes to the code need to be made — all you need is to use our SDK Wrapper (Beta).
Galaxy SDK API is still available alongside SDK Wrapper functionality so it's possible to create custom Galaxy SDK init but leave other functionality like stats or leaderboards to SDK Wrapper.
SDK Wrapper (Beta) is a middle layer that provides interoperability between Steam and GOG Galaxy API calls and speeds up the process of developing GOG builds by translating Steam API calls included in a build already created with Steam in mind into calls that can be understood by the GOG backends.
Not all SDK features are supported yet — and, obviously, some will never be — but basic functionality is preserved. Currently, SDK Wrapper (Beta) allows to use the following features out of the box:
- achievements,
- leaderboards,
- stats,
- friends,
- matchmaking
SDK Wrapper is not endorsed or sponsored by Valve.
Offline mode¶
Games on GOG being DRM-free require that the game is playable even without Galaxy Client installed.
Therefore it should be possible to play the game even when SteamAPI_Init
returns false
which isn't the case for Steam.
It means that in order to ensure that your game and its online features work, you need to either:
- keep Galaxy Client running
or
- allow your game to be able to continue running when
SteamAPI_Init
returnsfalse
(which will result in disabling its online features).
It's a special case that could imply game's code changes e.g only replace SteamAPI_Init
section with galaxy::api::Init
.
This feature is still being discussed and worked on.
DLC Discovery and Storage interface's methods using local filesystem (FileWrite/FileRead etc.) are still available even without Galaxy Client running and authorization.
Implementation¶
- Make sure you’re using a supported Steam API version.
- Add achievements to DevPortal, ideally using the VDF file from Steam.
- Download SDK Wrapper (Beta)
- Add SDK Wrapper (Beta) to your game. There are two ways to achieve that. Do one of the following:
- Option a: Rename GalaxySDKWrapper/Libraries/GalaxySDKWrapper[64].dll to steam_api[64].dll and use it to replace the original steam_api[64].dll file in your already built game
- Option b: Recompile your game linking against GalaxySDKWrapper/Libraries/GalaxySDKWrapper[64].lib
- Copy GalaxySDKWrapper/Libraries/Galaxy[64].dll to the same directory as steam_api[64].dll or executable, depending on how working directory and links are handled
- Create a GalaxyConfig.json file where you specify
client_id
and eitherclient_secret
orclient_code
and place it in the same directory as steam_api[64].dll (some exceptions may apply, please see Unity section below) - Your build (ideally no need for rebuilding if you chose 4.a) is now ready to be uploaded to DevPortal
Example:
Game folder
![SWGame folder](_assets/steam-wrapper-game-folder.jpg)
GalaxyConfig.json
![SWGame GalaxyConfig.json](_assets/steam-wrapper-galaxy-config.jpg)
Demo game¶
You can check our SDK Wrapper demo game and its source code for the exact implementation. Ask our support for its license (1931358602 SDK Wrapper Demo Game). Check its source code and build it yourself.
Supported Steam API Versions¶
SDK Wrapper (Beta) can currently support only specific versions of the Steam API headers and your game must be built with one of them. Currently, the supported versions are:
- 1.31 to 1.58
If your game already uses one of the above, no action is necessary. Otherwise, support for other versions of the Steam API headers can be added to SDK Wrapper (Beta) or you may update your current pipeline (your existing build) or set up a new one with a different Steam API version.
Whichever you choose, you can find the list of all Steamworks releases here.
Configuration File¶
In order for SDK Wrapper (Beta) to know the SDK credentials of your game, they must be specified in GalaxyConfig.json
(you can obtain them in Devportal). It is a flat JSON file used for configurations which should be distributed along with the game and located in its working directory — usually where the EXE file is. The only required properties are client_id
and either client_secret
or client_code
. The remaining options are read and parsed during SteamAPI_Init
.
client_code¶
Plaintext client_secret
can be used for testing purposes, but it is recommended to use client_code
instead when releasing. client_code
is an encrypted version of client_secret
. You can obtain it by accessing SDK credentials in the games menu.
Minimal Config Example¶
{
"client_id": "XXXXXXXXXXXXXXXXX",
"client_code": "YYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYY"
}
Available Options¶
Option | Type | Default | Description |
---|---|---|---|
client_id |
string | required | ID of the client |
client_secret |
string | "" |
Secret of the client (required unless client_code is specified) |
client_code |
string | "" |
Code of the client (required unless client_secret is specified) |
enable_logs |
bool | false |
Allows to generate GalaxySteamWrapper-specific logs for troubleshooting |
auth_on_init |
bool | true |
SteamAPI_Init will attempt blocking SignInGalaxy |
require_online |
bool | false |
Indicates if sign in with GOG GALAXY backend is required |
force_callbacks |
bool | false |
Introduced in SW version: 1.2.14. Force set to true if you notice issues with callback processing. |
stats_on_init |
bool | false |
Config option to request user stats on Init |
steam_appid |
number | none | Config option to overwrite Steam AppID |
init_always_true |
bool | false |
Config option so SteamAPI_Init always return 'true' |
dlcs |
array | [] |
Array of DlcInfo struct {"steam_id":number, "name":"string", "galaxy_id":number} |
auth_timeout |
int | 15 |
SignInGalaxy timeout which is set to 15 by default |
Bindings to other programming languages¶
As of 1.32 flat API is supported so projects like Steamworks.NET or Facepunch.Steamworks should work without any additional tinkering. Same goes with other projects that utilizes steam_api.dll
's calls to flat API.
CSteamworks is not currently supported.
Manual Callback Dispatch¶
As of version 1.1.2 of SDK Wrapper, Steam's manual callback dispatch (instead of running all callbacks at once with SteamAPI_RunCallbacks
you can dispatch them manually SteamAPI_ManualDispatch
) is fully supported.
Facepunch.Steamworks moved to using manual dispatch as of 2.3.0 so it should be possible to use this and later versions with SDK Wrapper.
Game engines¶
Game Engine | |
---|---|
Unreal Engine |
both native Steam implementation and EOS seems to work but a little tinkering might be needed. |
Unity |
appears to work without an issue |
Godot |
appears to work without an issue |
Game Maker |
appears to work without an issue |
RenPy |
some issues regarding working directory/dll placement has been reported |
Most of the issues with bindings and game engines seems to be related to working directory/dll or exe placement. We are working on a better approach regarding these issues.
Unity¶
Depending on your Steamworks implementation (missing explicit SteamStats()->RequestCurrentStats
call in Steamworks.NET, Facepunch) you might need to set stats_on_init
flag in GalaxyConfig.json
to true
to fix achievements not working properly.
Important
Correct placement of the GalaxyConfig.json for Unity games is in the root folder of the game installation directory, next to the game.exe.
Example placement of Wrapper files in the Unity game¶
.
├── root
│ └── gameTitle_data/
│ └── Plugins/
│ └── x86_64/
│ ├── Galaxy64.dll (new)
│ └── steam_api64.dll (replaced with renamed GalaxySteamWrapper64.dll)
├── GalaxyConfig.json
├── GalaxySteamWrapper.2021.09.28-09.33.33.log (example log file if logging is enabled)
└── game.exe
Godot¶
If you are using GodotSteam you might need to set stats_on_init
flag in GalaxyConfig.json
to true
.
Unreal Engine¶
Example placement of Wrapper files in the UE game¶
.
├── root
│ └── GameTitle/
│ └── Binaries/
│ └── Win64/ (working directory in UE4-built game)
│ ├── GameTitle-Win64-Shipping.exe
│ ├── GGalaxyConfig.json (new)
│ └── GalaxySteamWrapper.2021.09.28-09.33.33.log (example log file if logging is enabled)
└── Engine/
└── Binaries/
└── ThirdParty/
└── Steamworks/
└── Steamv139/ (of course number might be different if the game uses different Steamworks version)
└── Win64/
├── Galaxy64.dll (new)
└── steam_api64.dll (replaced with renamed GalaxySteamWrapper64.dll)
Native C++¶
If your Steam implementation is in native C++, it should work like any other game so the default drag & drop method mentioned above should suffice.
EOS¶
When using EOS Online Subsystem Steam and blueprints some changes to the engine itself need to be made.
First you need to check which Steamworks version does your version of UE support here: /YourUnrealEnginePath/Engine/Source/ThirdParty/Steamworks/
and Steamworks.build.cs
. For UE 4.27 it's Steamv151 so please stick to this version.
Online Subsystem is a private dependency so by default it is statically linked into your project using steam_api[64].dll
shipped with the engine thus recompilation is necessary.
This implies drag-and-dropping GalaxySteamWrapper[64].dll
and Galaxy[64].dll
e.g here /YourUnrealEnginePath/Engine/Binaries/ThirdParty/Steamworks/Steam[Current Version]/Win64
replacing the original steam_api[64].dll
and then recompiling your project.
After recompilation, please make sure to ship your package with GalaxyConfig.json
placed in YourPackage/YourProject/Binaries/Win[64]
beside your game's executable file and Galaxy[64].dll
in YourPackage/Engine/Binaries/ThirdParty/Steamworks/Steam[Current Version]
.
Writing achievements fails (UE4)¶
Check exactly how achievements are defined in the OSS configuration file in the engine. In case of Nemezis: Mysterious Journey III, achievements were declared in polish-style quotes (“NMJ_FIRST_PUZZLE” instead of "NMJ_FIRST_PUZZLE"). Steamworks offers some support against this, SteamWrapper does not.
Example placement of Wrapper files in the Misc engine¶
.
├── root
├── Galaxy.dll (new)
├── GalaxyConfig.json (new)
├── GalaxySteamWrapper.2021.09.28-09.33.33.log (example log file if logging is enabled)
├── Game.exe
└── steam_api.dll (replaced with renamed GalaxySteamWrapper.dll)
Methods Implemented¶
Note: Unlisted methods are not implemented.
SteamClient Interface¶
Method | Supported | Remarks |
---|---|---|
SetWarningMessageHook |
Yes | Does the same as SteamUtils::SetWarningMessageHook : outputs GOG GALAXY error messages |
SteamMatchmaking Interface¶
Method | Supported | Remarks |
---|---|---|
JoinLobby |
Yes | |
RequestLobbyList |
Partially | Does not request data for each lobby automatically |
GetLobbyByIndex |
Yes | |
CreateLobby |
Yes |
SteamRemoteStorage Interface¶
Method | Supported | Remarks |
---|---|---|
FileExists |
Yes | |
FileRead |
Yes | |
FileWrite |
Partially | No checking for reaching file limits or invalid paths/filenames |
GetFileCount |
Yes | |
GetFileSize |
Yes | |
GetQuota |
No |
SteamUser Interface¶
Method | Supported | Remarks |
---|---|---|
BLoggedOn |
Yes | |
RequestEncryptedAppTicket |
Yes |
SteamUserStats Interface¶
Method | Supported | Remarks |
---|---|---|
RequestCurrentStats |
Yes | |
GetStat (integer) |
Yes | |
GetStat (float) |
Yes | |
SetStat (integer) |
Yes | |
SetStat (float) |
Yes | |
UpdateAvgRateStat |
Yes | |
GetAchievement |
Yes | |
SetAchievement |
Yes | |
ClearAchievements |
Yes | |
GetAchievementAndUnlockTime |
Yes | |
StoreStats |
Yes | |
GetAchievementDisplayAttribute |
Yes | |
FindOrCreateLeaderboard |
Yes | |
FindLeaderboard |
Yes | |
GetLeaderboardName |
Yes | |
GetLeaderboardEntryCount |
Yes | |
GetLeaderboardSortMethod |
Yes | |
GetLeaderboardDisplayType |
Yes | |
DownloadLeaderboardEntries |
Yes | Cannot be called with k_ELeaderboardDataRequestUsers , as in Steam |
DownloadLeaderboardEntriesForUsers |
Yes | |
GetDownloadedLeaderboardEntry |
Yes | |
UploadLeaderboardScore |
Yes |
SteamUtils Interface¶
Method | Supported | Remarks |
---|---|---|
SetWarningMessageHook |
Yes | Does the same as SteamClient::SetWarningMessageHook : outputs GOG GALAXY error messages |
SteamFriends Interface¶
Method | Supported | Remarks |
---|---|---|
GetClanActivityCounts |
No | Always sets arguments to 0 |
GetFriendsGroupName |
Simplified | Assumes everybody is in a single group, always returns Friends . |
GetFriendPersonaNameHistory |
Simplified | Assumes no history exists |
GetPlayerNickname |
Simplified | Assumes no nickname is set |
GetFriendsGroupCount |
Simplified | Groups are not supported; assumes everybody is in a single group |
GetFriendCount |
Partially | In the friend flags passed to this method, only k_EFriendFlagImmediate is supported |
GetFriendByIndex |
Partially | In the friend flags passed to this method, only k_EFriendFlagImmediate is supported |
HasFriend |
Partially | In the friend flags passed to this method, only k_EFriendFlagImmediate is supported |
SetRichPresence |
Partially | No custom keys, only status , metadata and connect allowed |
GetFriendRichPresence |
Partially | No custom keys, only status , metadata and connect allowed |
GetFriendSteamLevel |
No | Returns 0 |
GetClanCount |
Simplified | Returns 0 |
GetFriendsGroupMembersCount |
Partially | Returns friend count regardless of a group |
GetFriendsGroupMembersList |
Partially | Returns friend list regardless of a group |
GetFriendRelationship |
Partially | Returns only k_EFriendRelationshipFriend or k_EFriendRelationshipNone |
GetPersonaState |
Partially | Returns only k_EPersonaStateOnline or k_EPersonaStateOffline |
GetFriendPersonaState |
Partially | Returns only k_EPersonaStateOnline or k_EPersonaStateOffline |
GetUserRestrictions |
Partially | Returns only k_nUserRestrictionNone or k_nUserRestrictionUnknown |
GetPersonaName |
Yes | |
SetPersonaName |
No | |
GetFriendPersonaName |
Yes | |
GetFriendGamePlayed |
No | |
GetFriendsGroupIDByIndex |
No | |
GetClanByIndex |
No | |
GetClanName |
No | |
GetClanTag |
No | |
DownloadClanActivityCounts |
No | |
SetInGameVoiceSpeaking |
No | |
SetPlayedWith |
No | |
GetSmallFriendAvatar |
Yes | |
GetMediumFriendAvatar |
Yes | |
GetLargeFriendAvatar |
Yes | |
RequestUserInformation |
Yes | |
RequestClanOfficerList |
No | |
GetClanOwner |
No | |
GetClanOfficerCount |
No | |
GetClanOfficerByIndex |
No | |
ClearRichPresence |
Yes | |
GetFriendRichPresenceKeyCount |
Yes | |
GetFriendRichPresenceKeyByIndex |
Yes | |
RequestFriendRichPresence |
Yes | |
GetCoplayFriendCount |
No | |
GetCoplayFriend |
No | |
GetFriendCoplayTime |
No | |
GetFriendCoplayGame |
No | |
GetFollowerCount |
No | |
IsFollowing |
No | |
EnumerateFollowingList |
No | |
IsClanPublic |
No | |
IsClanOfficialGameGroup |
No |
Flat API¶
From version 1.32 the flat API functions shipped with Steamworks are supported with some exceptions:
Method |
---|
DestructISteamHTMLSurface |
RequestPublisherOwnedAppData |
GetPublisherOwnedAppData |
Set_SteamAPI_CPostAPIResultInProcess |
Remove_SteamAPI_CPostAPIResultInProcess |
RunFrame |
Set_SteamAPI_CCheckCallbackRegisteredInProcess |
SetXboxPairwiseID |
GetXboxPairwiseID |
SetPSNID |
GetPSNID |
SetStadiaID |
GetStadiaID |
ShowModalGamepadTextInput |
GetConfigValueInfo |
IterateGenericEditableConfigValues |
ResetIdentity |
SetIPv4Addr |
GetFakeIPType |
GetIPv4 |
IsFakeIP |
Also SteamTV and SteamVR interfaces are not supported
Keep in mind this project is a Work In Progress, with many features and improvements to come.
Please leave your feedback.
SDK Wrapper is not endorsed or sponsored by Valve.