Compare commits
23 Commits
minecraft-
...
minecraft-
| Author | SHA1 | Date | |
|---|---|---|---|
| 6624df6bc0 | |||
| 991d0d47b1 | |||
| 5660e09092 | |||
| 097a5d0aca | |||
| 1cf556c87c | |||
| cfe9dfe362 | |||
| 4f7017e486 | |||
| e9313e8f9d | |||
| f30b7e1e4d | |||
| 96d87bdd97 | |||
| 46dce9d6c7 | |||
| edfbd4b22d | |||
| 814f19807d | |||
| 5a492512bc | |||
| 1a26e66e81 | |||
| a69f5ca764 | |||
| fde80452de | |||
| d41e4207e0 | |||
| 9bd96b0614 | |||
| 3fd28200ff | |||
| 79aa1cd730 | |||
| f5f6cecc5b | |||
| ffd99c5ff6 |
363
.gitignore
vendored
363
.gitignore
vendored
@@ -1,365 +1,22 @@
|
|||||||
## Ignore Visual Studio temporary files, build results, and
|
|
||||||
## files generated by popular Visual Studio add-ons.
|
|
||||||
##
|
|
||||||
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
|
||||||
|
|
||||||
# User-specific files
|
|
||||||
*.rsuser
|
|
||||||
*.suo
|
|
||||||
*.user
|
|
||||||
*.userosscache
|
|
||||||
*.sln.docstates
|
|
||||||
|
|
||||||
# User-specific files (MonoDevelop/Xamarin Studio)
|
|
||||||
*.userprefs
|
|
||||||
|
|
||||||
# Mono auto generated files
|
|
||||||
mono_crash.*
|
|
||||||
|
|
||||||
# Build results
|
# Build results
|
||||||
|
[Bb]in/
|
||||||
|
.bin/
|
||||||
[Dd]ebug/
|
[Dd]ebug/
|
||||||
[Dd]ebugPublic/
|
|
||||||
[Rr]elease/
|
[Rr]elease/
|
||||||
[Rr]eleases/
|
[Rr]eleases/
|
||||||
x64/
|
|
||||||
x86/
|
|
||||||
[Ww][Ii][Nn]32/
|
|
||||||
[Aa][Rr][Mm]/
|
|
||||||
[Aa][Rr][Mm]64/
|
|
||||||
bld/
|
|
||||||
[Bb]in/
|
|
||||||
[Oo]bj/
|
[Oo]bj/
|
||||||
[Oo]ut/
|
[Oo]ut/
|
||||||
[Ll]og/
|
[Ll]og/
|
||||||
[Ll]ogs/
|
[Ll]ogs/
|
||||||
|
[Pp]ublish/
|
||||||
|
|
||||||
# Visual Studio 2015/2017 cache/options directory
|
# IDE files
|
||||||
.vs/
|
.vs/
|
||||||
|
.vscode/
|
||||||
.vshistory/
|
.vshistory/
|
||||||
# Uncomment if you have tasks that create the project's static files in wwwroot
|
.idea/
|
||||||
#wwwroot/
|
.editorconfig
|
||||||
|
*.user
|
||||||
|
|
||||||
# Visual Studio 2017 auto generated files
|
#backups
|
||||||
Generated\ Files/
|
.old*/
|
||||||
|
|
||||||
# MSTest test Results
|
|
||||||
[Tt]est[Rr]esult*/
|
|
||||||
[Bb]uild[Ll]og.*
|
|
||||||
|
|
||||||
# NUnit
|
|
||||||
*.VisualState.xml
|
|
||||||
TestResult.xml
|
|
||||||
nunit-*.xml
|
|
||||||
|
|
||||||
# Build Results of an ATL Project
|
|
||||||
[Dd]ebugPS/
|
|
||||||
[Rr]eleasePS/
|
|
||||||
dlldata.c
|
|
||||||
|
|
||||||
# Benchmark Results
|
|
||||||
BenchmarkDotNet.Artifacts/
|
|
||||||
|
|
||||||
# .NET Core
|
|
||||||
project.lock.json
|
|
||||||
project.fragment.lock.json
|
|
||||||
artifacts/
|
|
||||||
|
|
||||||
# ASP.NET Scaffolding
|
|
||||||
ScaffoldingReadMe.txt
|
|
||||||
|
|
||||||
# StyleCop
|
|
||||||
StyleCopReport.xml
|
|
||||||
|
|
||||||
# Files built by Visual Studio
|
|
||||||
*_i.c
|
|
||||||
*_p.c
|
|
||||||
*_h.h
|
|
||||||
*.ilk
|
|
||||||
*.meta
|
|
||||||
*.obj
|
|
||||||
*.iobj
|
|
||||||
*.pch
|
|
||||||
*.pdb
|
|
||||||
*.ipdb
|
|
||||||
*.pgc
|
|
||||||
*.pgd
|
|
||||||
*.rsp
|
|
||||||
*.sbr
|
|
||||||
*.tlb
|
|
||||||
*.tli
|
|
||||||
*.tlh
|
|
||||||
*.tmp
|
|
||||||
*.tmp_proj
|
|
||||||
*_wpftmp.csproj
|
|
||||||
*.log
|
|
||||||
*.vspscc
|
|
||||||
*.vssscc
|
|
||||||
.builds
|
|
||||||
*.pidb
|
|
||||||
*.svclog
|
|
||||||
*.scc
|
|
||||||
*.editorconfig
|
|
||||||
|
|
||||||
# Chutzpah Test files
|
|
||||||
_Chutzpah*
|
|
||||||
|
|
||||||
# Visual C++ cache files
|
|
||||||
ipch/
|
|
||||||
*.aps
|
|
||||||
*.ncb
|
|
||||||
*.opendb
|
|
||||||
*.opensdf
|
|
||||||
*.sdf
|
|
||||||
*.cachefile
|
|
||||||
*.VC.db
|
|
||||||
*.VC.VC.opendb
|
|
||||||
|
|
||||||
# Visual Studio profiler
|
|
||||||
*.psess
|
|
||||||
*.vsp
|
|
||||||
*.vspx
|
|
||||||
*.sap
|
|
||||||
|
|
||||||
# Visual Studio Trace Files
|
|
||||||
*.e2e
|
|
||||||
|
|
||||||
# TFS 2012 Local Workspace
|
|
||||||
$tf/
|
|
||||||
|
|
||||||
# Guidance Automation Toolkit
|
|
||||||
*.gpState
|
|
||||||
|
|
||||||
# ReSharper is a .NET coding add-in
|
|
||||||
_ReSharper*/
|
|
||||||
*.[Rr]e[Ss]harper
|
|
||||||
*.DotSettings.user
|
|
||||||
|
|
||||||
# TeamCity is a build add-in
|
|
||||||
_TeamCity*
|
|
||||||
|
|
||||||
# DotCover is a Code Coverage Tool
|
|
||||||
*.dotCover
|
|
||||||
|
|
||||||
# AxoCover is a Code Coverage Tool
|
|
||||||
.axoCover/*
|
|
||||||
!.axoCover/settings.json
|
|
||||||
|
|
||||||
# Coverlet is a free, cross platform Code Coverage Tool
|
|
||||||
coverage*.json
|
|
||||||
coverage*.xml
|
|
||||||
coverage*.info
|
|
||||||
|
|
||||||
# Visual Studio code coverage results
|
|
||||||
*.coverage
|
|
||||||
*.coveragexml
|
|
||||||
|
|
||||||
# NCrunch
|
|
||||||
_NCrunch_*
|
|
||||||
.*crunch*.local.xml
|
|
||||||
nCrunchTemp_*
|
|
||||||
|
|
||||||
# MightyMoose
|
|
||||||
*.mm.*
|
|
||||||
AutoTest.Net/
|
|
||||||
|
|
||||||
# Web workbench (sass)
|
|
||||||
.sass-cache/
|
|
||||||
|
|
||||||
# Installshield output folder
|
|
||||||
[Ee]xpress/
|
|
||||||
|
|
||||||
# DocProject is a documentation generator add-in
|
|
||||||
DocProject/buildhelp/
|
|
||||||
DocProject/Help/*.HxT
|
|
||||||
DocProject/Help/*.HxC
|
|
||||||
DocProject/Help/*.hhc
|
|
||||||
DocProject/Help/*.hhk
|
|
||||||
DocProject/Help/*.hhp
|
|
||||||
DocProject/Help/Html2
|
|
||||||
DocProject/Help/html
|
|
||||||
|
|
||||||
# Click-Once directory
|
|
||||||
publish/
|
|
||||||
|
|
||||||
# Publish Web Output
|
|
||||||
*.[Pp]ublish.xml
|
|
||||||
*.azurePubxml
|
|
||||||
# Note: Comment the next line if you want to checkin your web deploy settings,
|
|
||||||
# but database connection strings (with potential passwords) will be unencrypted
|
|
||||||
*.pubxml
|
|
||||||
*.publishproj
|
|
||||||
|
|
||||||
# Microsoft Azure Web App publish settings. Comment the next line if you want to
|
|
||||||
# checkin your Azure Web App publish settings, but sensitive information contained
|
|
||||||
# in these scripts will be unencrypted
|
|
||||||
PublishScripts/
|
|
||||||
|
|
||||||
# NuGet Packages
|
|
||||||
*.nupkg
|
|
||||||
# NuGet Symbol Packages
|
|
||||||
*.snupkg
|
|
||||||
# The packages folder can be ignored because of Package Restore
|
|
||||||
**/[Pp]ackages/*
|
|
||||||
# except build/, which is used as an MSBuild target.
|
|
||||||
!**/[Pp]ackages/build/
|
|
||||||
# Uncomment if necessary however generally it will be regenerated when needed
|
|
||||||
#!**/[Pp]ackages/repositories.config
|
|
||||||
# NuGet v3's project.json files produces more ignorable files
|
|
||||||
*.nuget.props
|
|
||||||
*.nuget.targets
|
|
||||||
|
|
||||||
# Microsoft Azure Build Output
|
|
||||||
csx/
|
|
||||||
*.build.csdef
|
|
||||||
|
|
||||||
# Microsoft Azure Emulator
|
|
||||||
ecf/
|
|
||||||
rcf/
|
|
||||||
|
|
||||||
# Windows Store app package directories and files
|
|
||||||
AppPackages/
|
|
||||||
BundleArtifacts/
|
|
||||||
Package.StoreAssociation.xml
|
|
||||||
_pkginfo.txt
|
|
||||||
*.appx
|
|
||||||
*.appxbundle
|
|
||||||
*.appxupload
|
|
||||||
|
|
||||||
# Visual Studio cache files
|
|
||||||
# files ending in .cache can be ignored
|
|
||||||
*.[Cc]ache
|
|
||||||
# but keep track of directories ending in .cache
|
|
||||||
!?*.[Cc]ache/
|
|
||||||
|
|
||||||
# Others
|
|
||||||
ClientBin/
|
|
||||||
~$*
|
|
||||||
*~
|
|
||||||
*.dbmdl
|
|
||||||
*.dbproj.schemaview
|
|
||||||
*.jfm
|
|
||||||
*.pfx
|
|
||||||
*.publishsettings
|
|
||||||
orleans.codegen.cs
|
|
||||||
|
|
||||||
# Including strong name files can present a security risk
|
|
||||||
# (https://github.com/github/gitignore/pull/2483#issue-259490424)
|
|
||||||
#*.snk
|
|
||||||
|
|
||||||
# Since there are multiple workflows, uncomment next line to ignore bower_components
|
|
||||||
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
|
|
||||||
#bower_components/
|
|
||||||
|
|
||||||
# RIA/Silverlight projects
|
|
||||||
Generated_Code/
|
|
||||||
|
|
||||||
# Backup & report files from converting an old project file
|
|
||||||
# to a newer Visual Studio version. Backup files are not needed,
|
|
||||||
# because we have git ;-)
|
|
||||||
_UpgradeReport_Files/
|
|
||||||
Backup*/
|
|
||||||
UpgradeLog*.XML
|
|
||||||
UpgradeLog*.htm
|
|
||||||
ServiceFabricBackup/
|
|
||||||
*.rptproj.bak
|
|
||||||
|
|
||||||
# SQL Server files
|
|
||||||
*.mdf
|
|
||||||
*.ldf
|
|
||||||
*.ndf
|
|
||||||
|
|
||||||
# Business Intelligence projects
|
|
||||||
*.rdl.data
|
|
||||||
*.bim.layout
|
|
||||||
*.bim_*.settings
|
|
||||||
*.rptproj.rsuser
|
|
||||||
*- [Bb]ackup.rdl
|
|
||||||
*- [Bb]ackup ([0-9]).rdl
|
|
||||||
*- [Bb]ackup ([0-9][0-9]).rdl
|
|
||||||
|
|
||||||
# Microsoft Fakes
|
|
||||||
FakesAssemblies/
|
|
||||||
|
|
||||||
# GhostDoc plugin setting file
|
|
||||||
*.GhostDoc.xml
|
|
||||||
|
|
||||||
# Node.js Tools for Visual Studio
|
|
||||||
.ntvs_analysis.dat
|
|
||||||
node_modules/
|
|
||||||
|
|
||||||
# Visual Studio 6 build log
|
|
||||||
*.plg
|
|
||||||
|
|
||||||
# Visual Studio 6 workspace options file
|
|
||||||
*.opt
|
|
||||||
|
|
||||||
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
|
|
||||||
*.vbw
|
|
||||||
|
|
||||||
# Visual Studio LightSwitch build output
|
|
||||||
**/*.HTMLClient/GeneratedArtifacts
|
|
||||||
**/*.DesktopClient/GeneratedArtifacts
|
|
||||||
**/*.DesktopClient/ModelManifest.xml
|
|
||||||
**/*.Server/GeneratedArtifacts
|
|
||||||
**/*.Server/ModelManifest.xml
|
|
||||||
_Pvt_Extensions
|
|
||||||
|
|
||||||
# Paket dependency manager
|
|
||||||
.paket/paket.exe
|
|
||||||
paket-files/
|
|
||||||
|
|
||||||
# FAKE - F# Make
|
|
||||||
.fake/
|
|
||||||
|
|
||||||
# CodeRush personal settings
|
|
||||||
.cr/personal
|
|
||||||
|
|
||||||
# Python Tools for Visual Studio (PTVS)
|
|
||||||
__pycache__/
|
|
||||||
*.pyc
|
|
||||||
|
|
||||||
# Cake - Uncomment if you are using it
|
|
||||||
# tools/**
|
|
||||||
# !tools/packages.config
|
|
||||||
|
|
||||||
# Tabs Studio
|
|
||||||
*.tss
|
|
||||||
|
|
||||||
# Telerik's JustMock configuration file
|
|
||||||
*.jmconfig
|
|
||||||
|
|
||||||
# BizTalk build output
|
|
||||||
*.btp.cs
|
|
||||||
*.btm.cs
|
|
||||||
*.odx.cs
|
|
||||||
*.xsd.cs
|
|
||||||
|
|
||||||
# OpenCover UI analysis results
|
|
||||||
OpenCover/
|
|
||||||
|
|
||||||
# Azure Stream Analytics local run output
|
|
||||||
ASALocalRun/
|
|
||||||
|
|
||||||
# MSBuild Binary and Structured Log
|
|
||||||
*.binlog
|
|
||||||
|
|
||||||
# NVidia Nsight GPU debugger configuration file
|
|
||||||
*.nvuser
|
|
||||||
|
|
||||||
# MFractors (Xamarin productivity tool) working folder
|
|
||||||
.mfractor/
|
|
||||||
|
|
||||||
# Local History for Visual Studio
|
|
||||||
.localhistory/
|
|
||||||
|
|
||||||
# BeatPulse healthcheck temp database
|
|
||||||
healthchecksdb
|
|
||||||
|
|
||||||
# Backup folder for Package Reference Convert tool in Visual Studio 2017
|
|
||||||
MigrationBackup/
|
|
||||||
|
|
||||||
# Ionide (cross platform F# VS Code tools) working folder
|
|
||||||
.ionide/
|
|
||||||
|
|
||||||
# Fody - auto-generated XML schema
|
|
||||||
FodyWeavers.xsd
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8" ?>
|
|
||||||
<configuration>
|
|
||||||
<startup>
|
|
||||||
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8" />
|
|
||||||
</startup>
|
|
||||||
</configuration>
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
using DTLib;
|
|
||||||
using DTLib.Network;
|
|
||||||
using System;
|
|
||||||
using System.Net;
|
|
||||||
using System.Net.Sockets;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Linq;
|
|
||||||
|
|
||||||
namespace launcher_client
|
|
||||||
{
|
|
||||||
class InternalServer
|
|
||||||
{
|
|
||||||
Socket internalServerSocket;
|
|
||||||
Thread internalServerThread;
|
|
||||||
public InternalServer()
|
|
||||||
{
|
|
||||||
internalServerSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
internalServerThread = new(() => Start());
|
|
||||||
internalServerThread.Start();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Start()
|
|
||||||
{
|
|
||||||
internalServerSocket.Bind(new IPEndPoint(IPAddress.Loopback, 38888));
|
|
||||||
internalServerSocket.Listen(1000);
|
|
||||||
if (Launcher.debug) Launcher.Log("g", "internal server started\n");
|
|
||||||
var handlerSocket = internalServerSocket.Accept();
|
|
||||||
if (Launcher.debug) Launcher.Log("b", "new connection\n");
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
if (handlerSocket.Available >= 2)
|
|
||||||
{
|
|
||||||
var request = handlerSocket.GetPackage().ToString();
|
|
||||||
switch (request)
|
|
||||||
{
|
|
||||||
case "client connecting":
|
|
||||||
Launcher.Log("c", $"request from client: <{request}>\n");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Exception("unknown request: " + request);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else Thread.Sleep(10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (ThreadAbortException)
|
|
||||||
{
|
|
||||||
handlerSocket.Shutdown(SocketShutdown.Both);
|
|
||||||
handlerSocket.Close();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Launcher.Log("y", $"InternalServer.Start() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Stop()
|
|
||||||
{
|
|
||||||
if (internalServerSocket.Connected) internalServerSocket.Shutdown(SocketShutdown.Both);
|
|
||||||
internalServerSocket.Close();
|
|
||||||
internalServerThread.Abort();
|
|
||||||
if (Launcher.debug) Launcher.Log("g", "internal server stopped\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
97
minecraft-launcher-client/Launcher.LaunchGame.cs
Normal file
97
minecraft-launcher-client/Launcher.LaunchGame.cs
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
namespace launcher_client;
|
||||||
|
|
||||||
|
internal static partial class Launcher
|
||||||
|
{
|
||||||
|
static string ConstructGameLaunchArgs(string username, string uuid, int maxmemory, int width, int height,
|
||||||
|
string gameDir)
|
||||||
|
=> "-XX:+UnlockExperimentalVMOptions " +
|
||||||
|
"-XX:+UseG1GC " +
|
||||||
|
"-XX:G1NewSizePercent=20 " +
|
||||||
|
"-XX:G1ReservePercent=20 " +
|
||||||
|
"-XX:MaxGCPauseMillis=50 " +
|
||||||
|
"-XX:G1HeapRegionSize=32M " +
|
||||||
|
"-XX:+DisableExplicitGC " +
|
||||||
|
"-XX:+AlwaysPreTouch " +
|
||||||
|
"-XX:+ParallelRefProcEnabled " +
|
||||||
|
"-Xms512M " +
|
||||||
|
$"-Xmx{maxmemory}M " +
|
||||||
|
"-Dfile.encoding=UTF-8 " +
|
||||||
|
"-Dfml.ignoreInvalidMinecraftCertificates=true " +
|
||||||
|
"-Dfml.ignorePatchDiscrepancies=true " +
|
||||||
|
"-Djava.net.useSystemProxies=true " +
|
||||||
|
"-XX:HeapDumpPath=MojangTricksIntelDriversForPerformance_javaw.exe_minecraft.exe.heapdump " +
|
||||||
|
"\"-Dos.name=Windows 10\" " +
|
||||||
|
"-Dos.version=10.0 " +
|
||||||
|
@"-Djava.library.path=versions\1.12.2-forge-14.23.5.2860\natives " +
|
||||||
|
"-Dminecraft.launcher.brand=java-minecraft-launcher " +
|
||||||
|
"-Dminecraft.launcher.version=1.6.84-j " +
|
||||||
|
@"-Dminecraft.client.jar=versions\1.12.2-forge-14.23.5.2860\1.12.2-forge-14.23.5.2860.jar " +
|
||||||
|
"-cp " +
|
||||||
|
@"libraries\com\turikhay\ca-fixer\1.0\ca-fixer-1.0.jar;" +
|
||||||
|
@"libraries\net\minecraftforge\forge\1.12.2-14.23.5.2860\forge-1.12.2-14.23.5.2860.jar;" +
|
||||||
|
@"libraries\org\ow2\asm\asm-debug-all\5.2\asm-debug-all-5.2.jar;" +
|
||||||
|
@"libraries\net\minecraft\launchwrapper\1.12\launchwrapper-1.12.jar;" +
|
||||||
|
@"libraries\org\jline\jline\3.5.1\jline-3.5.1.jar;" +
|
||||||
|
@"libraries\com\typesafe\akka\akka-actor_2.11\2.3.3\akka-actor_2.11-2.3.3.jar;" +
|
||||||
|
@"libraries\com\typesafe\config\1.2.1\config-1.2.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-actors-migration_2.11\1.1.0\scala-actors-migration_2.11-1.1.0.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-compiler\2.11.1\scala-compiler-2.11.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\plugins\scala-continuations-library_2.11\1.0.2_mc\scala-continuations-library_2.11-1.0.2_mc.jar;" +
|
||||||
|
@"libraries\org\scala-lang\plugins\scala-continuations-plugin_2.11.1\1.0.2_mc\scala-continuations-plugin_2.11.1-1.0.2_mc.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-library\2.11.1\scala-library-2.11.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-parser-combinators_2.11\1.0.1\scala-parser-combinators_2.11-1.0.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-reflect\2.11.1\scala-reflect-2.11.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-swing_2.11\1.0.1\scala-swing_2.11-1.0.1.jar;" +
|
||||||
|
@"libraries\org\scala-lang\scala-xml_2.11\1.0.2\scala-xml_2.11-1.0.2.jar;" +
|
||||||
|
@"libraries\lzma\lzma\0.0.1\lzma-0.0.1.jar;" +
|
||||||
|
@"libraries\java3d\vecmath\1.5.2\vecmath-1.5.2.jar;" +
|
||||||
|
@"libraries\net\sf\trove4j\trove4j\3.0.3\trove4j-3.0.3.jar;" +
|
||||||
|
@"libraries\org\apache\maven\maven-artifact\3.5.3\maven-artifact-3.5.3.jar;" +
|
||||||
|
@"libraries\net\sf\jopt-simple\jopt-simple\5.0.3\jopt-simple-5.0.3.jar;" +
|
||||||
|
@"libraries\org\apache\logging\log4j\log4j-api\2.15.0\log4j-api-2.15.0.jar;" +
|
||||||
|
@"libraries\org\apache\logging\log4j\log4j-core\2.15.0\log4j-core-2.15.0.jar;" +
|
||||||
|
@"libraries\ru\tlauncher\patchy\1.0.0\patchy-1.0.0.jar;" +
|
||||||
|
@"libraries\oshi-project\oshi-core\1.1\oshi-core-1.1.jar;" +
|
||||||
|
@"libraries\net\java\dev\jna\jna\4.4.0\jna-4.4.0.jar;" +
|
||||||
|
@"libraries\net\java\dev\jna\platform\3.4.0\platform-3.4.0.jar;" +
|
||||||
|
@"libraries\com\ibm\icu\icu4j-core-mojang\51.2\icu4j-core-mojang-51.2.jar;" +
|
||||||
|
@"libraries\com\paulscode\codecjorbis\20101023\codecjorbis-20101023.jar;" +
|
||||||
|
@"libraries\com\paulscode\codecwav\20101023\codecwav-20101023.jar;" +
|
||||||
|
@"libraries\com\paulscode\libraryjavasound\20101123\libraryjavasound-20101123.jar;" +
|
||||||
|
@"libraries\com\paulscode\librarylwjglopenal\20100824\librarylwjglopenal-20100824.jar;" +
|
||||||
|
@"libraries\com\paulscode\soundsystem\20120107\soundsystem-20120107.jar;" +
|
||||||
|
@"libraries\io\netty\netty-all\4.1.9.Final\netty-all-4.1.9.Final.jar;" +
|
||||||
|
@"libraries\com\google\guava\guava\21.0\guava-21.0.jar;" +
|
||||||
|
@"libraries\org\apache\commons\commons-lang3\3.5\commons-lang3-3.5.jar;" +
|
||||||
|
@"libraries\commons-io\commons-io\2.5\commons-io-2.5.jar;" +
|
||||||
|
@"libraries\commons-codec\commons-codec\1.10\commons-codec-1.10.jar;" +
|
||||||
|
@"libraries\net\java\jinput\jinput\2.0.5\jinput-2.0.5.jar;" +
|
||||||
|
@"libraries\net\java\jutils\jutils\1.0.0\jutils-1.0.0.jar;" +
|
||||||
|
@"libraries\com\google\code\gson\gson\2.8.0\gson-2.8.0.jar;" +
|
||||||
|
@"libraries\by\ely\authlib\3.11.49.2\authlib-3.11.49.2.jar;" +
|
||||||
|
@"libraries\com\mojang\realms\1.10.22\realms-1.10.22.jar;" +
|
||||||
|
@"libraries\org\apache\commons\commons-compress\1.8.1\commons-compress-1.8.1.jar;" +
|
||||||
|
@"libraries\org\apache\httpcomponents\httpclient\4.3.3\httpclient-4.3.3.jar;" +
|
||||||
|
@"libraries\commons-logging\commons-logging\1.1.3\commons-logging-1.1.3.jar;" +
|
||||||
|
@"libraries\org\apache\httpcomponents\httpcore\4.3.2\httpcore-4.3.2.jar;" +
|
||||||
|
@"libraries\it\unimi\dsi\fastutil\7.1.0\fastutil-7.1.0.jar;" +
|
||||||
|
@"libraries\org\apache\logging\log4j\log4j-api\2.8.1\log4j-api-2.8.1.jar;" +
|
||||||
|
@"libraries\org\apache\logging\log4j\log4j-core\2.8.1\log4j-core-2.8.1.jar;" +
|
||||||
|
@"libraries\org\lwjgl\lwjgl\lwjgl\2.9.4-nightly-20150209\lwjgl-2.9.4-nightly-20150209.jar;" +
|
||||||
|
@"libraries\org\lwjgl\lwjgl\lwjgl_util\2.9.4-nightly-20150209\lwjgl_util-2.9.4-nightly-20150209.jar;" +
|
||||||
|
@"libraries\com\mojang\text2speech\1.10.3\text2speech-1.10.3.jar;" +
|
||||||
|
@"versions\1.12.2-forge-14.23.5.2860\1.12.2-forge-14.23.5.2860.jar " +
|
||||||
|
"-Xss2M net.minecraft.launchwrapper.Launch " +
|
||||||
|
$"--username {username} " +
|
||||||
|
"--version 1.12.2-forge-14.23.5.2860 " +
|
||||||
|
$"--gameDir {gameDir} " +
|
||||||
|
"--assetsDir assets " +
|
||||||
|
"--assetIndex 1.12 " +
|
||||||
|
$"--uuid {uuid} " +
|
||||||
|
"--accessToken null " +
|
||||||
|
"--userType mojang " +
|
||||||
|
"--tweakClass net.minecraftforge.fml.common.launcher.FMLTweaker " +
|
||||||
|
"--versionType Forge " +
|
||||||
|
$"--width {width} " +
|
||||||
|
$"--height {height}";
|
||||||
|
}
|
||||||
@@ -1,529 +1,254 @@
|
|||||||
using DTLib;
|
using System;
|
||||||
using DTLib.Dtsod;
|
|
||||||
using DTLib.Filesystem;
|
|
||||||
using DTLib.Network;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Net;
|
using System.Dynamic;
|
||||||
using System.Net.Sockets;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Security.Cryptography;
|
||||||
|
using System.Text;
|
||||||
|
using DTLib.Console;
|
||||||
|
using DTLib.Extensions;
|
||||||
|
using DTLib.Logging;
|
||||||
|
using DTLib.Network;
|
||||||
|
using DTLib.Filesystem;
|
||||||
|
using Directory = DTLib.Filesystem.Directory;
|
||||||
|
using File = DTLib.Filesystem.File;
|
||||||
|
using static launcher_client.Network;
|
||||||
|
namespace launcher_client;
|
||||||
|
|
||||||
namespace launcher_client
|
internal static partial class Launcher
|
||||||
{
|
{
|
||||||
class Launcher
|
private static FileLogger _fileLogger = new("launcher-logs", "launcher-client");
|
||||||
|
public static ILogger Logger = new CompositeLogger(
|
||||||
|
_fileLogger,
|
||||||
|
new ConsoleLogger());
|
||||||
|
public static LauncherConfig Config = null!;
|
||||||
|
public static bool debug, offline, updated;
|
||||||
|
private static dynamic tabs = new ExpandoObject();
|
||||||
|
|
||||||
|
private static void Main(string[] args)
|
||||||
{
|
{
|
||||||
static readonly string logfile = $"logs\\launcher_{DateTime.Now}.log".Replace(':', '-').Replace(' ', '_');
|
try
|
||||||
static Socket mainSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
static readonly string[] server_domains = new string[] { "timerix.cf", "m1net.keenetic.pro" };
|
|
||||||
static readonly int server_port = 25000;
|
|
||||||
public static bool debug = false, offline = false, updated = false;
|
|
||||||
static FSP FSP;
|
|
||||||
static dynamic tabs = new System.Dynamic.ExpandoObject();
|
|
||||||
static DtsodV22 config;
|
|
||||||
static public Process gameProcess;
|
|
||||||
|
|
||||||
static void Main(string[] args)
|
|
||||||
{
|
{
|
||||||
try
|
Console.Title = "anarx_2";
|
||||||
|
Console.OutputEncoding = Encoding.UTF8;
|
||||||
|
Console.InputEncoding = Encoding.UTF8;
|
||||||
|
Console.CursorVisible = false;
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
debug = true;
|
||||||
|
#else
|
||||||
|
if (args.Contains("debug")) debug = true;
|
||||||
|
#endif
|
||||||
|
if (args.Contains("offline")) offline = true;
|
||||||
|
if (args.Contains("updated")) updated = true;
|
||||||
|
Config = !File.Exists(LauncherConfig.ConfigFilePath)
|
||||||
|
? LauncherConfig.CreateDefault()
|
||||||
|
: LauncherConfig.LoadFromFile();
|
||||||
|
|
||||||
|
Logger.DebugLogEnabled = debug;
|
||||||
|
Logger.LogInfo("Main", "launcher is starting");
|
||||||
|
|
||||||
|
if(File.Exists("minecraft-launcher.exe_old"))
|
||||||
|
File.Delete("minecraft-launcher.exe_old");
|
||||||
|
|
||||||
|
// обновление лаунчера
|
||||||
|
if (!updated && !offline)
|
||||||
{
|
{
|
||||||
PublicLog.LogEvent += Log;
|
ConnectToLauncherServer();
|
||||||
PublicLog.LogNoTimeEvent += LogNoTime;
|
mainSocket.SendPackage("requesting launcher update");
|
||||||
if (args.Contains("debug")) debug = true;
|
Fsp.DownloadFile("minecraft-launcher.exe_new");
|
||||||
if (args.Contains("offline")) offline = true;
|
Logger.LogInfo("Main", "minecraft-launcher.exe_new downloaded");
|
||||||
if (args.Contains("updated")) updated = true;
|
System.IO.File.Move("minecraft-launcher.exe", "minecraft-launcher.exe_old");
|
||||||
|
Process.Start("cmd","/c " +
|
||||||
|
"move minecraft-launcher.exe_new minecraft-launcher.exe && " +
|
||||||
|
"minecraft-launcher.exe updated");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// обновление лаунчера
|
// если уже обновлён
|
||||||
if (!updated && !offline)
|
tabs.Login = EmbeddedResources.ReadText("launcher_client.gui.login.gui");
|
||||||
|
tabs.Settings = EmbeddedResources.ReadText("launcher_client.gui.settings.gui");
|
||||||
|
tabs.Exit = EmbeddedResources.ReadText("launcher_client.gui.exit.gui");
|
||||||
|
tabs.Log = "";
|
||||||
|
tabs.Current = "";
|
||||||
|
string username = "";
|
||||||
|
if (!Config.Username.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
tabs.Login = tabs.Login.Remove(833, Config.Username.Length).Insert(833, Config.Username);
|
||||||
|
username = Config.Username;
|
||||||
|
}
|
||||||
|
|
||||||
|
RenderTab(tabs.Login);
|
||||||
|
|
||||||
|
while (true) try
|
||||||
|
// ReSharper disable once BadChildStatementIndent
|
||||||
|
{
|
||||||
|
var pressedKey = Console.ReadKey(true); // Считывание ввода
|
||||||
|
switch (pressedKey.Key)
|
||||||
{
|
{
|
||||||
Connect("updater".ToBytes(), "updater");
|
case ConsoleKey.F1:
|
||||||
mainSocket.SendPackage("requesting launcher update".ToBytes());
|
RenderTab(tabs.Login);
|
||||||
FSP.DownloadFile("launcher.exe_new");
|
break;
|
||||||
Log("g", "launcher.exe_new downloaded\n", "gray", "\n");
|
case ConsoleKey.N:
|
||||||
Process.Start("cmd", "/c timeout 0 && copy launcher.exe_new launcher.exe && start launcher.exe updated && del /f launcher.exe_new");
|
if (tabs.Current == tabs.Login)
|
||||||
}
|
|
||||||
|
|
||||||
// если уже обновлён
|
|
||||||
else if (updated || offline)
|
|
||||||
{
|
|
||||||
var launcherAssembly = System.Reflection.Assembly.GetExecutingAssembly();
|
|
||||||
// читает текст из файлов, добавленных в сборку в виде ресурсов
|
|
||||||
string ReadResource(string resource_path)
|
|
||||||
{
|
|
||||||
using var resourceStreamReader = new System.IO.StreamReader(launcherAssembly.GetManifestResourceStream(resource_path), Encoding.UTF8);
|
|
||||||
return resourceStreamReader.ReadToEnd();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!File.Exists("launcher.dtsod")) File.WriteAllText("launcher.dtsod", ReadResource("launcher_client.launcher.dtsod"));
|
|
||||||
config = new(File.ReadAllText("launcher.dtsod"));
|
|
||||||
tabs.Login = ReadResource("launcher_client.gui.login.gui");
|
|
||||||
tabs.Settings = ReadResource("launcher_client.gui.settings.gui");
|
|
||||||
tabs.Exit = ReadResource("launcher_client.gui.exit.gui");
|
|
||||||
tabs.Log = "";
|
|
||||||
tabs.Current = "";
|
|
||||||
Console.Title = "Anarx 2 launcher";
|
|
||||||
Console.OutputEncoding = Encoding.UTF8;
|
|
||||||
Console.InputEncoding = Encoding.UTF8;
|
|
||||||
Console.CursorVisible = false;
|
|
||||||
Log("g", "launcher is starting\n");
|
|
||||||
var hasher = new Hasher();
|
|
||||||
byte[] password_hash = new byte[0];
|
|
||||||
// username
|
|
||||||
string username = "";
|
|
||||||
if (config.ContainsKey("username"))
|
|
||||||
{
|
|
||||||
tabs.Login = tabs.Login.Remove(833, config["username"].Length).Insert(833, config["username"]);
|
|
||||||
username = config["username"];
|
|
||||||
}
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
{
|
||||||
ConsoleKeyInfo pressedKey = Console.ReadKey(true); // Считывание ввода
|
tabs.Login = tabs.Login
|
||||||
switch (pressedKey.Key)
|
.Remove(751, 20).Insert(751, "┏━━━━━━━━━━━━━━━━━━┓")
|
||||||
|
.Remove(831, 20).Insert(831, "┃ ┃")
|
||||||
|
.Remove(911, 20).Insert(911, "┗━━━━━━━━━━━━━━━━━━┛");
|
||||||
|
RenderTab(tabs.Login);
|
||||||
|
var _username = ReadString(33, 10, 15);
|
||||||
|
tabs.Login = tabs.Login
|
||||||
|
.Remove(751, 20).Insert(751, "┌──────────────────┐")
|
||||||
|
.Remove(831, 20).Insert(831, "│ │")
|
||||||
|
.Remove(911, 20).Insert(911, "└──────────────────┘");
|
||||||
|
RenderTab(tabs.Login);
|
||||||
|
if (_username.Length < 5)
|
||||||
|
throw new Exception("username length should be > 4 and < 17");
|
||||||
|
Config.Username = _username;
|
||||||
|
Config.Save();
|
||||||
|
username = _username;
|
||||||
|
tabs.Login = tabs.Login.Remove(833, _username.Length).Insert(833, _username);
|
||||||
|
RenderTab(tabs.Login);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ConsoleKey.L:
|
||||||
|
if (tabs.Current == tabs.Login)
|
||||||
|
{
|
||||||
|
RenderTab(tabs.Current);
|
||||||
|
if (username.Length < 2) throw new Exception("username is too short");
|
||||||
|
|
||||||
|
// обновление клиента
|
||||||
|
if (!offline)
|
||||||
{
|
{
|
||||||
case ConsoleKey.F1:
|
ConnectToLauncherServer();
|
||||||
RenderTab(tabs.Login);
|
UpdateGame();
|
||||||
break;
|
|
||||||
case ConsoleKey.N:
|
|
||||||
if (tabs.Current == tabs.Login)
|
|
||||||
{
|
|
||||||
tabs.Login = tabs.Login.Remove(751, 20).Insert(751, "┏━━━━━━━━━━━━━━━━━━┓")
|
|
||||||
.Remove(831, 20).Insert(831, "┃ ┃")
|
|
||||||
.Remove(911, 20).Insert(911, "┗━━━━━━━━━━━━━━━━━━┛");
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
var _username = ReadString(33, 10, 15);
|
|
||||||
tabs.Login = tabs.Login.Remove(751, 20).Insert(751, "┌──────────────────┐")
|
|
||||||
.Remove(831, 20).Insert(831, "│ │")
|
|
||||||
.Remove(911, 20).Insert(911, "└──────────────────┘");
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
if (_username.Length < 5) throw new Exception("username length should be > 4 and < 17");
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (config.ContainsKey("username")) config["username"] = _username;
|
|
||||||
else config.Add("username", new DtsodV22.ValueStruct(DtsodV22.ValueTypes.String, _username));
|
|
||||||
File.WriteAllText("launcher.dtsod", config.ToString());
|
|
||||||
username = _username;
|
|
||||||
tabs.Login = tabs.Login.Remove(833, _username.Length).Insert(833, _username);
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.P:
|
|
||||||
if (tabs.Current == tabs.Login)
|
|
||||||
{
|
|
||||||
tabs.Login = tabs.Login.Remove(991, 20).Insert(991, "┏━━━━━━━━━━━━━━━━━━┓")
|
|
||||||
.Remove(1071, 20).Insert(1071, "┃ ┃")
|
|
||||||
.Remove(1151, 20).Insert(1151, "┗━━━━━━━━━━━━━━━━━━┛");
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
var password = ReadString(33, 13, 15);
|
|
||||||
tabs.Login = tabs.Login.Remove(991, 20).Insert(991, "┌──────────────────┐")
|
|
||||||
.Remove(1071, 20).Insert(1071, "│ │")
|
|
||||||
.Remove(1151, 20).Insert(1151, "└──────────────────┘");
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
if (password.Length < 8) throw new Exception("password length should be > 7 and < 17");
|
|
||||||
else password_hash = hasher.HashCycled(password.ToBytes(), 64);
|
|
||||||
tabs.Login = tabs.Login.Remove(1073, password.Length).Insert(1073, "*".Multiply(password.Length));
|
|
||||||
RenderTab(tabs.Login);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.L:
|
|
||||||
if (tabs.Current == tabs.Login)
|
|
||||||
{
|
|
||||||
RenderTab(tabs.Current);
|
|
||||||
if (username.Length < 5) throw new Exception("username is too short");
|
|
||||||
if (password_hash.Length == 0) throw new Exception("pasword is null");
|
|
||||||
// обновление клиента
|
|
||||||
if (!offline)
|
|
||||||
{
|
|
||||||
Connect(hasher.HashCycled(username.ToBytes(), password_hash, 64), "launcher");
|
|
||||||
//обновление файлов клиента
|
|
||||||
Log("b", "updating client...\n");
|
|
||||||
FSP.DownloadByManifest("download_if_not_exist", Directory.GetCurrent());
|
|
||||||
FSP.DownloadByManifest("sync_always", Directory.GetCurrent(), true);
|
|
||||||
foreach (string dir in new DtsodV22(FSP.DownloadFileToMemory("sync_and_remove\\dirlist.dtsod").ToString())["dirs"])
|
|
||||||
FSP.DownloadByManifest("sync_and_remove\\" + dir, Directory.GetCurrent() + '\\' + dir, true, true);
|
|
||||||
Log("g", "client updated\n");
|
|
||||||
}
|
|
||||||
// javapath
|
|
||||||
if (!config.ContainsKey("javapath")) config.Add("javapath", new DtsodV22.ValueStruct(DtsodV22.ValueTypes.String, "jre\\bin"));
|
|
||||||
// uuid
|
|
||||||
if (!config.ContainsKey("uuid"))
|
|
||||||
{
|
|
||||||
Log("y", "uuid not found in config. requesting from server\n");
|
|
||||||
mainSocket.SendPackage("requesting uuid".ToBytes());
|
|
||||||
config.Add("uuid", new DtsodV22.ValueStruct(DtsodV22.ValueTypes.String, mainSocket.GetPackage().ToString()));
|
|
||||||
}
|
|
||||||
File.WriteAllText("launcher.dtsod", config.ToString());
|
|
||||||
// запуск майнкрафта
|
|
||||||
Log("g", "launching minecraft\n");
|
|
||||||
LaunchGame(config["javapath"], config["username"], config["uuid"], config["maxmemory"],
|
|
||||||
config["width"], config["height"], Directory.GetCurrent());
|
|
||||||
//InternalServer internalServer = new();
|
|
||||||
Thread responder = new((_socket) =>
|
|
||||||
{
|
|
||||||
Socket socket = (Socket)_socket;
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (mainSocket.Available >= 2)
|
|
||||||
{
|
|
||||||
var request = mainSocket.GetPackage().ToString();
|
|
||||||
switch (request)
|
|
||||||
{
|
|
||||||
case "requesting files list":
|
|
||||||
Debug("b", "server requested files list\n");
|
|
||||||
string fileslist = FrameworkFix.MergeToString(Directory.GetAllFiles(Directory.GetCurrent()));
|
|
||||||
socket.SendPackage(fileslist.ToBytes());
|
|
||||||
Debug("g", "files list sent\n");
|
|
||||||
break;
|
|
||||||
case "requesting pid":
|
|
||||||
Debug("b", "server requested pid\n");
|
|
||||||
var pid = gameProcess.Id;
|
|
||||||
socket.SendPackage(pid.ToBytes());
|
|
||||||
Debug("g", "pid sent\n");
|
|
||||||
break;
|
|
||||||
//default:
|
|
||||||
//throw new Exception("unknown request: " + request);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else Thread.Sleep(10);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Debug("r", $"responder error: {ex.Message}\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
DTLib.Timer filechecker = new(true, 300000, () =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Debug("b", "checking client files\n");
|
|
||||||
List<string> excesses = new List<string>();
|
|
||||||
var hasher = new Hasher();
|
|
||||||
void CheckDir(string dir)
|
|
||||||
{
|
|
||||||
DtsodV22 manifest = new(FSP.DownloadFileToMemory($"sync_and_remove\\{dir}\\manifest.dtsod").ToString());
|
|
||||||
foreach (string file in Directory.GetAllFiles(dir))
|
|
||||||
{
|
|
||||||
if (!manifest.ContainsKey(file.Remove(0, dir.Length + 1))) excesses.Add(file);
|
|
||||||
else if (hasher.HashFile(file).HashToString() != manifest[file.Remove(0, dir.Length + 1)]) excesses.Add(file);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckDir("jre");
|
|
||||||
CheckDir("mods");
|
|
||||||
CheckDir("resourcepacks");
|
|
||||||
CheckDir("version");
|
|
||||||
CheckDir("libraries");
|
|
||||||
CheckDir("resources");
|
|
||||||
|
|
||||||
if (excesses.Count > 0)
|
|
||||||
{
|
|
||||||
File.WriteAllText("libraries\\commons-codec\\commons-codec\\1.10\\excesses.txt", FrameworkFix.MergeToString(excesses, "\n"));
|
|
||||||
mainSocket.SendPackage("excess files found".ToBytes());
|
|
||||||
FSP.UploadFile("libraries\\commons-codec\\commons-codec\\1.10\\excesses.txt");
|
|
||||||
File.Delete("libraries\\commons-codec\\commons-codec\\1.10\\excesses.txt");
|
|
||||||
}
|
|
||||||
Debug("g", "client files checked");
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Debug("r", $"filechecker error: {ex.Message}\n");
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Log("r", "$$&7$&&??2A%0%A%2\n");
|
|
||||||
mainSocket.SendPackage("sending launcher error".ToBytes());
|
|
||||||
mainSocket.SendPackage(ex.Message.ToBytes());
|
|
||||||
}
|
|
||||||
catch (Exception ex1)
|
|
||||||
{
|
|
||||||
Debug("r", $"filechecker error: {ex1.Message}\n");
|
|
||||||
Log("r", "D0&??/FF\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!offline)
|
|
||||||
{
|
|
||||||
filechecker.Start();
|
|
||||||
//responder.Start(mainSocket);
|
|
||||||
//internalServer.Start();
|
|
||||||
}
|
|
||||||
gameProcess.WaitForExit();
|
|
||||||
Log("b", "minecraft closed\n");
|
|
||||||
if (!offline)
|
|
||||||
{
|
|
||||||
filechecker.Stop();
|
|
||||||
//responder.Abort();
|
|
||||||
//internalServer.Stop();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.R:
|
|
||||||
if (tabs.Current == tabs.Login && !offline)
|
|
||||||
{
|
|
||||||
RenderTab(tabs.Current);
|
|
||||||
if (username.Length < 5) throw new Exception("username is too short");
|
|
||||||
if (password_hash.Length == 0) throw new Exception("pasword is null");
|
|
||||||
Connect("updater".ToBytes(), "updater");
|
|
||||||
mainSocket.SendPackage("register new user".ToBytes());
|
|
||||||
mainSocket.GetAnswer("ready");
|
|
||||||
mainSocket.SendPackage(hasher.HashCycled(username.ToBytes(), password_hash, 64));
|
|
||||||
mainSocket.SendPackage(username.ToBytes());
|
|
||||||
Thread.Sleep(300);
|
|
||||||
LogNoTime("c", ".");
|
|
||||||
Thread.Sleep(300);
|
|
||||||
LogNoTime("c", ".");
|
|
||||||
Thread.Sleep(300);
|
|
||||||
LogNoTime("c", ".");
|
|
||||||
Thread.Sleep(300);
|
|
||||||
LogNoTime("c", ". ");
|
|
||||||
Log("g", "registration request sent\n");
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.F2:
|
|
||||||
tabs.Log = File.ReadAllText(logfile);
|
|
||||||
RenderTab(tabs.Log, 9999);
|
|
||||||
break;
|
|
||||||
case ConsoleKey.F3:
|
|
||||||
RenderTab(tabs.Settings);
|
|
||||||
break;
|
|
||||||
case ConsoleKey.F4:
|
|
||||||
RenderTab(tabs.Exit);
|
|
||||||
break;
|
|
||||||
case ConsoleKey.Enter:
|
|
||||||
if (tabs.Current == tabs.Exit)
|
|
||||||
{
|
|
||||||
Console.Clear();
|
|
||||||
Console.BufferHeight = 9999;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.F5:
|
|
||||||
if (tabs.Current == tabs.Log) goto case ConsoleKey.F2;
|
|
||||||
else RenderTab(tabs.Current);
|
|
||||||
Console.CursorVisible = false;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log("r", $"Client.Main() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
else throw new Exception($"invalid args:<{args.MergeToString(">, <")}>\n");
|
// запуск майнкрафта
|
||||||
|
Logger.LogInfo("Main", "launching minecraft");
|
||||||
|
string gameOptions = ConstructGameLaunchArgs(Config.Username,
|
||||||
|
NameUUIDFromString("OfflinePlayer:" + Config.Username),
|
||||||
|
Config.GameMemory,
|
||||||
|
Config.GameWindowWidth,
|
||||||
|
Config.GameWindowHeight,
|
||||||
|
Directory.GetCurrent());
|
||||||
|
Logger.LogDebug("LaunchGame", gameOptions);
|
||||||
|
var gameProcess = Process.Start(Config.JavaPath.Str, gameOptions);
|
||||||
|
gameProcess.WaitForExit();
|
||||||
|
Logger.LogInfo("Main", "minecraft closed");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ConsoleKey.F2:
|
||||||
|
tabs.Log = File.ReadAllText(_fileLogger.LogfileName);
|
||||||
|
RenderTab(tabs.Log, 9999);
|
||||||
|
break;
|
||||||
|
case ConsoleKey.F3:
|
||||||
|
RenderTab(tabs.Settings);
|
||||||
|
break;
|
||||||
|
case ConsoleKey.F4:
|
||||||
|
RenderTab(tabs.Exit);
|
||||||
|
break;
|
||||||
|
case ConsoleKey.Enter:
|
||||||
|
if (tabs.Current == tabs.Exit)
|
||||||
|
{
|
||||||
|
Console.Clear();
|
||||||
|
// Console.BufferHeight = 9999;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ConsoleKey.F5:
|
||||||
|
if (tabs.Current == tabs.Log) goto case ConsoleKey.F2;
|
||||||
|
RenderTab(tabs.Current);
|
||||||
|
Console.CursorVisible = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Log("r", $"Client.Main() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
Logger.LogError("Main", ex);
|
||||||
ColoredConsole.Write("gray", "press any key to close...");
|
|
||||||
Console.ReadKey();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
catch (Exception ex)
|
||||||
// подключение серверу
|
|
||||||
static void Connect(byte[] hash, string server_answer)
|
|
||||||
{
|
{
|
||||||
if (mainSocket.Connected)
|
Logger.LogError("Main", ex);
|
||||||
{
|
ColoredConsole.Write("gray", "press any key to close...");
|
||||||
Log("b", "socket is connected already. disconnecting...\n");
|
Console.ReadKey();
|
||||||
mainSocket.Shutdown(SocketShutdown.Both);
|
|
||||||
mainSocket.Close();
|
|
||||||
}
|
|
||||||
mainSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
byte domain_num = 0;
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (domain_num >= server_domains.Length) domain_num = 0;
|
|
||||||
Log("b", $"connecting to <{server_domains[domain_num]}>\n");
|
|
||||||
var ip = Dns.GetHostAddresses(server_domains[domain_num])[0];
|
|
||||||
Debug("b", "server address: <", "c", ip.ToString(), "b",
|
|
||||||
">\nserver port: <", "c", server_port.ToString(), "b", ">\n");
|
|
||||||
mainSocket.Connect(new IPEndPoint(ip, server_port));
|
|
||||||
Log("g", "connected to server\n");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
catch (SocketException ex)
|
|
||||||
{
|
|
||||||
domain_num++;
|
|
||||||
Log("r", $"Client.Main() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
ColoredConsole.Write("r", $"Client.Main() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FSP = new(mainSocket);
|
|
||||||
FSP.debug = debug;
|
|
||||||
/*FSP.PackageRecieved += (size) =>
|
|
||||||
{
|
|
||||||
Console.SetCursorPosition(0, 30);
|
|
||||||
Log("b", "downloading file... [", "c", size.ToString(), "b","/", "c", FSP.Filesize = )
|
|
||||||
};*/
|
|
||||||
mainSocket.ReceiveTimeout = 2500;
|
|
||||||
mainSocket.SendTimeout = 2500;
|
|
||||||
mainSocket.GetAnswer("requesting hash");
|
|
||||||
mainSocket.SendPackage(hash);
|
|
||||||
mainSocket.GetAnswer(server_answer);
|
|
||||||
}
|
}
|
||||||
|
Console.CursorVisible = true;
|
||||||
|
}
|
||||||
|
|
||||||
static void RenderTab(string tab, ushort bufferHeight = 30)
|
private static void RenderTab(string tab, ushort bufferHeight = 30)
|
||||||
{
|
{
|
||||||
tabs.Current = tab;
|
tabs.Current = tab;
|
||||||
Console.Clear();
|
Console.Clear();
|
||||||
Console.SetWindowSize(80, 30);
|
Console.SetWindowSize(80, 30);
|
||||||
Console.SetBufferSize(80, bufferHeight);
|
// Console.SetBufferSize(80, bufferHeight);
|
||||||
ColoredConsole.Write("w", tab);
|
ColoredConsole.Write("w", tab);
|
||||||
}
|
}
|
||||||
|
|
||||||
static string ReadString(ushort x, ushort y, ushort maxlength)
|
private static string ReadString(ushort x, ushort y, ushort maxlength)
|
||||||
|
{
|
||||||
|
var output = "";
|
||||||
|
tabs.Current = tabs.Current.Remove(y * 80 + x, maxlength).Insert(y * 80 + x, " ".Multiply(maxlength));
|
||||||
|
while (true)
|
||||||
{
|
{
|
||||||
string output = "";
|
var pressedKey = Console.ReadKey(false);
|
||||||
tabs.Current = tabs.Current.Remove(y * 80 + x, maxlength).Insert(y * 80 + x, " ".Multiply(maxlength));
|
switch (pressedKey.Key)
|
||||||
while (true)
|
|
||||||
{
|
{
|
||||||
ConsoleKeyInfo pressedKey = Console.ReadKey(false);
|
case ConsoleKey.Enter:
|
||||||
switch (pressedKey.Key)
|
return output;
|
||||||
{
|
case ConsoleKey.Backspace:
|
||||||
case ConsoleKey.Enter:
|
if (output.Length > 0)
|
||||||
return output;
|
{
|
||||||
case ConsoleKey.Backspace:
|
output = output.Remove(output.Length - 1);
|
||||||
if (output.Length > 0)
|
|
||||||
{
|
|
||||||
output = output.Remove(output.Length - 1);
|
|
||||||
RenderTab(tabs.Current);
|
|
||||||
Console.SetCursorPosition(x, y);
|
|
||||||
ColoredConsole.Write("c", output);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ConsoleKey.Escape:
|
|
||||||
tabs.Current = tabs.Current.Remove(y * 80 + x, maxlength).Insert(y * 80 + x, " ".Multiply(maxlength));
|
|
||||||
RenderTab(tabs.Current);
|
|
||||||
return "";
|
|
||||||
//case ConsoleKey.Spacebar:
|
|
||||||
case ConsoleKey.UpArrow:
|
|
||||||
case ConsoleKey.DownArrow:
|
|
||||||
case ConsoleKey.LeftArrow:
|
|
||||||
case ConsoleKey.RightArrow:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
if (output.Length <= maxlength)
|
|
||||||
{
|
|
||||||
string thisChar;
|
|
||||||
if (pressedKey.Modifiers.HasFlag(ConsoleModifiers.Shift)) thisChar = pressedKey.KeyChar.ToString().ToUpper();
|
|
||||||
else thisChar = pressedKey.KeyChar.ToString();
|
|
||||||
output += thisChar;
|
|
||||||
}
|
|
||||||
RenderTab(tabs.Current);
|
RenderTab(tabs.Current);
|
||||||
Console.SetCursorPosition(x, y);
|
Console.SetCursorPosition(x, y);
|
||||||
ColoredConsole.Write("c", output);
|
ColoredConsole.Write("c", output);
|
||||||
break;
|
}
|
||||||
}
|
|
||||||
|
break;
|
||||||
|
case ConsoleKey.Escape:
|
||||||
|
tabs.Current = tabs.Current.Remove(y * 80 + x, maxlength)
|
||||||
|
.Insert(y * 80 + x, " ".Multiply(maxlength));
|
||||||
|
RenderTab(tabs.Current);
|
||||||
|
return "";
|
||||||
|
//case ConsoleKey.Spacebar:
|
||||||
|
case ConsoleKey.UpArrow:
|
||||||
|
case ConsoleKey.DownArrow:
|
||||||
|
case ConsoleKey.LeftArrow:
|
||||||
|
case ConsoleKey.RightArrow:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (output.Length <= maxlength)
|
||||||
|
{
|
||||||
|
string keyC = pressedKey.KeyChar.ToString();
|
||||||
|
string thisChar = pressedKey.Modifiers.HasFlag(ConsoleModifiers.Shift) ? keyC.ToUpper() : keyC;
|
||||||
|
output += thisChar;
|
||||||
|
}
|
||||||
|
|
||||||
|
RenderTab(tabs.Current);
|
||||||
|
Console.SetCursorPosition(x, y);
|
||||||
|
ColoredConsole.Write("c", output);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
static void LaunchGame(string javapath, string username, string uuid, string maxmemory, string width, string height, string gamedir)
|
|
||||||
=> gameProcess = Process.Start($"{javapath}\\javaw.exe ",
|
//minecraft player uuid explanation
|
||||||
$"-Djava.net.preferIPv4Stack=true \"-Dos.name=Windows 10\" -Dos.version=10.0 " +
|
//https://gist.github.com/CatDany/0e71ca7cd9b42a254e49/
|
||||||
$"-Xmn256M -Xmx{maxmemory}M -Djava.library.path=version\\natives -cp " +
|
//java uuid generation in c#
|
||||||
$"libraries\\net\\minecraftforge\\forge\\1.12.2-14.23.5.2855\\forge-1.12.2-14.23.5.2855.jar;" +
|
//https://stackoverflow.com/questions/18021808/uuid-interop-with-c-sharp-code
|
||||||
$"libraries\\org\\ow2\\asm\\asm-debug-all\\5.2\\asm-debug-all-5.2.jar;" +
|
public static string NameUUIDFromString(string input)
|
||||||
$"libraries\\net\\minecraft\\launchwrapper\\1.12\\launchwrapper-1.12.jar;" +
|
=> NameUUIDFromBytes(Encoding.UTF8.GetBytes(input));
|
||||||
$"libraries\\org\\jline\\jline\\3.5.1\\jline-3.5.1.jar;" +
|
|
||||||
$"libraries\\com\\typesafe\\akka\\akka-actor_2.11\\2.3.3\\akka-actor_2.11-2.3.3.jar;" +
|
public static string NameUUIDFromBytes(byte[] input)
|
||||||
$"libraries\\com\\typesafe\\config\\1.2.1\\config-1.2.1.jar;" +
|
{
|
||||||
$"libraries\\org\\scala-lang\\scala-actors-migration_2.11\\1.1.0\\scala-actors-migration_2.11-1.1.0.jar;" +
|
byte[] hash = MD5.HashData(input);
|
||||||
$"libraries\\org\\scala-lang\\scala-compiler\\2.11.1\\scala-compiler-2.11.1.jar;" +
|
hash[6] &= 0x0f;
|
||||||
$"libraries\\org\\scala-lang\\plugins\\scala-continuations-library_2.11\\1.0.2_mc\\scala-continuations-library_2.11-1.0.2_mc.jar;l" +
|
hash[6] |= 0x30;
|
||||||
$"ibraries\\org\\scala-lang\\plugins\\scala-continuations-plugin_2.11.1\\1.0.2_mc\\scala-continuations-plugin_2.11.1-1.0.2_mc.jar;" +
|
hash[8] &= 0x3f;
|
||||||
$"libraries\\org\\scala-lang\\scala-library\\2.11.1\\scala-library-2.11.1.jar;" +
|
hash[8] |= 0x80;
|
||||||
$"libraries\\org\\scala-lang\\scala-parser-combinators_2.11\\1.0.1\\scala-parser-combinators_2.11-1.0.1.jar;" +
|
string hex = BitConverter.ToString(hash).Replace("-", string.Empty).ToLower();
|
||||||
$"libraries\\org\\scala-lang\\scala-reflect\\2.11.1\\scala-reflect-2.11.1.jar;" +
|
return hex.Insert(8, "-").Insert(13, "-").Insert(18, "-").Insert(23, "-");
|
||||||
$"libraries\\org\\scala-lang\\scala-swing_2.11\\1.0.1\\scala-swing_2.11-1.0.1.jar;" +
|
|
||||||
$"libraries\\org\\scala-lang\\scala-xml_2.11\\1.0.2\\scala-xml_2.11-1.0.2.jar;" +
|
|
||||||
$"libraries\\lzma\\lzma\\0.0.1\\lzma-0.0.1.jar;" +
|
|
||||||
$"libraries\\java3d\\vecmath\\1.5.2\\vecmath-1.5.2.jar;" +
|
|
||||||
$"libraries\\net\\sf\\trove4j\\trove4j\\3.0.3\\trove4j-3.0.3.jar;" +
|
|
||||||
$"libraries\\org\\apache\\maven\\maven-artifact\\3.5.3\\maven-artifact-3.5.3.jar;" +
|
|
||||||
$"libraries\\net\\sf\\jopt-simple\\jopt-simple\\5.0.3\\jopt-simple-5.0.3.jar;" +
|
|
||||||
$"libraries\\oshi-project\\oshi-core\\1.1\\oshi-core-1.1.jar;" +
|
|
||||||
$"libraries\\net\\java\\dev\\jna\\jna\\4.4.0\\jna-4.4.0.jar;" +
|
|
||||||
$"libraries\\net\\java\\dev\\jna\\platform\\3.4.0\\platform-3.4.0.jar;" +
|
|
||||||
$"libraries\\com\\ibm\\icu\\icu4j-core-mojang\\51.2\\icu4j-core-mojang-51.2.jar;" +
|
|
||||||
$"libraries\\net\\sf\\jopt-simple\\jopt-simple\\5.0.3\\jopt-simple-5.0.3.jar;" +
|
|
||||||
$"libraries\\com\\paulscode\\codecjorbis\\20101023\\codecjorbis-20101023.jar;" +
|
|
||||||
$"libraries\\com\\paulscode\\codecwav\\20101023\\codecwav-20101023.jar;" +
|
|
||||||
$"libraries\\com\\paulscode\\libraryjavasound\\20101123\\libraryjavasound-20101123.jar;" +
|
|
||||||
$"libraries\\com\\paulscode\\librarylwjglopenal\\20100824\\librarylwjglopenal-20100824.jar;" +
|
|
||||||
$"libraries\\com\\paulscode\\soundsystem\\20120107\\soundsystem-20120107.jar;" +
|
|
||||||
$"libraries\\io\\netty\\netty-all\\4.1.9.Final\\netty-all-4.1.9.Final.jar;" +
|
|
||||||
$"libraries\\com\\google\\guava\\guava\\21.0\\guava-21.0.jar;" +
|
|
||||||
$"libraries\\org\\apache\\commons\\commons-lang3\\3.5\\commons-lang3-3.5.jar;" +
|
|
||||||
$"libraries\\commons-io\\commons-io\\2.5\\commons-io-2.5.jar;" +
|
|
||||||
$"libraries\\commons-codec\\commons-codec\\1.10\\commons-codec-1.10.jar;" +
|
|
||||||
$"libraries\\net\\java\\jinput\\jinput\\2.0.5\\jinput-2.0.5.jar;" +
|
|
||||||
$"libraries\\net\\java\\jutils\\jutils\\1.0.0\\jutils-1.0.0.jar;" +
|
|
||||||
$"libraries\\com\\google\\code\\gson\\gson\\2.8.0\\gson-2.8.0.jar;" +
|
|
||||||
$"libraries\\com\\mojang\\authlib\\1.5.25\\authlib-1.5.25.jar;" +
|
|
||||||
$"libraries\\com\\mojang\\realms\\1.10.22\\realms-1.10.22.jar;" +
|
|
||||||
$"libraries\\org\\apache\\commons\\commons-compress\\1.8.1\\commons-compress-1.8.1.jar;" +
|
|
||||||
$"libraries\\org\\apache\\httpcomponents\\httpclient\\4.3.3\\httpclient-4.3.3.jar;" +
|
|
||||||
$"libraries\\commons-logging\\commons-logging\\1.1.3\\commons-logging-1.1.3.jar;" +
|
|
||||||
$"libraries\\org\\apache\\httpcomponents\\httpcore\\4.3.2\\httpcore-4.3.2.jar;" +
|
|
||||||
$"libraries\\it\\unimi\\dsi\\fastutil\\7.1.0\\fastutil-7.1.0.jar;" +
|
|
||||||
$"libraries\\org\\apache\\logging\\log4j\\log4j-api\\2.8.1\\log4j-api-2.8.1.jar;" +
|
|
||||||
$"libraries\\org\\apache\\logging\\log4j\\log4j-core\\2.8.1\\log4j-core-2.8.1.jar;" +
|
|
||||||
$"libraries\\org\\lwjgl\\lwjgl\\lwjgl\\2.9.4-nightly-20150209\\lwjgl-2.9.4-nightly-20150209.jar;" +
|
|
||||||
$"libraries\\org\\lwjgl\\lwjgl\\lwjgl_util\\2.9.4-nightly-20150209\\lwjgl_util-2.9.4-nightly-20150209.jar;" +
|
|
||||||
$"libraries\\com\\mojang\\text2speech\\1.10.3\\text2speech-1.10.3.jar;" +
|
|
||||||
$"version\\1.12.2-forge-14.23.5.2855.jar " +
|
|
||||||
$"-Dminecraft.applet.TargetDirectory={gamedir} " +
|
|
||||||
$"-Dfml.ignoreInvalidMinecraftCertificates=true -Dfml.ignorePatchDiscrepancies=true " +
|
|
||||||
$"net.minecraft.launchwrapper.Launch --username {username} --version 1.12.2-forge-14.23.5.2855 " +
|
|
||||||
$"--gameDir {gamedir} --assetsDir assets --assetIndex 1.12 " +
|
|
||||||
$"--uuid {uuid} --accessToken null --userType mojang --tweakClass net.minecraftforge.fml.common.launcher.FMLTweaker " +
|
|
||||||
$"--versionType Forge --width {width} --height {height}");
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// вывод лога в консоль и файл
|
|
||||||
public static void Log(params string[] msg)
|
|
||||||
{
|
|
||||||
if (msg.Length == 1) msg[0] = "[" + DateTime.Now.ToString() + "]: " + msg[0];
|
|
||||||
else msg[1] = "[" + DateTime.Now.ToString() + "]: " + msg[1];
|
|
||||||
LogNoTime(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void LogNoTime(params string[] msg)
|
|
||||||
{
|
|
||||||
ColoredConsole.Write(msg);
|
|
||||||
if (msg.Length == 1) File.AppendAllText(logfile, msg[0]);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
StringBuilder strB = new();
|
|
||||||
for (ushort i = 0; i < msg.Length; i++)
|
|
||||||
strB.Append(msg[++i]);
|
|
||||||
File.AppendAllText(logfile, strB.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public static void Debug(params string[] msg)
|
|
||||||
{
|
|
||||||
if (debug) Log(msg);
|
|
||||||
}
|
|
||||||
public static void DebugNoTime(params string[] msg)
|
|
||||||
{
|
|
||||||
if (debug) LogNoTime(msg);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
56
minecraft-launcher-client/LauncherConfig.cs
Normal file
56
minecraft-launcher-client/LauncherConfig.cs
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
using DTLib.Dtsod;
|
||||||
|
using DTLib.Filesystem;
|
||||||
|
|
||||||
|
namespace launcher_client;
|
||||||
|
|
||||||
|
public class LauncherConfig
|
||||||
|
{
|
||||||
|
public static IOPath ConfigFilePath = "minecraft-launcher.dtsod";
|
||||||
|
|
||||||
|
public int GameMemory = 3000;
|
||||||
|
public int GameWindowHeight = 500;
|
||||||
|
public int GameWindowWidth = 900;
|
||||||
|
public IOPath JavaPath = "jre/bin/java.exe";
|
||||||
|
public string ServerAddress = "127.0.0.1";
|
||||||
|
public int ServerPort = 25000;
|
||||||
|
public string Username = "";
|
||||||
|
|
||||||
|
private LauncherConfig(){}
|
||||||
|
|
||||||
|
private LauncherConfig(DtsodV23 dtsod)
|
||||||
|
{
|
||||||
|
GameMemory = dtsod["gameMemory"];
|
||||||
|
GameWindowHeight = dtsod["gameWindowHeight"];
|
||||||
|
GameWindowWidth = dtsod["gameWindowWidth"];
|
||||||
|
JavaPath = dtsod["javaPath"];
|
||||||
|
ServerAddress = dtsod["serverAddress"];
|
||||||
|
ServerPort = dtsod["serverPort"];
|
||||||
|
Username = dtsod["username"];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static LauncherConfig LoadFromFile() => new(new DtsodV23(File.ReadAllText(ConfigFilePath)));
|
||||||
|
|
||||||
|
public DtsodV23 ToDtsod() =>
|
||||||
|
new()
|
||||||
|
{
|
||||||
|
{ "gameMemory", GameMemory },
|
||||||
|
{ "gameWindowHeight", GameWindowHeight },
|
||||||
|
{ "gameWindowWidth", GameWindowWidth },
|
||||||
|
{ "javaPath", JavaPath.Str },
|
||||||
|
{ "serverAddress", ServerAddress },
|
||||||
|
{ "serverPort", ServerPort },
|
||||||
|
{ "username", Username },
|
||||||
|
};
|
||||||
|
|
||||||
|
public void Save()
|
||||||
|
{
|
||||||
|
File.WriteAllText(ConfigFilePath, ToDtsod().ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static LauncherConfig CreateDefault()
|
||||||
|
{
|
||||||
|
var c = new LauncherConfig();
|
||||||
|
c.Save();
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
}
|
||||||
91
minecraft-launcher-client/Network.cs
Normal file
91
minecraft-launcher-client/Network.cs
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
using System.Net;
|
||||||
|
using System.Net.Sockets;
|
||||||
|
using System.Threading;
|
||||||
|
using DTLib;
|
||||||
|
using DTLib.Dtsod;
|
||||||
|
using DTLib.Extensions;
|
||||||
|
using DTLib.Filesystem;
|
||||||
|
using DTLib.Logging;
|
||||||
|
using DTLib.Network;
|
||||||
|
using static launcher_client.Launcher;
|
||||||
|
|
||||||
|
namespace launcher_client;
|
||||||
|
|
||||||
|
public class Network
|
||||||
|
{
|
||||||
|
public static Socket mainSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||||
|
public static FSP Fsp = new(mainSocket);
|
||||||
|
|
||||||
|
// подключение серверу
|
||||||
|
public static void ConnectToLauncherServer()
|
||||||
|
{
|
||||||
|
if (mainSocket.Connected)
|
||||||
|
{
|
||||||
|
Logger.LogInfo(nameof(Network), "socket is connected already. disconnecting...");
|
||||||
|
mainSocket.Shutdown(SocketShutdown.Both);
|
||||||
|
mainSocket.Close();
|
||||||
|
mainSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||||
|
Fsp = new(mainSocket);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Logger.LogInfo(nameof(Network), $"connecting to server {Config.ServerAddress}:{Config.ServerPort}");
|
||||||
|
var ip = Dns.GetHostAddresses(Config.ServerAddress)[0];
|
||||||
|
mainSocket.Connect(new IPEndPoint(ip, Config.ServerPort));
|
||||||
|
Logger.LogInfo(nameof(Network), $"connected to server {ip}");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch (SocketException ex)
|
||||||
|
{
|
||||||
|
Logger.LogError(nameof(Network), ex);
|
||||||
|
Thread.Sleep(2000);
|
||||||
|
}
|
||||||
|
|
||||||
|
mainSocket.ReceiveTimeout = 2500;
|
||||||
|
mainSocket.SendTimeout = 2500;
|
||||||
|
mainSocket.GetAnswer("requesting user name");
|
||||||
|
mainSocket.SendPackage("minecraft-launcher");
|
||||||
|
mainSocket.GetAnswer("minecraft-launcher OK");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void DownloadByManifest(IOPath dirOnServer, IOPath dirOnClient, bool overwrite = false, bool delete_excess = false)
|
||||||
|
{
|
||||||
|
var manifestPath = Path.Concat(dirOnServer, "manifest.dtsod");
|
||||||
|
Logger.LogDebug(nameof(Network), manifestPath);
|
||||||
|
string manifestContent = Fsp.DownloadFileToMemory(manifestPath).BytesToString();
|
||||||
|
var manifest = new DtsodV23(manifestContent);
|
||||||
|
var hasher = new Hasher();
|
||||||
|
foreach (var fileOnServerData in manifest)
|
||||||
|
{
|
||||||
|
IOPath fileOnClient = Path.Concat(dirOnClient, fileOnServerData.Key);
|
||||||
|
if (!File.Exists(fileOnClient) || (overwrite && hasher.HashFile(fileOnClient).HashToString() != fileOnServerData.Value))
|
||||||
|
Fsp.DownloadFile(Path.Concat(dirOnServer, fileOnServerData.Key), fileOnClient);
|
||||||
|
}
|
||||||
|
// удаление лишних файлов
|
||||||
|
if (delete_excess)
|
||||||
|
{
|
||||||
|
foreach (var file in Directory.GetAllFiles(dirOnClient))
|
||||||
|
{
|
||||||
|
if (!manifest.ContainsKey(file.RemoveBase(dirOnClient).Str.Replace('\\','/')))
|
||||||
|
File.Delete(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void UpdateGame()
|
||||||
|
{
|
||||||
|
//обновление файлов клиента
|
||||||
|
Logger.LogInfo(nameof(Network), "updating client...");
|
||||||
|
DownloadByManifest("download_if_not_exist", Directory.GetCurrent());
|
||||||
|
DownloadByManifest("sync_always", Directory.GetCurrent(), true);
|
||||||
|
var dirlistDtsod = new DtsodV23(Fsp
|
||||||
|
.DownloadFileToMemory(Path.Concat("sync_and_remove","dirlist.dtsod"))
|
||||||
|
.BytesToString());
|
||||||
|
foreach (string dir in dirlistDtsod["dirs"])
|
||||||
|
DownloadByManifest(Path.Concat("sync_and_remove", dir),
|
||||||
|
Path.Concat(Directory.GetCurrent(), dir), true, true);
|
||||||
|
Logger.LogInfo(nameof(Network), "client updated");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
using System.Reflection;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
// Общие сведения об этой сборке предоставляются следующим набором
|
|
||||||
// набора атрибутов. Измените значения этих атрибутов для изменения сведений,
|
|
||||||
// связанные с этой сборкой.
|
|
||||||
[assembly: AssemblyTitle("minecraft-launcher-client")]
|
|
||||||
[assembly: AssemblyDescription("")]
|
|
||||||
[assembly: AssemblyConfiguration("")]
|
|
||||||
[assembly: AssemblyCompany("")]
|
|
||||||
[assembly: AssemblyProduct("minecraft-launcher-client")]
|
|
||||||
[assembly: AssemblyCopyright("Copyright © 2021")]
|
|
||||||
[assembly: AssemblyTrademark("")]
|
|
||||||
[assembly: AssemblyCulture("")]
|
|
||||||
|
|
||||||
// Установка значения False для параметра ComVisible делает типы в этой сборке невидимыми
|
|
||||||
// для компонентов COM. Если необходимо обратиться к типу в этой сборке через
|
|
||||||
// из модели COM задайте для атрибута ComVisible этого типа значение true.
|
|
||||||
[assembly: ComVisible(false)]
|
|
||||||
|
|
||||||
// Следующий GUID представляет идентификатор typelib, если этот проект доступен из модели COM
|
|
||||||
[assembly: Guid("49adefce-da46-4229-997c-3d43dd600627")]
|
|
||||||
|
|
||||||
// Сведения о версии сборки состоят из указанных ниже четырех значений:
|
|
||||||
//
|
|
||||||
// Основной номер версии
|
|
||||||
// Дополнительный номер версии
|
|
||||||
// Номер сборки
|
|
||||||
// Номер редакции
|
|
||||||
//
|
|
||||||
// Можно задать все значения или принять номера сборки и редакции по умолчанию
|
|
||||||
// используя "*", как показано ниже:
|
|
||||||
// [assembly: AssemblyVersion("1.0.*")]
|
|
||||||
[assembly: AssemblyVersion("1.0.0.0")]
|
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
@@ -10,13 +10,13 @@
|
|||||||
┃ ┌──────────────────┐ ┃
|
┃ ┌──────────────────┐ ┃
|
||||||
┃ [N] nickname:│ │ ┃
|
┃ [N] nickname:│ │ ┃
|
||||||
┃ └──────────────────┘ ┃
|
┃ └──────────────────┘ ┃
|
||||||
┃ ┌──────────────────┐ ┃
|
|
||||||
┃ [P] password:│ │ ┃
|
|
||||||
┃ └──────────────────┘ ┃
|
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ ┏━━━━━━━━━━━━━━━┓ ┏━━━━━━━━━━━━━━┓ ┃
|
┃ ┏━━━━━━━━━━━━━━━━┓ ┃
|
||||||
┃ ┃ [L] login ┃ ┃ [R] register ┃ ┃
|
┃ ┃ [L] login ┃ ┃
|
||||||
┃ ┗━━━━━━━━━━━━━━━┛ ┗━━━━━━━━━━━━━━┛ ┃
|
┃ ┗━━━━━━━━━━━━━━━━┛ ┃
|
||||||
|
┃ ┃
|
||||||
|
┃ ┃
|
||||||
|
┃ ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
┣━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━━┫
|
┣━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━┻━━━━━━━━━━━━━━━┫
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ гыгыг ┃
|
┃ ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
┃ я ещё не добавил настройки ┃
|
┃ я ещё не добавил настройки ┃
|
||||||
┃ ┃
|
┃ ┃
|
||||||
|
|||||||
@@ -1,79 +1,20 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
<PropertyGroup>
|
||||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
<PropertyGroup>
|
<ImplicitUsings>disable</ImplicitUsings>
|
||||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
<Nullable>enable</Nullable>
|
||||||
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
<OutputType>Exe</OutputType>
|
||||||
<ProjectGuid>{49ADEFCE-DA46-4229-997C-3D43DD600627}</ProjectGuid>
|
<RootNamespace>launcher_client</RootNamespace>
|
||||||
<OutputType>Exe</OutputType>
|
<AssemblyName>minecraft-launcher</AssemblyName>
|
||||||
<RootNamespace>launcher_client</RootNamespace>
|
<ApplicationIcon>launcher.ico</ApplicationIcon>
|
||||||
<AssemblyName>launcher</AssemblyName>
|
</PropertyGroup>
|
||||||
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
<ItemGroup>
|
||||||
<LangVersion>9.0</LangVersion>
|
<None Remove="gui\exit.gui" />
|
||||||
<FileAlignment>512</FileAlignment>
|
<EmbeddedResource Include="gui\**" />
|
||||||
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
|
</ItemGroup>
|
||||||
<Deterministic>true</Deterministic>
|
<ItemGroup>
|
||||||
</PropertyGroup>
|
<PackageReference Include="DTLib.Dtsod" Version="1.3.1" />
|
||||||
<PropertyGroup>
|
<PackageReference Include="DTLib.Logging" Version="1.3.1" />
|
||||||
<StartupObject>launcher_client.Launcher</StartupObject>
|
<PackageReference Include="DTLib.Network" Version="1.3.3" />
|
||||||
</PropertyGroup>
|
</ItemGroup>
|
||||||
<PropertyGroup>
|
|
||||||
<ApplicationIcon>logo-D.ico</ApplicationIcon>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup>
|
|
||||||
<NoWin32Manifest>true</NoWin32Manifest>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Build|AnyCPU'">
|
|
||||||
<OutputPath>bin\</OutputPath>
|
|
||||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
|
||||||
<Optimize>true</Optimize>
|
|
||||||
<PlatformTarget>AnyCPU</PlatformTarget>
|
|
||||||
<LangVersion>9.0</LangVersion>
|
|
||||||
<ErrorReport>prompt</ErrorReport>
|
|
||||||
<Prefer32Bit>true</Prefer32Bit>
|
|
||||||
</PropertyGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<Reference Include="System" />
|
|
||||||
<Reference Include="System.Core" />
|
|
||||||
<Reference Include="System.Xml.Linq" />
|
|
||||||
<Reference Include="System.Data.DataSetExtensions" />
|
|
||||||
<Reference Include="Microsoft.CSharp" />
|
|
||||||
<Reference Include="System.Data" />
|
|
||||||
<Reference Include="System.Net.Http" />
|
|
||||||
<Reference Include="System.Xml" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<Compile Include="Launcher.cs" />
|
|
||||||
<Compile Include="InternalServer.cs" />
|
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<None Include="..\.editorconfig">
|
|
||||||
<Link>.editorconfig</Link>
|
|
||||||
</None>
|
|
||||||
<None Include="App.config" />
|
|
||||||
<EmbeddedResource Include="launcher.dtsod">
|
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
|
||||||
</EmbeddedResource>
|
|
||||||
<EmbeddedResource Include="gui\exit.gui" />
|
|
||||||
<EmbeddedResource Include="gui\login.gui" />
|
|
||||||
<EmbeddedResource Include="gui\settings.gui" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<Content Include="logo-D.ico" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<ProjectReference Include="..\..\DTLib\DTLib\DTLib.csproj">
|
|
||||||
<Project>{57cdc0ef-31c9-4859-90e5-ad0b302c5eae}</Project>
|
|
||||||
<Name>DTLib</Name>
|
|
||||||
</ProjectReference>
|
|
||||||
</ItemGroup>
|
|
||||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
|
||||||
<PropertyGroup>
|
|
||||||
<PostBuildEvent>del /f /q launcher.exe.config
|
|
||||||
rmdir /s /q C:\projects\c#\minecraft-launcher\release\src
|
|
||||||
mkdir C:\projects\c#\minecraft-launcher\release\src
|
|
||||||
copy launcher.exe C:\projects\c#\minecraft-launcher\release\src\launcher.exe
|
|
||||||
copy DTLib.dll C:\projects\c#\minecraft-launcher\release\src\DTLib.dll</PostBuildEvent>
|
|
||||||
</PropertyGroup>
|
|
||||||
</Project>
|
</Project>
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
maxmemory: "3000";
|
|
||||||
width: "1600";
|
|
||||||
height: "1000";
|
|
||||||
javapath: "jre\bin";
|
|
||||||
|
Before Width: | Height: | Size: 37 KiB After Width: | Height: | Size: 37 KiB |
10
minecraft-launcher-client/publish.sh
Executable file
10
minecraft-launcher-client/publish.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
dotnet publish -c release -o bin/publish \
|
||||||
|
--self-contained \
|
||||||
|
--use-current-runtime \
|
||||||
|
-p:PublishSingleFile=true \
|
||||||
|
-p:PublishTrimmed=true \
|
||||||
|
-p:TrimMode=partial \
|
||||||
|
-p:EnableCompressionInSingleFile=true \
|
||||||
|
-p:OptimizationPreference=Size \
|
||||||
|
-p:InvariantGlobalization=true \
|
||||||
|
-p:DebugType=none
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8" ?>
|
|
||||||
<configuration>
|
|
||||||
<startup>
|
|
||||||
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8" />
|
|
||||||
</startup>
|
|
||||||
</configuration>
|
|
||||||
69
minecraft-launcher-server/Manifests.cs
Normal file
69
minecraft-launcher-server/Manifests.cs
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using DTLib;
|
||||||
|
using DTLib.Extensions;
|
||||||
|
using DTLib.Filesystem;
|
||||||
|
using static launcher_server.Server;
|
||||||
|
|
||||||
|
namespace launcher_server;
|
||||||
|
|
||||||
|
public static class Manifests
|
||||||
|
{
|
||||||
|
static object manifestLocker = new();
|
||||||
|
|
||||||
|
public static void CreateManifest(IOPath dir)
|
||||||
|
{
|
||||||
|
if(!Directory.Exists(dir))
|
||||||
|
{
|
||||||
|
Directory.Create(dir);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
StringBuilder manifestBuilder = new();
|
||||||
|
Hasher hasher = new();
|
||||||
|
var manifestPath = Path.Concat(dir, "manifest.dtsod");
|
||||||
|
if (Directory.GetFiles(dir).Contains(manifestPath))
|
||||||
|
File.Delete(manifestPath);
|
||||||
|
foreach (var fileInDir in Directory.GetAllFiles(dir))
|
||||||
|
{
|
||||||
|
var fileRelative = fileInDir.RemoveBase(dir);
|
||||||
|
manifestBuilder.Append(fileRelative);
|
||||||
|
manifestBuilder.Append(": \"");
|
||||||
|
byte[] hash = hasher.HashFile(Path.Concat(fileInDir));
|
||||||
|
manifestBuilder.Append(hash.HashToString());
|
||||||
|
manifestBuilder.Append("\";\n");
|
||||||
|
}
|
||||||
|
File.WriteAllText(manifestPath, manifestBuilder.ToString().Replace('\\','/'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void CreateAllManifests()
|
||||||
|
{
|
||||||
|
lock (manifestLocker)
|
||||||
|
{
|
||||||
|
var sync_and_remove_dir = Path.Concat(shared_dir, "sync_and_remove");
|
||||||
|
CreateManifest(Path.Concat(shared_dir, "download_if_not_exist"));
|
||||||
|
CreateManifest(Path.Concat(shared_dir, "sync_always"));
|
||||||
|
if (!Directory.Exists(sync_and_remove_dir))
|
||||||
|
Directory.Create(sync_and_remove_dir);
|
||||||
|
else foreach (var dir in Directory.GetDirectories(sync_and_remove_dir))
|
||||||
|
CreateManifest(dir);
|
||||||
|
StringBuilder dirlist_content_builder = new("dirs: [\n");
|
||||||
|
|
||||||
|
var dirs = Directory.GetDirectories(sync_and_remove_dir);
|
||||||
|
for (var i = 0; i < dirs.Length-1; i++)
|
||||||
|
{
|
||||||
|
dirlist_content_builder
|
||||||
|
.Append("\t\"")
|
||||||
|
.Append(dirs[i].RemoveBase(sync_and_remove_dir).Str.Replace('\\','/'))
|
||||||
|
.Append("\",\n");
|
||||||
|
}
|
||||||
|
dirlist_content_builder
|
||||||
|
.Append("\t\"")
|
||||||
|
.Append(dirs[dirs.Length-1].RemoveBase(sync_and_remove_dir).Str.Replace('\\','/'))
|
||||||
|
.Append("\"\n");
|
||||||
|
|
||||||
|
dirlist_content_builder.Append("];");
|
||||||
|
File.WriteAllText(Path.Concat(sync_and_remove_dir, "dirlist.dtsod"), dirlist_content_builder.ToString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
using System.Reflection;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
// Общие сведения об этой сборке предоставляются следующим набором
|
|
||||||
// набора атрибутов. Измените значения этих атрибутов для изменения сведений,
|
|
||||||
// связанные с этой сборкой.
|
|
||||||
[assembly: AssemblyTitle("minecraft-launcher-server")]
|
|
||||||
[assembly: AssemblyDescription("")]
|
|
||||||
[assembly: AssemblyConfiguration("")]
|
|
||||||
[assembly: AssemblyCompany("")]
|
|
||||||
[assembly: AssemblyProduct("minecraft-launcher-server")]
|
|
||||||
[assembly: AssemblyCopyright("Copyright © 2021")]
|
|
||||||
[assembly: AssemblyTrademark("")]
|
|
||||||
[assembly: AssemblyCulture("")]
|
|
||||||
|
|
||||||
// Установка значения False для параметра ComVisible делает типы в этой сборке невидимыми
|
|
||||||
// для компонентов COM. Если необходимо обратиться к типу в этой сборке через
|
|
||||||
// из модели COM задайте для атрибута ComVisible этого типа значение true.
|
|
||||||
[assembly: ComVisible(false)]
|
|
||||||
|
|
||||||
// Следующий GUID представляет идентификатор typelib, если этот проект доступен из модели COM
|
|
||||||
[assembly: Guid("1dc6892c-5dc8-4c1c-94c1-ce695bd2dbc2")]
|
|
||||||
|
|
||||||
// Сведения о версии сборки состоят из указанных ниже четырех значений:
|
|
||||||
//
|
|
||||||
// Основной номер версии
|
|
||||||
// Дополнительный номер версии
|
|
||||||
// Номер сборки
|
|
||||||
// Номер редакции
|
|
||||||
//
|
|
||||||
// Можно задать все значения или принять номера сборки и редакции по умолчанию
|
|
||||||
// используя "*", как показано ниже:
|
|
||||||
// [assembly: AssemblyVersion("1.0.*")]
|
|
||||||
[assembly: AssemblyVersion("1.0.0.0")]
|
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
@@ -1,218 +1,118 @@
|
|||||||
using DTLib;
|
using System;
|
||||||
using DTLib.Dtsod;
|
|
||||||
using DTLib.Filesystem;
|
|
||||||
using DTLib.Network;
|
|
||||||
using System;
|
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Net.Sockets;
|
using System.Net.Sockets;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Linq;
|
using DTLib.Dtsod;
|
||||||
|
using DTLib.Extensions;
|
||||||
|
using DTLib.Filesystem;
|
||||||
|
using DTLib.Logging;
|
||||||
|
using DTLib.Network;
|
||||||
|
|
||||||
namespace launcher_server
|
namespace launcher_server;
|
||||||
|
|
||||||
|
static class Server
|
||||||
{
|
{
|
||||||
class Server
|
private static ILogger logger = new CompositeLogger(
|
||||||
|
new FileLogger("logs","launcher-server"),
|
||||||
|
new ConsoleLogger());
|
||||||
|
static readonly Socket mainSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||||
|
static DtsodV23 config = null!;
|
||||||
|
public static readonly IOPath shared_dir = "public";
|
||||||
|
|
||||||
|
|
||||||
|
static void Main(string[] args)
|
||||||
{
|
{
|
||||||
static readonly string logfile = $"logs\\launcher-server_{DateTime.Now}.log".Replace(':', '-').Replace(' ', '_');
|
try
|
||||||
static readonly Socket mainSocket = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
static DtsodV22 config;
|
|
||||||
static bool debug = false;
|
|
||||||
|
|
||||||
static object manifestLocker = new();
|
|
||||||
|
|
||||||
static void Main(string[] args)
|
|
||||||
{
|
{
|
||||||
try
|
Console.Title = "minecraft_launcher_server";
|
||||||
|
Console.InputEncoding = Encoding.Unicode;
|
||||||
|
Console.OutputEncoding = Encoding.Unicode;
|
||||||
|
|
||||||
|
config = new DtsodV23(File.ReadAllText("minecraft-launcher-server.dtsod"));
|
||||||
|
|
||||||
|
logger.LogInfo("Main", $"local address: {config["local_ip"]}");
|
||||||
|
logger.LogInfo("Main", $"public address: {Functions.GetPublicIP()}");
|
||||||
|
logger.LogInfo("Main", $"port: {config["local_port"]}");
|
||||||
|
mainSocket.Bind(new IPEndPoint(IPAddress.Parse(config["local_ip"]), config["local_port"]));
|
||||||
|
mainSocket.Listen(1000);
|
||||||
|
Manifests.CreateAllManifests();
|
||||||
|
logger.LogInfo("Main", "server started succesfully");
|
||||||
|
// запуск отдельного потока для каждого юзера
|
||||||
|
logger.LogInfo("Main", "waiting for users");
|
||||||
|
while (true)
|
||||||
{
|
{
|
||||||
Console.Title = "minecraft_launcher_server";
|
var userSocket = mainSocket.Accept();
|
||||||
Console.InputEncoding = Encoding.Unicode;
|
var userThread = new Thread(obj => HandleUser((Socket)obj!));
|
||||||
Console.OutputEncoding = Encoding.Unicode;
|
userThread.Start(userSocket);
|
||||||
PublicLog.LogEvent += Log;
|
}
|
||||||
PublicLog.LogNoTimeEvent += LogNoTime;
|
}
|
||||||
config = new DtsodV22(File.ReadAllText("launcher-server.dtsod"));
|
catch (Exception ex)
|
||||||
if (args.Contains("debug")) debug = true;
|
{
|
||||||
Log("b", "local address: <", "c", config["local_ip"], "b",
|
logger.LogError("Main", ex);
|
||||||
">\npublic address: <", "c", OldNetwork.GetPublicIP(), "b",
|
mainSocket.Close();
|
||||||
">\nport: <", "c", config["local_port"].ToString(), "b", ">\n");
|
}
|
||||||
mainSocket.Bind(new IPEndPoint(IPAddress.Parse(config["local_ip"]), config["local_port"]));
|
logger.LogInfo("Main", "");
|
||||||
mainSocket.Listen(1000);
|
}
|
||||||
CreateManifestы();
|
|
||||||
Log("g", "server started succesfully\n");
|
// запускается для каждого юзера в отдельном потоке
|
||||||
// запуск отдельного потока для каждого юзера
|
static void HandleUser(Socket handlerSocket)
|
||||||
Log("b", "waiting for users\n");
|
{
|
||||||
|
logger.LogInfo(nameof(HandleUser), "user connecting... ");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// тут запрос пароля заменён запросом заглушки
|
||||||
|
handlerSocket.SendPackage("requesting user name");
|
||||||
|
string connectionString = handlerSocket.GetPackage().BytesToString();
|
||||||
|
FSP fsp = new(handlerSocket);
|
||||||
|
|
||||||
|
// запрос от апдейтера
|
||||||
|
if (connectionString == "minecraft-launcher")
|
||||||
|
{
|
||||||
|
logger.LogInfo(nameof(HandleUser), "incoming connection from minecraft-launcher");
|
||||||
|
handlerSocket.SendPackage("minecraft-launcher OK");
|
||||||
|
// обработка запросов
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
var userSocket = mainSocket.Accept();
|
if (handlerSocket.Available >= 2)
|
||||||
var userThread = new Thread(new ParameterizedThreadStart((obj) => UserHandle((Socket)obj)));
|
|
||||||
userThread.Start(userSocket);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log("r", $"Server.Main() error:\n{ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
mainSocket.Close();
|
|
||||||
}
|
|
||||||
Log("press any key to close... ");
|
|
||||||
Console.ReadKey();
|
|
||||||
Log("gray", "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
// вывод лога в консоль и файл
|
|
||||||
public static void Log(params string[] msg)
|
|
||||||
{
|
|
||||||
if (msg.Length == 1) msg[0] = "[" + DateTime.Now.ToString() + "]: " + msg[0];
|
|
||||||
else msg[1] = "[" + DateTime.Now.ToString() + "]: " + msg[1];
|
|
||||||
LogNoTime(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void LogNoTime(params string[] msg)
|
|
||||||
{
|
|
||||||
lock (new object())
|
|
||||||
{
|
|
||||||
ColoredConsole.Write(msg);
|
|
||||||
if (msg.Length == 1) File.AppendAllText(logfile, msg[0]);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
StringBuilder strB = new();
|
|
||||||
for (ushort i = 0; i < msg.Length; i++)
|
|
||||||
strB.Append(msg[++i]);
|
|
||||||
File.AppendAllText(logfile, strB.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// запускается для каждого юзера в отдельном потоке
|
|
||||||
static void UserHandle(Socket handlerSocket)
|
|
||||||
{
|
|
||||||
Log("b", "user connecting... ");
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// тут запрос пароля заменён запросом заглушки
|
|
||||||
handlerSocket.SendPackage("requesting hash".ToBytes());
|
|
||||||
var hasher = new Hasher();
|
|
||||||
var hash = hasher.HashCycled(handlerSocket.GetPackage(), 64);
|
|
||||||
FSP fsp = new(handlerSocket);
|
|
||||||
FSP.debug = debug;
|
|
||||||
// запрос от апдейтера
|
|
||||||
if (hash.HashToString() == "39368b9c9ca9a74007acd2358fb7945cf172fc86c93969d0933e40aee6c10ca8")
|
|
||||||
{
|
|
||||||
LogNoTime("b", "user is ", "c", "updater\n");
|
|
||||||
handlerSocket.SendPackage("updater".ToBytes());
|
|
||||||
// обработка запросов
|
|
||||||
while (true)
|
|
||||||
{
|
{
|
||||||
if (handlerSocket.Available >= 2)
|
string request = handlerSocket.GetPackage().BytesToString();
|
||||||
|
switch (request)
|
||||||
{
|
{
|
||||||
var request = handlerSocket.GetPackage().ToString();
|
case "requesting launcher update":
|
||||||
switch (request)
|
logger.LogInfo(nameof(HandleUser), "updater requested launcher update");
|
||||||
{
|
// ReSharper disable once InconsistentlySynchronizedField
|
||||||
case "requesting launcher update":
|
fsp.UploadFile(Path.Concat(shared_dir, "minecraft-launcher.exe"));
|
||||||
Log("b", "updater requested client.exe\n");
|
break;
|
||||||
fsp.UploadFile("share\\launcher.exe");
|
case "requesting file download":
|
||||||
break;
|
var file = handlerSocket.GetPackage().BytesToString();
|
||||||
case "register new user":
|
logger.LogInfo(nameof(HandleUser), $"updater requested file {file}");
|
||||||
Log("b", "new user registration requested\n");
|
// ReSharper disable once InconsistentlySynchronizedField
|
||||||
handlerSocket.SendPackage("ready".ToBytes());
|
fsp.UploadFile(Path.Concat(shared_dir, file));
|
||||||
var req = FrameworkFix.MergeToString(
|
break;
|
||||||
hasher.HashCycled(handlerSocket.GetPackage(), 64).HashToString(),
|
default:
|
||||||
":\n{\n\tusername: \"", handlerSocket.GetPackage().ToString(),
|
throw new Exception("unknown request: " + request);
|
||||||
"\";\n\tuuid: \"null\";\n};\n");
|
|
||||||
var filepath = $"registration_requests\\{DateTime.Now.ToString().Replace(':', '-').Replace(' ', '_')}.req";
|
|
||||||
File.WriteAllText(filepath, req);
|
|
||||||
Log("b", $"text wrote to file <", "c", $"registration_requests\\{filepath}", "b", ">\n");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Exception("unknown request: " + request);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else Thread.Sleep(10);
|
|
||||||
}
|
}
|
||||||
}
|
else Thread.Sleep(50);
|
||||||
// запрос от юзера
|
|
||||||
else if (FindUser(hash, out var user))
|
|
||||||
{
|
|
||||||
LogNoTime("b", $"user is ", "c", user.name + "\n");
|
|
||||||
handlerSocket.SendPackage("launcher".ToBytes());
|
|
||||||
// обработка запросов
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
if (handlerSocket.Available >= 2)
|
|
||||||
{
|
|
||||||
var request = handlerSocket.GetPackage().ToString();
|
|
||||||
switch (request)
|
|
||||||
{
|
|
||||||
case "requesting file download":
|
|
||||||
var file = handlerSocket.GetPackage().ToString();
|
|
||||||
Log("b", $"user ", "c", user.name, "b", " requested file ", "c", file + "\n");
|
|
||||||
if (file == "manifest.dtsod")
|
|
||||||
{
|
|
||||||
lock (manifestLocker) fsp.UploadFile("share\\manifest.dtsod");
|
|
||||||
}
|
|
||||||
else fsp.UploadFile("share\\" + file);
|
|
||||||
break;
|
|
||||||
case "requesting uuid":
|
|
||||||
Log("b", $"user ", "c", user.name, "b", " requested uuid\n");
|
|
||||||
handlerSocket.SendPackage(user.uuid.ToBytes());
|
|
||||||
break;
|
|
||||||
case "excess files found":
|
|
||||||
Log("b", $"user ", "c", user.name, "b", " sent excess files list\n");
|
|
||||||
fsp.DownloadFile($"excesses\\{user.name}-{DateTime.Now.ToString().Replace(':', '-').Replace(' ', '_')}.txt");
|
|
||||||
break;
|
|
||||||
case "sending launcher error":
|
|
||||||
Log("y", "user ", "c", user.name, "y", "is sending error:\n");
|
|
||||||
string error = handlerSocket.GetPackage().ToString();
|
|
||||||
Log("y", error + '\n');
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Exception("unknown request: " + request);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else Thread.Sleep(10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// неизвестный юзер
|
|
||||||
else
|
|
||||||
{
|
|
||||||
LogNoTime("y", $"user with hash <{hash.HashToString()}> not found\n");
|
|
||||||
handlerSocket.SendPackage("user not found".ToBytes());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
// неизвестный юзер
|
||||||
{
|
|
||||||
Log("y", $"UserStart() error:\n message:\n {ex.Message}\n{ex.StackTrace}\n");
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (handlerSocket.Connected) handlerSocket.Shutdown(SocketShutdown.Both);
|
|
||||||
handlerSocket.Close();
|
|
||||||
Log("g", "user disconnected\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void CreateManifestы()
|
logger.LogWarn(nameof(HandleUser),$"invalid connection string: '{connectionString}'");
|
||||||
{
|
handlerSocket.SendPackage("invalid connection string");
|
||||||
lock (manifestLocker)
|
|
||||||
{
|
|
||||||
FSP.CreateManifest("share\\download_if_not_exist");
|
|
||||||
FSP.CreateManifest("share\\sync_always");
|
|
||||||
foreach (string dir in Directory.GetDirectories("share\\sync_and_remove"))
|
|
||||||
FSP.CreateManifest(dir);
|
|
||||||
File.WriteAllText("share\\sync_and_remove\\dirlist.dtsod",
|
|
||||||
$"dirs: [\"{Directory.GetDirectories("share\\sync_and_remove").MergeToString("\",\"").Replace("share\\sync_and_remove\\", "")}\"];\n");
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
catch (Exception ex)
|
||||||
static bool FindUser(byte[] hash, out (string name, string uuid) user)
|
|
||||||
{
|
{
|
||||||
DtsodV22 usersdb = new(File.ReadAllText("users.dtsod"));
|
logger.LogWarn(nameof(HandleUser), ex);
|
||||||
user = new();
|
}
|
||||||
if (usersdb.ContainsKey(hash.HashToString()))
|
finally
|
||||||
{
|
{
|
||||||
user.name = usersdb[hash.HashToString()]["username"];
|
if (handlerSocket.Connected) handlerSocket.Shutdown(SocketShutdown.Both);
|
||||||
user.uuid = usersdb[hash.HashToString()]["uuid"];
|
handlerSocket.Close();
|
||||||
return true;
|
logger.LogInfo(nameof(HandleUser), "user disconnected");
|
||||||
}
|
|
||||||
else return false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -1,59 +1,16 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
|
||||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
<ImplicitUsings>disable</ImplicitUsings>
|
||||||
<ProjectGuid>{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}</ProjectGuid>
|
<Nullable>enable</Nullable>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<RootNamespace>launcher_server</RootNamespace>
|
<RootNamespace>launcher_server</RootNamespace>
|
||||||
<AssemblyName>minecraft-launcher-server</AssemblyName>
|
<AssemblyName>minecraft-launcher-server</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
<ApplicationIcon>launcher.ico</ApplicationIcon>
|
||||||
<LangVersion>9.0</LangVersion>
|
|
||||||
<FileAlignment>512</FileAlignment>
|
|
||||||
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
|
|
||||||
<Deterministic>true</Deterministic>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Build|AnyCPU' ">
|
|
||||||
<PlatformTarget>AnyCPU</PlatformTarget>
|
|
||||||
<DebugType>none</DebugType>
|
|
||||||
<Optimize>true</Optimize>
|
|
||||||
<OutputPath>bin\</OutputPath>
|
|
||||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
|
||||||
<ErrorReport>prompt</ErrorReport>
|
|
||||||
<WarningLevel>4</WarningLevel>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup>
|
|
||||||
<StartupObject>launcher_server.Server</StartupObject>
|
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Reference Include="System" />
|
<PackageReference Include="DTLib.Dtsod" Version="1.3.1" />
|
||||||
<Reference Include="System.Core" />
|
<PackageReference Include="DTLib.Logging" Version="1.3.1" />
|
||||||
<Reference Include="System.Xml.Linq" />
|
<PackageReference Include="DTLib.Network" Version="1.3.3" />
|
||||||
<Reference Include="System.Data.DataSetExtensions" />
|
|
||||||
<Reference Include="Microsoft.CSharp" />
|
|
||||||
<Reference Include="System.Data" />
|
|
||||||
<Reference Include="System.Net.Http" />
|
|
||||||
<Reference Include="System.Xml" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
|
||||||
<Compile Include="Server.cs" />
|
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<None Include="App.config" />
|
|
||||||
<None Include="launcher-server.dtsod">
|
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
|
||||||
</None>
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<ProjectReference Include="..\..\DTLib\DTLib\DTLib.csproj">
|
|
||||||
<Project>{57cdc0ef-31c9-4859-90e5-ad0b302c5eae}</Project>
|
|
||||||
<Name>DTLib</Name>
|
|
||||||
</ProjectReference>
|
|
||||||
</ItemGroup>
|
|
||||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
|
||||||
<PropertyGroup>
|
|
||||||
<PostBuildEvent>del /f /q minecraft-launcher-server.exe.config</PostBuildEvent>
|
|
||||||
</PropertyGroup>
|
|
||||||
</Project>
|
</Project>
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
local_ip: "";
|
|
||||||
local_port: 25000;
|
|
||||||
BIN
minecraft-launcher-server/launcher.ico
Normal file
BIN
minecraft-launcher-server/launcher.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 37 KiB |
@@ -0,0 +1,2 @@
|
|||||||
|
local_ip: "127.0.0.1";
|
||||||
|
local_port: 25000;
|
||||||
11
minecraft-launcher-server/minecraft-launcher-server.service
Normal file
11
minecraft-launcher-server/minecraft-launcher-server.service
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# put this file in /etc/systemd/system/
|
||||||
|
[Unit]
|
||||||
|
Description=minecraft launcher backend in c#
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
WorkingDirectory=/opt/minecraft-launcher/minecraft-launcher-server/bin/publish
|
||||||
|
ExecStart=/opt/minecraft-launcher/minecraft-launcher-server/bin/publish/minecraft-launcher-server
|
||||||
|
Restart=always
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
10
minecraft-launcher-server/publish.sh
Executable file
10
minecraft-launcher-server/publish.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
dotnet publish -c release -o bin/publish \
|
||||||
|
--self-contained \
|
||||||
|
--use-current-runtime \
|
||||||
|
-p:PublishSingleFile=true \
|
||||||
|
-p:PublishTrimmed=true \
|
||||||
|
-p:TrimMode=partial \
|
||||||
|
-p:EnableCompressionInSingleFile=true \
|
||||||
|
-p:OptimizationPreference=Size \
|
||||||
|
-p:InvariantGlobalization=true \
|
||||||
|
-p:DebugType=none
|
||||||
@@ -7,40 +7,20 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "launcher-client", "minecraf
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "launcher-server", "minecraft-launcher-server\launcher-server.csproj", "{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "launcher-server", "minecraft-launcher-server\launcher-server.csproj", "{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DTLib", "..\DTLib\DTLib\DTLib.csproj", "{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}"
|
|
||||||
EndProject
|
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Build|Any CPU = Build|Any CPU
|
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
Release|Any CPU = Release|Any CPU
|
Release|Any CPU = Release|Any CPU
|
||||||
Release-net48|Any CPU = Release-net48|Any CPU
|
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Build|Any CPU.ActiveCfg = Build|Any CPU
|
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Build|Any CPU.Build.0 = Build|Any CPU
|
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Debug|Any CPU.ActiveCfg = Build|Any CPU
|
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Debug|Any CPU.Build.0 = Build|Any CPU
|
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release|Any CPU.ActiveCfg = Build|Any CPU
|
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release|Any CPU.Build.0 = Build|Any CPU
|
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release-net48|Any CPU.ActiveCfg = Build|Any CPU
|
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{49ADEFCE-DA46-4229-997C-3D43DD600627}.Release-net48|Any CPU.Build.0 = Build|Any CPU
|
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Build|Any CPU.ActiveCfg = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Build|Any CPU.Build.0 = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Debug|Any CPU.ActiveCfg = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Debug|Any CPU.Build.0 = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release|Any CPU.ActiveCfg = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release|Any CPU.Build.0 = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release-net48|Any CPU.ActiveCfg = Build|Any CPU
|
|
||||||
{1DC6892C-5DC8-4C1C-94C1-CE695BD2DBC2}.Release-net48|Any CPU.Build.0 = Build|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Build|Any CPU.ActiveCfg = Debug|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Build|Any CPU.Build.0 = Debug|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Release|Any CPU.Build.0 = Release|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Release-net48|Any CPU.ActiveCfg = Release-net48|Any CPU
|
|
||||||
{57CDC0EF-31C9-4859-90E5-AD0B302C5EAE}.Release-net48|Any CPU.Build.0 = Release-net48|Any CPU
|
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
|
|||||||
Reference in New Issue
Block a user