Files
SPT-Server-Build/Libraries/SPTarkov.Server.Core/Controllers/LauncherController.cs
T
DrakiaXYZ d2e2f04c93 Merge 4.0.3 changes to main (#674)
* Fix exception sometimes thrown on save
- Switch back from File.Rename to File.Move, as Rename is throwing exceptions on some users systems

* Change BTR skin to tarcola during Christmas event

* Added comment

* Remove unused using

* Add wipe Response model

* formatting and add Wipe Endpoint to V2

* Format Style Fixes

* Merge pull request #669 from sp-tarkov/Assembly-ref-validation

Validate core assembly reference when loading mods

* removed zombies from customs and interchange + increased infection across other maps that have zombie kill quests

* Don't apply hostility changes to maps without zombies during halloween

`ReplaceBotHostiltiy` has optional map whitelist param

* Updated hostility values for maps with infection:
bosses = hostile to player not to pmc bots
followers = hostile to player not to pmc bots
pmcs = hostile to player + always hostile to scavs
scavs = hostile to player and pmc bots
raiders = hostile to player and pmc bots

Adjusted infection rates to just maps with zombie kill quests

* Format Style Fixes

* Added missing values for event bosses

* Format Style Fixes

* Added missing values for `ravangezryachiyevent`
Fixed preset typo `bossTagillaAgro`

* Format Style Fixes

* Flagged `Night of The Cult` as halloween quest

* Fixed incorrect logic

* Enabled `Night of The Cult` bosses to spawn

* Format Style Fixes

* Addd a new ReleaseCheckService to notify users of updates (#670)

* Addd a new ReleaseCheckService to notify users of updates
- Pulls the latest release from GitHub API to compare the tag against the users current SPT version
- Runs at the very end of the startup process to avoid being pushed off screen by mod logging
- Only notifies of patch version increments, not major or minor increments
- Links the release notes so users can Ctrl+Click to open directly to the upgrade page
- Is run on its own thread, and discards all errors, so as to not impact users without an internet connection or ability to access GitHub

* Formatting

* Use record for the ReleaseInformation class

---------

Co-authored-by: DrakiaXYZ <565558+TheDgtl@users.noreply.github.com>

* ProfileDataService changes:

Added `ClearProfileData()`
Replaced filepath access with `Path.Combine`
Reduced various sources of duplication

* Adjusted `Goons` spawn chance to 20% across `Customs/Lighthouse/Woods/Shoreline`

* Account for compound items in DialogHelper.GetMessageItemContents

* Generate weapon/armor price based on the child item price total

* Added halloween event bosses to april event

* Flagged infected spawns as `ForceSpawn` and ``

* Add migration for invalid pockets

* Default assign IEnumerable

* Post raid effect fixes:
When exiting raid with severe muscle pain, prevent client instructing server to add mild muscle pain
When exiting a raid with effect that has a timer, decrease timer value by amount of time spent in raid

* Updated nuget packages

* Fixed player scav not having correct HP values on limbs #642

* Remove unused record

* Revert "Updated nuget packages"

This reverts commit f6d9d461a6.

* Added `IMP mine detector` to reward and flea blacklist

* Fixed weapon builds not overwriting existing #654

Cleaned up `SaveWeaponBuild` and `SaveEquipmentBuild`

---------

Co-authored-by: DrakiaXYZ <565558+TheDgtl@users.noreply.github.com>
Co-authored-by: Chomp <27521899+chompDev@users.noreply.github.com>
Co-authored-by: Chomp <dev@dev.sp-tarkov.com>
Co-authored-by: CWX <CWXDEV@outlook.com>
Co-authored-by: sp-tarkov-bot <singleplayertarkov@gmail.com>
Co-authored-by: Cj <161484149+CJ-SPT@users.noreply.github.com>
Co-authored-by: Tyfon <29051038+tyfon7@users.noreply.github.com>
Co-authored-by: Archangel <jesse@archangel.wtf>
2025-10-31 14:55:07 -07:00

252 lines
7.7 KiB
C#

using SPTarkov.DI.Annotations;
using SPTarkov.Server.Core.Helpers;
using SPTarkov.Server.Core.Models.Common;
using SPTarkov.Server.Core.Models.Eft.Common.Tables;
using SPTarkov.Server.Core.Models.Eft.Launcher;
using SPTarkov.Server.Core.Models.Eft.Profile;
using SPTarkov.Server.Core.Models.Spt.Config;
using SPTarkov.Server.Core.Models.Spt.Mod;
using SPTarkov.Server.Core.Servers;
using SPTarkov.Server.Core.Services;
using SPTarkov.Server.Core.Services.Mod;
using SPTarkov.Server.Core.Utils;
using Info = SPTarkov.Server.Core.Models.Eft.Profile.Info;
namespace SPTarkov.Server.Core.Controllers;
[Injectable]
public class LauncherController(
IReadOnlyList<SptMod> loadedMods,
HashUtil hashUtil,
SaveServer saveServer,
HttpServerHelper httpServerHelper,
ProfileHelper profileHelper,
DatabaseService databaseService,
ServerLocalisationService serverLocalisationService,
ProfileDataService profileDataService,
ConfigServer configServer
)
{
protected readonly CoreConfig CoreConfig = configServer.GetConfig<CoreConfig>();
/// <summary>
/// Handle launcher connecting to server
/// </summary>
/// <returns>ConnectResponse</returns>
public ConnectResponse Connect()
{
// Get all possible profile types + filter out any that are blacklisted
var profileTemplates = databaseService
.GetProfileTemplates()
.Where(profile => !CoreConfig.Features.CreateNewProfileTypesBlacklist.Contains(profile.Key))
.ToDictionary();
return new ConnectResponse
{
BackendUrl = httpServerHelper.GetBackendUrl(),
Name = CoreConfig.ServerName,
Editions = profileTemplates.Select(x => x.Key).ToList(),
ProfileDescriptions = GetProfileDescriptions(profileTemplates),
};
}
/// <summary>
/// Get descriptive text for each of the profile editions a player can choose, keyed by profile.json profile type e.g. "Edge Of Darkness"
/// </summary>
/// <param name="profileTemplates">Profiles to get descriptions of</param>
/// <returns>Dictionary of profile types with related descriptive text</returns>
protected Dictionary<string, string> GetProfileDescriptions(Dictionary<string, ProfileSides> profileTemplates)
{
var result = new Dictionary<string, string>();
foreach (var (profileKey, profile) in profileTemplates)
{
result.TryAdd(profileKey, serverLocalisationService.GetText(profile.DescriptionLocaleKey));
}
return result;
}
/// <summary>
/// </summary>
/// <param name="sessionId">Session/Player id</param>
/// <returns></returns>
public Info? Find(MongoId sessionId)
{
return saveServer.GetProfiles().TryGetValue(sessionId, out var profile) ? profile.ProfileInfo : null;
}
/// <summary>
/// </summary>
/// <param name="info"></param>
/// <returns></returns>
public MongoId Login(LoginRequestData? info)
{
foreach (var (sessionId, profile) in saveServer.GetProfiles())
{
var account = profile.ProfileInfo;
if (info?.Username == account?.Username)
{
return sessionId;
}
}
return MongoId.Empty();
}
/// <summary>
/// </summary>
/// <param name="info"></param>
/// <returns></returns>
public async Task<MongoId> Register(RegisterData info)
{
foreach (var (_, profile) in saveServer.GetProfiles())
{
if (info.Username == profile.ProfileInfo?.Username)
{
return MongoId.Empty();
}
}
return await CreateAccount(info);
}
/// <summary>
/// </summary>
/// <param name="info"></param>
/// <returns></returns>
protected async Task<MongoId> CreateAccount(RegisterData info)
{
var profileId = new MongoId();
var scavId = new MongoId();
var newProfileDetails = new Info
{
ProfileId = profileId,
ScavengerId = scavId,
Aid = hashUtil.GenerateAccountId(),
Username = info.Username,
IsWiped = true,
Edition = info.Edition,
};
saveServer.CreateProfile(newProfileDetails);
await saveServer.LoadProfileAsync(profileId);
await saveServer.SaveProfileAsync(profileId);
return profileId;
}
/// <summary>
/// </summary>
/// <param name="info"></param>
/// <returns></returns>
public MongoId ChangeUsername(ChangeRequestData info)
{
var sessionID = Login(info);
if (!sessionID.IsEmpty)
{
saveServer.GetProfile(sessionID).ProfileInfo!.Username = info.Change;
}
return sessionID;
}
/// <summary>
/// Handle launcher requesting profile be wiped
/// </summary>
/// <param name="info">Registration data</param>
/// <returns>Session id</returns>
public MongoId Wipe(RegisterData info)
{
if (!CoreConfig.AllowProfileWipe)
{
return MongoId.Empty();
}
var sessionId = Login(info);
if (!sessionId.IsEmpty)
{
var profileInfo = saveServer.GetProfile(sessionId).ProfileInfo;
profileInfo!.Edition = info.Edition;
profileInfo.IsWiped = true;
// Clear any data modders may have stored
profileDataService.ClearProfileData(sessionId);
}
return sessionId;
}
/// <summary>
/// </summary>
/// <returns></returns>
public string GetCompatibleTarkovVersion()
{
return CoreConfig.CompatibleTarkovVersion;
}
/// <summary>
/// Get the mods the server has currently loaded
/// </summary>
/// <returns>Dictionary of mod name and mod details</returns>
public Dictionary<string, AbstractModMetadata> GetLoadedServerMods()
{
return loadedMods.ToDictionary(sptMod => sptMod.ModMetadata?.Name ?? "UNKNOWN MOD", sptMod => sptMod.ModMetadata);
}
/// <summary>
/// Get the mods a profile has ever loaded into game with
/// </summary>
/// <param name="sessionId">Session/Player id</param>
/// <returns>Array of mod details</returns>
public List<ModDetails> GetServerModsProfileUsed(MongoId sessionId)
{
var profile = profileHelper.GetFullProfile(sessionId);
if (profile?.SptData?.Mods is not null)
{
return GetProfileModsGroupedByModName(profile?.SptData?.Mods);
}
return [];
}
/// <summary>
/// </summary>
/// <param name="profileMods"></param>
/// <returns></returns>
public List<ModDetails> GetProfileModsGroupedByModName(List<ModDetails> profileMods)
{
// Group all mods used by profile by name
var modsGroupedByName = new Dictionary<string, List<ModDetails>>();
foreach (var mod in profileMods)
{
if (!modsGroupedByName.ContainsKey(mod.Name))
{
modsGroupedByName[mod.Name] = [];
}
modsGroupedByName[mod.Name].Add(mod);
}
// Find the highest versioned mod and add to results array
var result = new List<ModDetails>();
foreach (var (modName, modDatas) in modsGroupedByName)
{
var modVersions = modDatas.Select(x => x.Version);
// var highestVersion = MaxSatisfying(modVersions, "*"); ?? TODO: Node used SemVer here
var chosenVersion = modDatas.FirstOrDefault(x => x.Name == modName); // && x.Version == highestVersion
if (chosenVersion is null)
{
continue;
}
result.Add(chosenVersion);
}
return result;
}
}