Compare commits

...

No commits in common. "master" and "main" have entirely different histories.
master ... main

391 changed files with 2 additions and 29424 deletions

63
.gitattributes vendored
View File

@ -1,63 +0,0 @@
###############################################################################
# Set default behavior to automatically normalize line endings.
###############################################################################
* text=auto
###############################################################################
# Set default behavior for command prompt diff.
#
# This is need for earlier builds of msysgit that does not have it on by
# default for csharp files.
# Note: This is only used by command line
###############################################################################
#*.cs diff=csharp
###############################################################################
# Set the merge driver for project and solution files
#
# Merging from the command prompt will add diff markers to the files if there
# are conflicts (Merging from VS is not affected by the settings below, in VS
# the diff markers are never inserted). Diff markers may cause the following
# file extensions to fail to load in VS. An alternative would be to treat
# these files as binary and thus will always conflict and require user
# intervention with every merge. To do so, just uncomment the entries below
###############################################################################
#*.sln merge=binary
#*.csproj merge=binary
#*.vbproj merge=binary
#*.vcxproj merge=binary
#*.vcproj merge=binary
#*.dbproj merge=binary
#*.fsproj merge=binary
#*.lsproj merge=binary
#*.wixproj merge=binary
#*.modelproj merge=binary
#*.sqlproj merge=binary
#*.wwaproj merge=binary
###############################################################################
# behavior for image files
#
# image files are treated as binary by default.
###############################################################################
#*.jpg binary
#*.png binary
#*.gif binary
###############################################################################
# diff behavior for common document formats
#
# Convert binary document formats to text before diffing them. This feature
# is only available from the command line. Turn it on by uncommenting the
# entries below.
###############################################################################
#*.doc diff=astextplain
#*.DOC diff=astextplain
#*.docx diff=astextplain
#*.DOCX diff=astextplain
#*.dot diff=astextplain
#*.DOT diff=astextplain
#*.pdf diff=astextplain
#*.PDF diff=astextplain
#*.rtf diff=astextplain
#*.RTF diff=astextplain

363
.gitignore vendored
View File

@ -1,363 +0,0 @@
## 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
[Dd]ebug/
[Dd]ebugPublic/
[Rr]elease/
[Rr]eleases/
x64/
x86/
[Ww][Ii][Nn]32/
[Aa][Rr][Mm]/
[Aa][Rr][Mm]64/
bld/
[Bb]in/
[Oo]bj/
[Oo]ut/
[Ll]og/
[Ll]ogs/
# Visual Studio 2015/2017 cache/options directory
.vs/
# Uncomment if you have tasks that create the project's static files in wwwroot
#wwwroot/
# Visual Studio 2017 auto generated files
Generated\ Files/
# 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
# 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

View File

@ -1,50 +0,0 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.9.34728.123
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Server", "Server\Server.csproj", "{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{18D5BEEA-6AF2-411C-BBCE-508B066FB339}"
ProjectSection(SolutionItems) = preProject
.gitignore = .gitignore
README.md = README.md
EndProjectSection
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ServerTest", "ServerTest\ServerTest.csproj", "{024D84F3-734A-421E-BDA9-1C069F9FFD90}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Debug|x86 = Debug|x86
Release|Any CPU = Release|Any CPU
Release|x86 = Release|x86
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Debug|Any CPU.Build.0 = Debug|Any CPU
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Debug|x86.ActiveCfg = Debug|x86
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Debug|x86.Build.0 = Debug|x86
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Release|Any CPU.ActiveCfg = Release|Any CPU
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Release|Any CPU.Build.0 = Release|Any CPU
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Release|x86.ActiveCfg = Release|x86
{E54FC88C-B212-4F47-B2F1-927D39EE3DBA}.Release|x86.Build.0 = Release|x86
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Debug|Any CPU.Build.0 = Debug|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Debug|x86.ActiveCfg = Debug|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Debug|x86.Build.0 = Debug|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Release|Any CPU.ActiveCfg = Release|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Release|Any CPU.Build.0 = Release|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Release|x86.ActiveCfg = Release|Any CPU
{024D84F3-734A-421E-BDA9-1C069F9FFD90}.Release|x86.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {CAC7CBA3-A6C8-4BF1-8316-174820A3CE26}
EndGlobalSection
GlobalSection(MonoDevelopProperties) = preSolution
version = 0.1.0-pre
EndGlobalSection
EndGlobal

View File

@ -1,84 +1,5 @@
主页:http://pso2.fun:88/
# PSO2SERVER
PolarisServer rebuild
梦幻之星OL2 中国服务器 Polaris二次开发
目前开发的版本:
Pre-NGS (JP: 6.1201.0)
[![AGPL License](http://img.shields.io/badge/license-AGPL%20v3-red.svg?style=flat-square)](http://opensource.org/licenses/AGPL-3.0)
[![Build status](https://ci.appveyor.com/api/projects/status/3ltwll4bck12ey0t?svg=true)](https://ci.appveyor.com/project/cyberkitsune/polarisserver)
## Table of Contents
* [What is it?](#what-is-it)
* [Installation](#installation)
* [Building The Latest Version](#building-the-latest-version)
* [Downloading The Latest Version](#downloading-the-latest-version)
* [Documentation](#documentation)
* [Licensing](#licensing)
* [Third Party Licenses](#third-party-licenses)
* [Contributing and Feedback](#contributing-and-feedback)
* [Core Maintainers](#core-maintainers)
## What is it?
`Polaris Private Server` is an open source game private server. It is currently work-in-progress alpha, under heavy development and is not yet recommended for production use.
## Installation
As `Polaris Private Server` is a work-in-progress there is no set way of installaing and running, in future releases we hope to have this information stored in an INSTALL file. Until then you can either build or download and run the latest version.
### Building The Latest Version
For the time being, we only support using mono in order to build Polaris.
You need to have MonoDevelop 4 / Xamarin studio installed. You also need the Mono runtime, even on Windows!
After installing MD / Xamarin and setting up the Mono runtime, open the solution and build!
In the future we will hopefully have a buildserver, xbuild / msbuild support and other fancy things.
### Downloading The Latest Version
As the `Polaris Private Server` is a work-in-progress alpha, you can find the latest unstable built version here @ [AppVeyor](https://ci.appveyor.com/project/cyberkitsune/polarisserver/build/artifacts)
### Documentation
All available documentation for the server can be found on the project wiki @ [pso2proxy.cyberkitsune.net/wiki](http://pso2proxy.cyberkitsune.net/wiki)
## Licensing
All code is licensed under the
[AGPL](https://github.com/PolarisTeam/PolarisServer/blob/master/LICENSE), v3 or later.
### Third Party Licenses
Copyright (c) 2006 Damien Miller <djm@mindrot.org> (jBCrypt)
Copyright (c) 2013 Ryan D. Emerle (.Net port)
Permission to use, copy, modify, and distribute this software for any
purpose with or without fee is hereby granted, provided that the above
copyright notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
## Contributing and Feedback
Currently, you can contribute to the Polaris Private Server project by:
* Submitting a detailed [issue](https://github.com/PolarisTeam/PolarisServer/issues/new).
* [Forking the project](https://github.com/PolarisTeam/PolarisServer/fork), and sending a pull request back to for review.
There is an IRC channel `#pso2` on BadnikZone (irc.badnik.zone:6667), for talking directly
with testers and developers (when awake and present, etc.). However any questions pertaining to the release dates or asking for hacks will be ignored, and you may be banned from the channel.
### Core Maintainers
* "cyberkitsune" <https://github.com/cyberkitsune>
* "KeyPhact" <https://github.com/KeyPhact>
* "Kyle873" <https://github.com/Kyle873>
* "LightningDragon" <https://github.com/LightningDragon>
* "SonicFreak94" <https://github.com/SonicFreak94>
* "Treeki" <https://github.com/Treeki>
梦幻之星OL2 中国服务器 Polaris二次开发

View File

@ -1,128 +0,0 @@
using System;
using System.Linq;
using System.Text;
namespace PSO2SERVER
{
public class AsciiString
{
private byte[] data;
// 获取字符串的长度
public int Length => data.Length;
// 从字符串初始化
public AsciiString(string value)
{
// 确保字符串中只包含 ASCII 字符
if (value.Any(c => c > 127))
{
throw new ArgumentException("字符串包含非 ASCII 字符。");
}
data = Encoding.ASCII.GetBytes(value);
}
// 从字节数组初始化
public AsciiString(byte[] bytes)
{
data = bytes ?? throw new ArgumentNullException(nameof(bytes), "字节数组不能为 null");
}
// 获取 ASCII 字符串,去除尾部的 null 字节
public string GetString()
{
return Encoding.ASCII.GetString(data).TrimEnd('\0');
}
// 获取底层字节数组
public byte[] ToBytes()
{
return data;
}
// 重写 ToString 方法,直接返回 ASCII 字符串
public override string ToString()
{
return GetString();
}
// 检查字节数组是否只包含有效的 ASCII 字符
public bool IsValidAscii()
{
return data.All(b => b < 128);
}
// 静态方法,从字节数组创建 AsciiString 对象
public static AsciiString FromBytes(byte[] bytes)
{
return new AsciiString(bytes);
}
// 静态方法,如果字符串过长则截断
public static AsciiString TruncateIfNeeded(string value, int maxLength)
{
if (value.Length > maxLength)
{
value = value.Substring(0, maxLength);
}
return new AsciiString(value);
}
// 写入固定长度 ASCII 字符串,超出部分截断,空余部分用零填充
public void WriteFixedLengthASCII(string str, int charCount)
{
// 确保字符串长度不超过规定的 charCount
var writeAmount = Math.Min(str.Length, charCount);
var paddingAmount = charCount - writeAmount;
byte[] byteArray = Encoding.ASCII.GetBytes(str.Substring(0, writeAmount));
// 可能存在的填充
byte[] resultArray = new byte[charCount];
Array.Copy(byteArray, resultArray, byteArray.Length);
// 用零填充剩余部分
if (paddingAmount > 0)
{
for (int i = byteArray.Length; i < charCount; i++)
{
resultArray[i] = 0;
}
}
data = resultArray;
}
// 写入固定长度的 UTF-16 字符串,超出部分截断,空余部分用零填充
public void WriteFixedLengthUtf16(string str, int charCount)
{
var writeAmount = Math.Min(str.Length, charCount);
var paddingAmount = charCount - writeAmount;
byte[] byteArray = Encoding.Unicode.GetBytes(str.Substring(0, writeAmount));
// 可能存在的填充
byte[] resultArray = new byte[charCount * 2]; // UTF-16 每个字符占 2 个字节
Array.Copy(byteArray, resultArray, byteArray.Length);
// 用零填充剩余部分
if (paddingAmount > 0)
{
for (int i = byteArray.Length; i < charCount * 2; i++)
{
resultArray[i] = 0;
}
}
data = resultArray;
}
// 重写 ToString 方法,返回字符串表示
public string GetStringRepresentation()
{
return Encoding.ASCII.GetString(data);
}
}
}

View File

@ -1,364 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
using PSO2SERVER.Database;
using PSO2SERVER.Models;
using PSO2SERVER.Network;
using PSO2SERVER.Protocol;
using PSO2SERVER.Protocol.Handlers;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using static Google.Protobuf.Reflection.UninterpretedOption.Types;
namespace PSO2SERVER
{
public class Client
{
public Client()
{
IsClosed = false;
_readBuffer = new byte[1024 * 64];
_readBufferSize = 0;
InputArc4 = null;
OutputArc4 = null;
Palette = new PSOPalette();
battle_stats = new PlayerStats();
SendPacket(new ServerHelloPacket(0x03, 100, 68833280));
}
public Client(Server server, SocketClient socket)
{
IsClosed = false;
_server = server;
Socket = socket;
socket.DataReceived += HandleDataReceived;
socket.ConnectionLost += HandleConnectionLost;
_readBuffer = new byte[1024 * 64];
_readBufferSize = 0;
InputArc4 = null;
OutputArc4 = null;
Palette = new PSOPalette();
battle_stats = new PlayerStats();
SendPacket(new ServerHelloPacket(0x03, 100, 68833280));
}
public enum PacketType : byte
{
/// NGS packet.
NGS,
/// Classic packet. (i.e. NA, JP and Vita)
Classic,
/// NA packet.
NA,
/// JP packet.
JP,
/// Vita packet.
Vita,
/// Raw packet. (i.e. don't parse the packet)
Raw,
}
internal static RSACryptoServiceProvider RsaCsp = null;
private readonly byte[] _readBuffer;
private readonly Server _server;
internal ICryptoTransform InputArc4, OutputArc4;
//private int _packetId;
private uint _readBufferSize;
/// <summary>
/// Server
/// </summary>
public bool IsClosed { get; private set; }
public SocketClient Socket { get; private set; }
/// <summary>
/// Game
/// </summary>
// TODO Consider moving these somewhere else
public Account _account { get; set; }
public Character Character { get; set; }
public uint MovementTimestamp { get; internal set; }
public Flags _flags { get; set; }
public PSOPalette Palette { get; set; }
public PlayerStats battle_stats { get; set; }
public UserState UserState { get; set; }
/// <summary>
/// Party
/// </summary>
public Party.Party currentParty;
/// <summary>
/// Map
/// </summary>
public Map CurrentMap;
public PSOLocation CurrentLocation;
public PSOLocation LastLocation;
private void HandleDataReceived(byte[] data, int size)
{
if ((_readBufferSize + size) > _readBuffer.Length)
{
Logger.WriteError("[<--] 接收到 {0} 字节大于预设buf长度", size);
// Buffer overrun
// TODO: Drop the connection when this occurs?
return;
}
Logger.Write("[<--] 接收到 {0} 字节", size);
Array.Copy(data, 0, _readBuffer, _readBufferSize, size);
InputArc4?.TransformBlock(_readBuffer, (int)_readBufferSize, size, _readBuffer, (int)_readBufferSize);
_readBufferSize += (uint)size;
// Process ALL the packets
uint position = 0;
while ((position + 8) <= _readBufferSize)
{
var packetSize =
_readBuffer[position] |
((uint)_readBuffer[position + 1] << 8) |
((uint)_readBuffer[position + 2] << 16) |
((uint)_readBuffer[position + 3] << 24);
// Minimum size, just to avoid possible infinite loops etc
if (packetSize < 8)
packetSize = 8;
// If we don't have enough data for this one...
if (packetSize > 0x1000000 || (packetSize + position) > _readBufferSize)
break;
// Now handle this one
HandlePacket(
_readBuffer[position + 4], _readBuffer[position + 5],
_readBuffer[position + 6], _readBuffer[position + 7],
_readBuffer, position + 8, packetSize - 8);
// If the connection was closed, we have no more business here
if (IsClosed)
break;
position += packetSize;
}
// Wherever 'position' is up to, is what was successfully processed
if (position > 0)
{
if (position >= _readBufferSize)
_readBufferSize = 0;
else
{
Array.Copy(_readBuffer, position, _readBuffer, 0, _readBufferSize - position);
_readBufferSize -= position;
}
}
}
private void HandleConnectionLost()
{
// :(
Logger.Write("[BYE] 连接丢失. :(");
CurrentMap?.RemoveClient(this);
IsClosed = true;
}
public void SendPacket(byte[] blob, string name = null)
{
var typeA = blob[4];
var typeB = blob[5];
var flags1 = blob[6];
var flags2 = blob[7];
string sendName;
if (name != null)
{
sendName = $"0x{typeA:X2} - 0x{typeB:X2} ({name})";
}
else
{
sendName = $"0x{typeA:X2} - 0x{typeB:X2}";
}
if ((typeA != 0x08 && typeB != 0x0B) && (typeA != 0x08 && typeB != 0x0C) && (typeA != 0x04))
Logger.WriteSend($"[-->] {sendName} (Flags {(PacketFlags)flags1}) ({blob.Length} 字节)");
if (Logger.VerbosePackets)
{
var info = string.Format("[-->] 0x{0:X2} - 0x{1:X2} 数据包:", typeA, typeB);
Logger.WriteHex(info, blob);
}
LogPacket(false, typeA, typeB, flags1, flags2, blob);
OutputArc4?.TransformBlock(blob, 0, blob.Length, blob, 0);
try
{
Socket.Write(blob);
}
catch (Exception ex)
{
Logger.WriteException("发送数据包错误", ex);
}
}
public void SendPacket(byte typeA, byte typeB, byte flags, byte[] data, string name = null)
{
if (data == null)
{
throw new ArgumentNullException(nameof(data), "Data array cannot be null.");
}
using (var memoryStream = new MemoryStream())
using (var binaryWriter = new BinaryWriter(memoryStream))
{
// 写入数据长度总长度为头部4字节 + typeA(1) + typeB(1) + flags(1) + 额外字节的长度))
uint dataLen = (uint)data.Length + 8;
binaryWriter.Write(dataLen); // 自动处理为小端序
// 写入类型A、类型B和标志
binaryWriter.Write(typeA);
binaryWriter.Write(typeB);
binaryWriter.Write(flags);
binaryWriter.Write((byte)0); // 额外的字节
// 写入数据
binaryWriter.Write(data);
// 调用实际发送数据的方法
SendPacket(memoryStream.ToArray(), name);
}
}
public void SendPacket(Packet packet)
{
var h = packet.GetHeader();
SendPacket(h.Type, h.Subtype, h.Flags1, packet.Build(), packet.GetType().Name);
}
private void HandlePacket(byte typeA, byte typeB, byte flags1, byte flags2, byte[] data, uint position, uint size)
{
var handler = PacketHandlers.GetHandlerFor(typeA, typeB);
string packetName;
if (handler != null)
{
packetName = $"0x{typeA:X2} - 0x{typeB:X2} ({handler.GetType().Name})";
}
else
{
packetName = $"0x{typeA:X2} - 0x{typeB:X2}";
}
if ((typeA != 0x04))
Logger.WriteRecv($"[<--] {packetName} (Flags {(PacketFlags)flags1}) ({size + 8} 字节)");
var packet = new byte[size];
Array.Copy(data, position, packet, 0, size);
if (Logger.VerbosePackets && size > 0) // TODO: This is trimming too far?
{
var dataTrimmed = new byte[size];
for (var i = 0; i < size; i++)
dataTrimmed[i] = data[i];
var info = string.Format("[<--] 0x{0:X2} - 0x{1:X2} 数据包:", typeA, typeB);
Logger.WriteHex(info, dataTrimmed);
}
LogPacket(true, typeA, typeB, flags1, flags2, packet);
if (handler != null)
handler.HandlePacket(this, flags1, packet, 0, size);
else
{
//Logger.WriteWarning("[<--未解析] 0x{0:X2} - 0x{1:X2} (UNK) (Flags {2}) ({3} 字节)", typeA,
// typeB, (PacketFlags)flags1, size);
var info = string.Format("[<--未解析] 0x{0:X2} - 0x{1:X2} (UNK) (Flags {2}) ({3} 字节)", typeA,
typeB, (PacketFlags)flags1, size);
Logger.WriteUnkHex(info, packet);
LogUnkClientPacket(typeA, typeB, flags1, flags2, packet);
}
// throw new NotImplementedException();
}
private void LogPacket(bool fromClient, byte typeA, byte typeB, byte flags1, byte flags2, byte[] packet)
{
// Check for and create packets directory if it doesn't exist
var packetPath = Path.Combine("packets", fromClient ? "客户端" : "服务端");
// 记录数据包
LogPacketInternal(packetPath, fromClient, typeA, typeB, flags1, flags2, packet);
}
private void LogUnkClientPacket(byte typeA, byte typeB, byte flags1, byte flags2, byte[] packet)
{
// Check for and create packets directory if it doesn't exist
var packetPath = Path.Combine("UnkClientPackets", "客户端");
// 记录未知数据包
LogPacketInternal(packetPath, true, typeA, typeB, flags1, flags2, packet, "C-unk");
}
private void LogPacketInternal(string directory, bool fromClient, byte typeA, byte typeB, byte flags1, byte flags2, byte[] packet, string customPrefix = null)
{
// 生成格式化日期和时间
var datePart = _server.StartTime.ToString("yyyy-MM-dd");
var timePart = _server.StartTime.ToString("HH-mm-ss-fff");
// Check for and create packets directory if it doesn't exist
var packetPath = Path.Combine(directory, $"0x{typeA:X2}-0x{typeB:X2}", datePart);
// 创建目录
if (!Directory.Exists(packetPath))
{
Directory.CreateDirectory(packetPath);
}
// 构造文件名
var prefix = string.IsNullOrEmpty(customPrefix) ? (fromClient ? "客户端" : "服务端") : customPrefix;
// 确保文件名不包含不允许的字符
var safePrefix = string.Join("_", prefix.Split(Path.GetInvalidFileNameChars()));
var filename = Path.Combine(packetPath, $"{safePrefix}-0x{typeA:X2}-0x{typeB:X2}-{timePart}-{packet.Length}字节.bin");
try
{
using (var stream = new FileStream(filename, FileMode.Append, FileAccess.Write, FileShare.None))
{
if (fromClient || !string.IsNullOrEmpty(customPrefix))
{
stream.Write(new byte[] { typeA, typeB, flags1, flags2 }, 0, 4);
}
stream.Write(packet, 0, packet.Length);
}
}
catch (Exception ex)
{
Console.WriteLine($"记录数据包时出错: {ex.Message}");
}
}
}
public enum Language : uint
{
Japanese = 0,
English = 1
}
}

View File

@ -1,258 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Reflection;
namespace PSO2SERVER
{
public class ConfigComment : Attribute
{
public string Comment;
public ConfigComment(string comment)
{
Comment = comment;
}
}
public class Config
{
private readonly string _configFile = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location) +
Path.DirectorySeparatorChar + "config\\Server.cfg";
// settings
[ConfigComment("服务器对外绑定的地址(支持域名或者IP,优先获取IPV4)")]
public IPAddress BindAddress = IPAddress.Loopback;
[ConfigComment("从客户端向服务器发送命令时要检查的前缀")]
public string CommandPrefix = "|";
[ConfigComment("数据库地址")]
public string DatabaseAddress = "127.0.0.1";
[ConfigComment("数据库端口")]
public string DatabasePort = "3306";
[ConfigComment("数据库表名称")]
public string DatabaseName = "pso2server";
[ConfigComment("数据库用户名")]
public string DatabaseUsername = "root";
[ConfigComment("数据库密码")]
public string DatabasePassword = "root";
[ConfigComment("登录时显示给用户的当天消息")]
public string motd = "Wellcom PSO2SERVER";
[ConfigComment("对所有连接到服务器的客户机执行ping操作的时间(以秒为单位)")]
public double PingTime = 60;
[ConfigComment("为控制台文本启用前景色(在linux上不稳定)")]
public bool UseConsoleColors = true;
[ConfigComment("记录从数据包发送和接收的数据")]
public bool VerbosePackets = false;
public void Load()
{
try
{
// No config exists, save a default one
if (!File.Exists(_configFile))
{
Save(true);
return;
}
var fields = GetType().GetFields();
var lines = File.ReadAllLines(_configFile);
foreach (var option in lines)
{
// Blank Line
if (option.Length == 0)
continue;
// Comment
if (option.StartsWith("//"))
continue;
var split = option.Split('=');
// Trim trailing/leading space
for (var i = 0; i < split.Length; i++)
split[i] = split[i].Trim();
// Check length
if (split.Length != 2)
{
Logger.WriteWarning("[设置] 发现分割大小不正确的配置行");
continue;
}
var field = fields.FirstOrDefault(o => o.Name == split[0]);
if (field != null)
ParseField(field, split[1]);
}
}
catch (Exception ex)
{
Logger.WriteException("[设置] 设置文件载入错误", ex);
}
// Display all settings
//DisplaySettings();
// Some settings require manual refreshing
SettingsChanged();
Logger.WriteInternal("[设置] 设置文件载入完成");
}
private void DisplaySettings()
{
var fields = GetType().GetFields();
foreach (var field in fields)
{
var value = field.GetValue(this);
Logger.WriteInternal($"[设置] 设置项 {field.Name} = {value}");
}
}
public void Save(bool silent = false)
{
try
{
var data = new List<string>();
var fields = GetType().GetFields();
foreach (var field in fields)
SaveField(field, data);
File.WriteAllLines(_configFile, data);
}
catch (Exception ex)
{
Logger.WriteException("保存设置错误", ex);
}
if (!silent)
Logger.WriteInternal("[设置] 设置已保存");
}
public void SettingsChanged()
{
ServerApp.BindAddress = BindAddress;
Logger.VerbosePackets = VerbosePackets;
ServerApp.Instance.Server.PingTimer.Interval = 1000 * PingTime;
}
public bool SetField(string name, string value)
{
var fields = GetType().GetFields();
var field = fields.FirstOrDefault(o => o.Name == name);
if (field != null)
{
ParseField(field, value);
return true;
}
return false;
}
private void ParseField(FieldInfo field, string value)
{
// Bool
if (field.GetValue(this) is bool)
field.SetValue(this, bool.Parse(value));
// Int32
if (field.GetValue(this) is int)
field.SetValue(this, int.Parse(value));
// Float
if (field.GetValue(this) is float)
field.SetValue(this, float.Parse(value));
// Double
if (field.GetValue(this) is double)
field.SetValue(this, double.Parse(value));
// String
if (field.GetValue(this) is string)
{
value = value.Replace("\\n", "\n");
field.SetValue(this, value);
}
// IP Address
if (field.GetValue(this) is IPAddress)
{
try
{
IPAddress ipAddress;
if (IPAddress.TryParse(value, out ipAddress))
{
field.SetValue(this, ipAddress);
}
else
{
// Try resolving domain name to IP address
var addresses = Dns.GetHostAddresses(value);
if (addresses.Length > 0)
{
// Prefer IPv4 addresses over IPv6
ipAddress = addresses.FirstOrDefault(addr => addr.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork) ?? addresses[0];
field.SetValue(this, ipAddress);
}
else
{
Logger.WriteError($"No IP addresses found for domain: {value}");
}
}
}
catch (Exception ex)
{
Logger.WriteError($"Error resolving domain {value}: {ex.Message}");
}
}
// Add more handling for special/custom types as needed
}
private void SaveField(FieldInfo field, List<string> data)
{
// Comment
var attributes = (Attribute[])field.GetCustomAttributes(typeof(ConfigComment), false);
if (attributes.Length > 0)
{
var commentAttr = (ConfigComment)attributes[0];
data.Add("// " + commentAttr.Comment);
}
// IP Address
if (field.GetValue(this).GetType() == typeof(IPAddress))
{
var address = (IPAddress)field.GetValue(this);
data.Add(field.Name + " = " + address);
}
else if (field.GetValue(this).GetType() == typeof(string))
{
var str = (string)field.GetValue(this);
data.Add(field.Name + " = " + str.Replace("\n", "\\n"));
}
else // Basic field
{
data.Add(field.Name + " = " + field.GetValue(this));
}
// Leave a blank line between options
data.Add(string.Empty);
// Add more handling for special/custom types as needed
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -1,212 +0,0 @@
//
// ARC4Managed.cs: Alleged RC4(tm) compatible symmetric stream cipher
// RC4 is a trademark of RSA Security
//
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
using System;
using System.Security.Cryptography;
namespace PSO2SERVER.Crypto
{
// References:
// a. Usenet 1994 - RC4 Algorithm revealed
// http://www.qrst.de/html/dsds/rc4.htm
#if !INSIDE_CORLIB
public
#endif
class Arc4Managed : Rc4, ICryptoTransform
{
private byte[] _key;
private bool _mDisposed;
private byte[] _state;
private byte _x;
private byte _y;
public Arc4Managed()
{
_state = new byte[256];
_mDisposed = false;
}
public override byte[] Key
{
get
{
if (KeyValue == null)
GenerateKey();
return (byte[])KeyValue.Clone();
}
set
{
if (value == null)
throw new ArgumentNullException("Key");
KeyValue = _key = (byte[]) value.Clone();
KeySetup(_key);
}
}
public bool CanReuseTransform
{
get { return false; }
}
public bool CanTransformMultipleBlocks
{
get { return true; }
}
public int InputBlockSize
{
get { return 1; }
}
public int OutputBlockSize
{
get { return 1; }
}
public int TransformBlock(byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer,
int outputOffset)
{
CheckInput(inputBuffer, inputOffset, inputCount);
// check output parameters
if (outputBuffer == null)
throw new ArgumentNullException("outputBuffer");
if (outputOffset < 0)
throw new ArgumentOutOfRangeException("outputOffset", "< 0");
// ordered to avoid possible integer overflow
if (outputOffset > outputBuffer.Length - inputCount)
throw new ArgumentException("outputBuffer overflow");
return InternalTransformBlock(inputBuffer, inputOffset, inputCount, outputBuffer, outputOffset);
}
public byte[] TransformFinalBlock(byte[] inputBuffer, int inputOffset, int inputCount)
{
CheckInput(inputBuffer, inputOffset, inputCount);
var output = new byte[inputCount];
InternalTransformBlock(inputBuffer, inputOffset, inputCount, output, 0);
return output;
}
~Arc4Managed()
{
Dispose(true);
}
protected override void Dispose(bool disposing)
{
if (!_mDisposed)
{
_x = 0;
_y = 0;
if (_key != null)
{
Array.Clear(_key, 0, _key.Length);
_key = null;
}
Array.Clear(_state, 0, _state.Length);
_state = null;
GC.SuppressFinalize(this);
_mDisposed = true;
}
}
public override ICryptoTransform CreateEncryptor(byte[] rgbKey, byte[] rgvIv)
{
Key = rgbKey;
return this;
}
public override ICryptoTransform CreateDecryptor(byte[] rgbKey, byte[] rgvIv)
{
Key = rgbKey;
return CreateEncryptor();
}
public override void GenerateIV()
{
// not used for a stream cipher
IV = new byte[0];
}
public override void GenerateKey()
{
throw new NotImplementedException();
}
private void KeySetup(byte[] key)
{
byte index1 = 0;
byte index2 = 0;
for (var counter = 0; counter < 256; counter++)
_state[counter] = (byte) counter;
_x = 0;
_y = 0;
for (var counter = 0; counter < 256; counter++)
{
index2 = (byte) (key[index1] + _state[counter] + index2);
// swap byte
var tmp = _state[counter];
_state[counter] = _state[index2];
_state[index2] = tmp;
index1 = (byte) ((index1 + 1)%key.Length);
}
}
private void CheckInput(byte[] inputBuffer, int inputOffset, int inputCount)
{
if (inputBuffer == null)
throw new ArgumentNullException("inputBuffer");
if (inputOffset < 0)
throw new ArgumentOutOfRangeException("inputOffset", "< 0");
if (inputCount < 0)
throw new ArgumentOutOfRangeException("inputCount", "< 0");
// ordered to avoid possible integer overflow
if (inputOffset > inputBuffer.Length - inputCount)
throw new ArgumentException("inputBuffer overflow");
}
private int InternalTransformBlock(byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer,
int outputOffset)
{
for (var counter = 0; counter < inputCount; counter++)
{
_x = (byte) (_x + 1);
_y = (byte) (_state[_x] + _y);
// swap byte
var tmp = _state[_x];
_state[_x] = _state[_y];
_state[_y] = tmp;
var xorIndex = (byte) (_state[_x] + _state[_y]);
outputBuffer[outputOffset + counter] = (byte) (inputBuffer[inputOffset + counter] ^ _state[xorIndex]);
}
return inputCount;
}
}
}

View File

@ -1,84 +0,0 @@
using System;
using System.IO;
using Org.BouncyCastle.OpenSsl;
using Org.BouncyCastle.Security;
using Org.BouncyCastle.Crypto;
using Org.BouncyCastle.Crypto.Parameters;
using System.Security.Cryptography;
using PSO2SERVER;
public static class KeyLoader
{
public static RSACryptoServiceProvider LoadPrivateKeyFromPem(string filePath)
{
using (var reader = new StreamReader(filePath))
{
var pemReader = new PemReader(reader);
var privateKeyParams = (RsaPrivateCrtKeyParameters)pemReader.ReadObject();
var rsaParams = new RSAParameters
{
Modulus = privateKeyParams.Modulus.ToByteArrayUnsigned(),
Exponent = privateKeyParams.PublicExponent.ToByteArrayUnsigned(),
P = privateKeyParams.P.ToByteArrayUnsigned(),
Q = privateKeyParams.Q.ToByteArrayUnsigned(),
DP = privateKeyParams.DP.ToByteArrayUnsigned(),
DQ = privateKeyParams.DQ.ToByteArrayUnsigned(),
InverseQ = privateKeyParams.QInv.ToByteArrayUnsigned(),
D = privateKeyParams.Exponent.ToByteArrayUnsigned()
};
var rsa = new RSACryptoServiceProvider();
rsa.ImportParameters(rsaParams);
return rsa;
}
}
public static RSACryptoServiceProvider LoadPublicKeyFromPem(string filePath)
{
using (var reader = new StreamReader(filePath))
{
var pemReader = new PemReader(reader);
var publicKeyParams = (RsaKeyParameters)pemReader.ReadObject();
var rsaParams = new RSAParameters
{
Modulus = publicKeyParams.Modulus.ToByteArrayUnsigned(),
Exponent = publicKeyParams.Exponent.ToByteArrayUnsigned()
};
var rsa = new RSACryptoServiceProvider();
rsa.ImportParameters(rsaParams);
return rsa;
}
}
public static void SaveKeyToFile(string filePath, byte[] keyBlob)
{
using (FileStream outFile = File.Create(filePath))
{
outFile.Write(keyBlob, 0, keyBlob.Length);
}
}
public static void ProcessKeyFiles(string pemFile, string blobFile, bool isPrivateKey, bool isGenerating)
{
if (File.Exists(pemFile) && !File.Exists(blobFile) && isGenerating)
{
Logger.Write("[KEY] 发现{0}文件, 正在生成新的{1}密钥 {2}...", pemFile, isPrivateKey ? "私有" : "公共", blobFile);
RSACryptoServiceProvider rsa = isPrivateKey ? LoadPrivateKeyFromPem(pemFile) : LoadPublicKeyFromPem(pemFile);
byte[] cspBlob = rsa.ExportCspBlob(isPrivateKey);
SaveKeyToFile(blobFile, cspBlob);
}
}
public static void GenerateAndSaveKeyIfNotExists(RSACryptoServiceProvider rsa, string keyBlobFile, bool isPrivateKey)
{
if (!File.Exists(keyBlobFile))
{
Logger.WriteWarning("[KEY] 未找到 {0} 文件, 正在生成新的{1}密钥...", keyBlobFile, isPrivateKey ? "私有" : "公共");
byte[] cspBlob = rsa.ExportCspBlob(isPrivateKey);
SaveKeyToFile(keyBlobFile, cspBlob);
}
}
}

View File

@ -1,89 +0,0 @@
//
// RC4.cs: RC4(tm) symmetric stream cipher
// RC4 is a trademark of RSA Security
//
// Author:
// Sebastien Pouliot (sebastien@xamarin.com)
//
// (C) 2003 Motus Technologies Inc. (http://www.motus.com)
// Copyright 2013 Xamarin Inc. (http://www.xamarin.com)
//
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
using System;
using System.Security.Cryptography;
namespace PSO2SERVER.Crypto
{
#if !INSIDE_CORLIB
public
#endif
abstract class Rc4 : SymmetricAlgorithm
{
private static readonly KeySizes[] SLegalBlockSizes =
{
new KeySizes(64, 64, 0)
};
private static readonly KeySizes[] SLegalKeySizes =
{
new KeySizes(40, 2048, 8)
};
protected Rc4()
{
KeySizeValue = 128;
BlockSizeValue = 64;
FeedbackSizeValue = BlockSizeValue;
LegalBlockSizesValue = SLegalBlockSizes;
LegalKeySizesValue = SLegalKeySizes;
}
// required for compatibility with .NET 2.0
public override byte[] IV
{
get { return new byte[0]; }
set
{
if (value == null) throw new ArgumentNullException("value");
}
}
public new static Rc4 Create()
{
#if FULL_AOT_RUNTIME
return new ARC4Managed ();
#else
return Create("RC4");
#endif
}
public new static Rc4 Create(string algName)
{
var o = CryptoConfig.CreateFromName(algName) ?? new Arc4Managed();
// in case machine.config isn't configured to use
// any RC4 implementation
return (Rc4) o;
}
}
}

View File

@ -1,356 +0,0 @@
using System;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity;
using System.IO;
using System.Runtime.InteropServices;
using MySql.Data.EntityFramework;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol;
using static PSO2SERVER.Models.CharacterStruct;
namespace PSO2SERVER.Database
{
public class Account
{
[Key]
public int AccountId { get; set; }
public string Username { get; set; }
public string Password { get; set; }
public string Nickname { get; set; }
public string SettingsIni { get; set; }
public int TextLang { get; set; }
public int VoiceLang { get; set; }
public int TextLang2 { get; set; }
public int LangLang { get; set; }
public string LanguageCode { get; set; }
public int IsGM { get; set; }
public int PacketType { get; set; }
public string PSNID { get; set; }
}
public class AccountNetInterFace
{
[Key]
public int id { get; set; }
public int AccountId { get; set; }
public string Username { get; set; }
public int State { get; set; }
public string Mac { get; set; }
}
public class AccountSystemInfo
{
[Key]
public int id { get; set; }
public int AccountId { get; set; }
public string Username { get; set; }
public string CpuInfo { get; set; }
public string VideoInfo { get; set; }
public long Vram { get; set; }
public long TotalRam { get; set; }
public int Unk1 { get; set; }
public int Unk2 { get; set; }
public string WindowsVersion { get; set; }
public string WindowSize { get; set; }
public string AudioDevices { get; set; }
public string Unk4 { get; set; }
public string VideoDriver { get; set; }
public long TotalDiskSpace { get; set; }
public long FreeDiskSpace { get; set; }
}
public class Character
{
// Probably more info than this
[Key, Column(Order = 1)]
public int CharacterID { get; set; }
[Key, Column(Order = 2)]
public int AccountID { get; set; }
public int Unk1 { get; set; }
public int VoiceType { get; set; }
public short Unk2 { get; set; }
public short VoicePitch { get; set; }
public string Name { get; set; }
public LooksParam Looks { get; set; }
public byte[] LooksBinary
{
get
{
PacketWriter w = new PacketWriter();
w.WriteStruct(Looks);
return w.ToArray();
}
set
{
Looks = Helper.ByteArrayToStructure<LooksParam>(value);
}
}
public int Unk3 { get; set; }
public JobParam Jobs { get; set; }
public byte[] JobsBinary
{
get
{
PacketWriter w = new PacketWriter();
w.WriteStruct(Jobs);
return w.ToArray();
}
set
{
Jobs = Helper.ByteArrayToStructure<JobParam>(value);
}
}
public byte[] Unk4 { get; set; } = new byte[148];
public byte[] Fulldata { get; set; }
public PSO2Items[] EquipedItems { get; set; } = new PSO2Items[10];
public byte[] EquipedItemsBinary
{
get
{
PacketWriter w = new PacketWriter();
foreach (var item in EquipedItems)
{
w.WriteStruct(item);
}
return w.ToArray();
}
set
{
// 每个PSO2Items的大小包含uuid、ItemId、Items
int itemSize = Marshal.SizeOf(typeof(PSO2Items));
// 计算字节数组中包含多少个PSO2Items对象
int itemCount = value.Length / itemSize;
// 创建一个新的数组来存储这些对象
EquipedItems = new PSO2Items[itemCount];
// 逐个反序列化
for (int i = 0; i < itemCount; i++)
{
// 提取对应的字节块
byte[] itemBytes = new byte[itemSize];
Array.Copy(value, i * itemSize, itemBytes, 0, itemSize);
// 使用Helper.ByteArrayToStructure来反序列化每个PSO2Items对象
EquipedItems[i] = Helper.ByteArrayToStructure<PSO2Items>(itemBytes);
}
}
}
public uint Playe_time { get; set; }
public byte[] BuildCharacterByteArray()
{
PacketWriter writer = new PacketWriter();
// 序列化字段CharacterID, AccountID, Unk1, VoiceType, Unk2, VoicePitch
writer.Write((uint)CharacterID);
writer.Write((uint)AccountID);
writer.Write((uint)Unk1);
writer.Write((uint)VoiceType);
writer.Write((ushort)Unk2);
writer.Write(VoicePitch);
// 序列化 name (假设固定长度字符串16个字节UTF-16 编码)
writer.WriteFixedLengthUtf16(Name, 0x10);
// 序列化 Looks
writer.WriteStruct(Looks);
// 序列化 Unk3
writer.Write((uint)Unk3);
// 序列化 Jobs
writer.WriteStruct(Jobs);
// 序列化 Unk4
writer.WriteBytes(0, 148);
// 最后返回字节流
return writer.ToArray();
}
public virtual Account Account { get; set; }
}
public class NPC
{
[Key, Column(Order = 1)]
public int EntityID { get; set; }
[Key, Column(Order = 2)]
public string ZoneName { get; set; }
public string NPCName { get; set; }
public float RotX { get; set; }
public float RotY { get; set; }
public float RotZ { get; set; }
public float RotW { get; set; }
public float PosX { get; set; }
public float PosY { get; set; }
public float PosZ { get; set; }
public int is_active { get; set; }
}
public class GameObject
{
[Key, Column(Order = 1)]
public int ObjectID { get; set; }
[Key, Column(Order = 2)]
public string ZoneName { get; set; }
public string ObjectName { get; set; }
public byte[] ObjectFlags { get; set; }
public float RotX { get; set; }
public float RotY { get; set; }
public float RotZ { get; set; }
public float RotW { get; set; }
public float PosX { get; set; }
public float PosY { get; set; }
public float PosZ { get; set; }
}
public class ServerInfo
{
[Key, MaxLength(255)]
public string Info { get; set; }
public string Setting { get; set; }
}
public class Teleport
{
[Key, Column(Order = 1)]
public string ZoneName { get; set; }
[Key, Column(Order = 2)]
public int ObjectID { get; set; }
public float RotX { get; set; }
public float RotY { get; set; }
public float RotZ { get; set; }
public float RotW { get; set; }
public float PosX { get; set; }
public float PosY { get; set; }
public float PosZ { get; set; }
}
[DbConfigurationType(typeof(MySqlEFConfiguration))]
public class ServerEf : DbContext
{
public DbSet<Account> Accounts { get; set; }
public DbSet<AccountNetInterFace> AccountsNetInterFaces { get; set; }
public DbSet<AccountSystemInfo> AccountsSystemInfoes { get; set; }
public DbSet<Character> Characters { get; set; }
public DbSet<GameObject> GameObjects { get; set; }
public DbSet<NPC> NPCs { get; set; }
public DbSet<ServerInfo> ServerInfoes { get; set; }
public DbSet<Teleport> Teleports { get; set; }
public ServerEf()
: base(
string.Format("server={0};port={1};database={2};username={3};password={4}",
ServerApp.Config.DatabaseAddress,
ServerApp.Config.DatabasePort,
ServerApp.Config.DatabaseName,
ServerApp.Config.DatabaseUsername,
ServerApp.Config.DatabasePassword)
)
{
}
public void SetupDB()
{
try
{
foreach (
var f in
Directory.EnumerateFiles(Directory.GetCurrentDirectory() + "/Resources/sql/scripts/", "*.sql"))
{
Logger.WriteInternal("[DBC] 执行数据库脚本 {0}", f);
Database.ExecuteSqlCommand(File.ReadAllText(f));
}
var revision = ServerInfoes.Find("Revision");
if (revision == null)
{
revision = new ServerInfo { Info = "Revision", Setting = "0" };
ServerInfoes.Add(revision);
//TODO Possibly move this somewhere else?
Database.ExecuteSqlCommand("ALTER TABLE Accounts AUTO_INCREMENT=10000000");
}
SaveChanges();
Logger.WriteInternal("[DBC] 加载数据集修订的数据库 {0}", revision.Setting);
}
catch (Exception ex)
{
Logger.WriteException("数据库异常", ex);
}
}
public bool TestDatabaseConnection2()
{
try
{
using (var context = new ServerEf())
{
// 执行一个简单的查询来测试数据库连接
context.Database.ExecuteSqlCommand("SELECT 1");
Logger.WriteInternal("[DBT] 数据库连接成功。");
return true;
}
}
catch (Exception ex)
{
Logger.WriteException("数据库连接异常", ex);
return false;
}
}
public bool TestDatabaseConnection()
{
try
{
using (var context = new ServerEf())
{
context.Database.Initialize(force: false);
Logger.WriteInternal("[DBT] 数据库连接成功。");
return true;
}
}
catch (Exception ex)
{
Logger.WriteException("数据库连接异常", ex);
return false;
}
}
}
}

View File

@ -1,123 +0,0 @@
using System;
using System.ComponentModel;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
namespace PSO2SERVER
{
public static class Helper
{
public static string ByteArrayToString(byte[] ba)
{
var hex = new StringBuilder(ba.Length * 2);
foreach (var b in ba)
hex.AppendFormat("{0:x2}", b);
return hex.ToString();
}
public static byte[] StringToByteArray(String hex)
{
var numberChars = hex.Length;
var bytes = new byte[numberChars / 2];
for (var i = 0; i < numberChars; i += 2)
bytes[i / 2] = Convert.ToByte(hex.Substring(i, 2), 16);
return bytes;
}
public static T ByteArrayToStructure<T>(byte[] bytes) where T : struct
{
var handle = GCHandle.Alloc(bytes, GCHandleType.Pinned);
var stuff = (T)Marshal.PtrToStructure(handle.AddrOfPinnedObject(),
typeof(T));
handle.Free();
return stuff;
}
public static string ObjectToString(object obj)
{
var data = string.Empty;
data += "{ ";
data = TypeDescriptor.GetProperties(obj).Cast<PropertyDescriptor>().Aggregate(data, (current, descriptor) => current + string.Format("{0} = {1}, ", descriptor.Name, descriptor.GetValue(obj)));
data = data.Remove(data.Length - 3);
data += " }";
return data;
}
public static int FindPlayerByUsername(string name)
{
for (var i = 0; i < ServerApp.Instance.Server.Clients.Count; i++)
if (name.ToLower() == ServerApp.Instance.Server.Clients[i]._account.Username.ToLower())
return i;
return -1;
}
public static ushort PacketTypeToUShort(uint type, uint subtype)
{
return (ushort)((type << 8) | subtype);
}
#region Float Manipulation
public static unsafe float UIntToFloat(uint input)
{
var fp = (float*)(&input);
return *fp;
}
public static unsafe uint FloatToUInt(float input)
{
var ip = (uint*)(&input);
return *ip;
}
public static float FloatFromHalfPrecision(ushort value)
{
if ((value & 0x7FFF) != 0)
{
var sign = (uint)((value & 0x8000) << 16);
var exponent = (uint)(((value & 0x7C00) >> 10) + 0x70) << 23;
var mantissa = (uint)((value & 0x3FF) << 13);
return UIntToFloat(sign | exponent | mantissa);
}
return 0;
}
public static ushort FloatToHalfPrecision(float value)
{
var ivalue = FloatToUInt(value);
if ((ivalue & 0x7FFFFFFF) != 0)
{
var sign = (ushort)((ivalue >> 16) & 0x8000);
var exponent = (ushort)(((ivalue & 0x7F800000) >> 23) - 0x70);
if ((exponent & 0xFFFFFFE0) != 0)
{
return (ushort)((exponent >> 17) ^ 0x7FFF | sign);
}
var a = (ushort)((ivalue & 0x7FFFFF) >> 13);
var b = (ushort)(exponent << 10);
return (ushort)(a | b | sign);
}
return (ushort)(ivalue >> 16);
}
#endregion
#region Timestamps
public static long Timestamp(DateTime time)
{
return time.ToFileTimeUtc() / 10000;
}
public static DateTime Timestamp(long stamp)
{
return DateTime.FromFileTimeUtc(stamp * 10000);
}
#endregion
}
}

View File

@ -1,8 +0,0 @@
<linker>
<assembly fullname="System.Diagnostics.DiagnosticSource">
<type fullname="System.Diagnostics.Metrics.MetricsEventSource">
<!-- Used by System.Private.CoreLib via reflection to init the EventSource -->
<method name="GetInstance" />
</type>
</assembly>
</linker>

View File

@ -1,45 +0,0 @@
using Newtonsoft.Json;
using PSO2SERVER.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
public class ClassDefaultData
{
public List<DefaultClassesData> defaultClassesDatas = new List<DefaultClassesData>();
}
public class DefaultClassesData
{
[JsonProperty("class")]
public string ClassNames = string.Empty;
[JsonProperty("data")]
public DefaultClassData Data { get; set; } = new DefaultClassData();
}
public class DefaultClassData
{
[JsonProperty("items")]
public List<DefaultItem> Items { get; set; } = new List<DefaultItem>();
[JsonProperty("subpalettes")]
[JsonConverter(typeof(FixedListConverter<SubPalette>))]
public FixedList<SubPalette> Subpalettes { get; set; } = new FixedList<SubPalette>(6);
}
public class DefaultItem
{
[JsonProperty("item_data")]
public PSO2Items Item_data { get; set; } = new PSO2Items();
[JsonProperty("weapon_palette_data")]
public WeaponPalette Weapon_palette_data { get; set; } = new WeaponPalette();
[JsonProperty("weapon_palette_id")]
public byte Weapon_palette_id { get; set; } = new byte();
[JsonProperty("unit_equiped_id")]
public byte Unit_equiped_id { get; set; } = new byte();
}
}

View File

@ -1,189 +0,0 @@
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;
using PSO2SERVER.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
// JSON 序列化和反序列化的自定义处理
public class FixedListConverter<T> : JsonConverter<FixedList<T>>
{
// 读取 JSON 转换为 FixedList<T>
public override FixedList<T> ReadJson(JsonReader reader, Type objectType, FixedList<T> existingValue, bool hasExistingValue, JsonSerializer serializer)
{
var list = serializer.Deserialize<List<T>>(reader);
var fixedList = new FixedList<T>(list.Count);
// 将 list 中的元素添加到 FixedList 中
fixedList.SetListItem(list);
return fixedList;
}
// 将 FixedList<T> 转换为 JSON
public override void WriteJson(JsonWriter writer, FixedList<T> value, JsonSerializer serializer)
{
var list = new List<T>(value.Length);
for (int i = 0; i < value.Length; i++)
{
list.Add(value[i]);
}
serializer.Serialize(writer, list);
}
}
public class EnumJsonConverter
{
public static string EnumToJson<T>(T enumValue)
{
// 序列化枚举值为 JSON 字符串
return JsonConvert.SerializeObject(enumValue, Formatting.Indented, new StringEnumConverter());
}
public static T JsonToEnum<T>(string json)
{
// 反序列化 JSON 字符串为枚举值
return JsonConvert.DeserializeObject<T>(json);
}
}
public class QuestPartyTypeConverter : JsonConverter<QuestPartyType>
{
public override void WriteJson(JsonWriter writer, QuestPartyType value, JsonSerializer serializer)
{
// Optionally, you can use the enum name or custom string
writer.WriteValue(value.ToString());
}
public override QuestPartyType ReadJson(JsonReader reader, Type objectType, QuestPartyType existingValue, bool hasExistingValue, JsonSerializer serializer)
{
// Read the string and convert it back to the enum value
var value = reader.Value.ToString();
return Enum.TryParse(value, out QuestPartyType result) ? result : QuestPartyType.Solo; // Default to Solo if parsing fails
}
}
public class ObjectTypeConverter : JsonConverter<ObjectType>
{
public override ObjectType ReadJson(JsonReader reader, Type objectType, ObjectType existingValue, bool hasExistingValue, JsonSerializer serializer)
{
if (reader.TokenType == JsonToken.String)
{
string value = reader.Value.ToString();
ObjectType objectTypeResult;
switch (value)
{
case "Player":
objectTypeResult = ObjectType.Player;
break;
case "Map":
objectTypeResult = ObjectType.Map;
break;
case "Object":
objectTypeResult = ObjectType.Object;
break;
case "StaticObject":
objectTypeResult = ObjectType.StaticObject;
break;
case "PartyQuest":
objectTypeResult = ObjectType.Quest;
break;
case "Party":
objectTypeResult = ObjectType.Party;
break;
case "World":
objectTypeResult = ObjectType.World;
break;
case "APC":
objectTypeResult = ObjectType.APC;
break;
default:
objectTypeResult = ObjectType.Unknown; // 默认返回 Unknown
break;
}
return objectTypeResult;
}
if (reader.TokenType == JsonToken.Integer)
{
// 如果是数字类型,直接转换为枚举
return (ObjectType)reader.Value;
}
return ObjectType.Unknown; // 默认返回 Unknown
}
public override void WriteJson(JsonWriter writer, ObjectType value, JsonSerializer serializer)
{
string stringValue;
switch (value)
{
case ObjectType.Player:
stringValue = "Player";
break;
case ObjectType.Map:
stringValue = "Map";
break;
case ObjectType.Object:
stringValue = "Object";
break;
case ObjectType.StaticObject:
stringValue = "StaticObject";
break;
case ObjectType.Quest:
stringValue = "PartyQuest";
break;
case ObjectType.Party:
stringValue = "Party";
break;
case ObjectType.World:
stringValue = "World";
break;
case ObjectType.APC:
stringValue = "APC";
break;
case ObjectType.Unknown:
stringValue = "Unknown";
break;
default:
stringValue = "Undefined";
break;
}
writer.WriteValue(stringValue);
}
}
// Custom converter for AsciiString, assuming it's a fixed length string.
public class AsciiStringConverter : JsonConverter
{
public override bool CanConvert(System.Type objectType)
{
return objectType == typeof(string);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
string str = value as string;
if (str != null)
{
writer.WriteValue(str.PadRight(32, ' ')); // Ensure the string is 0x20 (32) bytes long.
}
else
{
writer.WriteNull();
}
}
public override object ReadJson(JsonReader reader, System.Type objectType, object existingValue, JsonSerializer serializer)
{
string str = (string)reader.Value;
return str?.TrimEnd(); // Removing padding spaces
}
}
}

View File

@ -1,106 +0,0 @@
using Newtonsoft.Json;
using System;
using System.IO;
using System.Collections.Generic;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
public class JsonRead
{
// 泛型方法:读取并反序列化 JSON 文件
public static async Task<T> DeserializeJsonAsync<T>(string filePath, JsonSerializerSettings settings = null)
{
if (string.IsNullOrEmpty(filePath))
{
Logger.Write("文件路径不能为空");
return default(T);
}
if (!File.Exists(filePath))
{
Logger.Write($"文件不存在: {filePath}");
return default(T);
}
try
{
// 异步读取文件内容
string json = await Task.Run(() => File.ReadAllText(filePath));
// 如果文件内容为空,记录日志并返回默认值
if (string.IsNullOrWhiteSpace(json))
{
Logger.Write($"文件内容为空: {filePath}");
return default(T);
}
// 反序列化 JSON
return JsonConvert.DeserializeObject<T>(json, settings ?? new JsonSerializerSettings());
}
catch (JsonException jsonEx)
{
// 捕获 JSON 反序列化异常
Logger.Write($"JSON 反序列化错误:{jsonEx.Message} 文件: {filePath}");
return default(T);
}
catch (FileNotFoundException fnfEx)
{
// 捕获文件找不到异常
Logger.Write($"文件未找到错误:{fnfEx.Message} 文件: {filePath}");
return default(T);
}
catch (UnauthorizedAccessException uaEx)
{
// 捕获权限相关的异常
Logger.Write($"权限错误:{uaEx.Message} 文件: {filePath}");
return default(T);
}
catch (Exception ex)
{
// 捕获其他未知异常
Logger.Write($"读取或反序列化文件 {filePath} 时发生异常: {ex.Message}");
return default(T);
}
}
// 同步版本:用于不需要异步操作的场景
public static T DeserializeJson<T>(string filePath, JsonSerializerSettings settings = null)
{
if (string.IsNullOrEmpty(filePath))
{
Logger.Write("文件路径不能为空");
return default(T);
}
if (!File.Exists(filePath))
{
Logger.Write($"文件不存在: {filePath}");
return default(T);
}
try
{
string json = File.ReadAllText(filePath);
if (string.IsNullOrWhiteSpace(json))
{
Logger.Write($"文件内容为空: {filePath}");
return default(T);
}
return JsonConvert.DeserializeObject<T>(json, settings ?? new JsonSerializerSettings());
}
catch (JsonException jsonEx)
{
Logger.Write($"JSON 反序列化错误:{jsonEx.Message} 文件: {filePath}");
return default(T);
}
catch (Exception ex)
{
Logger.Write($"读取或反序列化文件 {filePath} 时发生异常: {ex.Message}");
return default(T);
}
}
}
}

View File

@ -1,94 +0,0 @@
using Google.Protobuf.Compiler;
using Newtonsoft.Json;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
public class JsonTest
{
public static void JsonReadTest()
{
// 测试数据文件路径
string jsonFilePath1 = "data\\maps\\lobby\\data.json";
string jsonFilePath2 = "data\\maps\\lobby\\events\\main_lobby_1\\all_events.json";
string jsonFilePath3 = "data\\item_attrs.json";
string jsonFilePath4 = "data\\quests\\Story Quests\\EP1\\700000 - An Encounter with Xion\\data.json";
// 读取并反序列化 JSON 文件
var map = JsonRead.DeserializeJson<MapData>(jsonFilePath1);
var mapEvent = JsonRead.DeserializeJson<List<EventData>>(jsonFilePath2)?.FirstOrDefault();
var attributes = JsonRead.DeserializeJson<ItemAttributesRootObject>(jsonFilePath3);
var quest = JsonRead.DeserializeJson<QuestData>(jsonFilePath4);
var quests = new List<QuestData>();
quests.Add(quest);
var avaq = QuestAvailablePacket.Load(quests);
// 获取类型信息
Type type = avaq.GetType();
// 获取所有公共属性
PropertyInfo[] properties = type.GetProperties();
Logger.Write("属性:");
foreach (var property in properties)
{
// 获取属性的值
var value = property.GetValue(avaq);
Logger.Write($"{property.Name}: {value}");
}
// 获取所有字段(包括私有字段)
FieldInfo[] fields = type.GetFields(BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Public);
Logger.Write("\n字段:");
foreach (var field in fields)
{
// 获取字段的值
var value = field.GetValue(avaq);
Logger.Write($"{field.Name}: {value}");
}
//// 输出 MapData 信息
//if (map != null)
//{
// Logger.Write($"map_object ID: {map.Mapdata.map_object.ID}");
// Logger.Write($"Zones[0].Name: {map.Zones[0].Name}");
// Logger.Write($"Zones[0].Is_special_zone: {map.Zones[0].Is_special_zone}");
//}
//// 输出 EventData 信息
//if (mapEvent != null)
//{
// Logger.Write($"map_event zone_id: {mapEvent.zone_id}");
// Logger.Write($"map_event is_active: {mapEvent.is_active}");
// Logger.Write($"map_event objName: {mapEvent.data.objName}");
//}
//// 输出 ItemAttributesPC 信息
//if (attributes != null)
//{
// Logger.Write($"PC unk1: {attributes.PC.Unk1}");
// ItemAttributesPC.LogWeapons(attributes.PC.Weapons);
// ItemAttributesPC.LogData17(attributes.PC.Data17);
//}
// 输出 MapData 信息
if (quest != null)
{
Logger.Write($"quest date: {quest.QuestDefiniton.Date}");
Logger.Write($"quest party_type: {quest.QuestDefiniton.PartyType}");
}
}
}
}

View File

@ -1,208 +0,0 @@
using Newtonsoft.Json;
using Org.BouncyCastle.Asn1.Pkcs;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
public class MapData
{
[JsonProperty("map_data")]
public LoadingLevelPacket Mapdata { get; set; }
[JsonProperty("objects")]
public List<ObjectData> Objects { get; set; }
[JsonProperty("events")]
public List<EventData> Events { get; set; }
[JsonProperty("npcs")]
public List<NPCData> Npcs { get; set; }
[JsonProperty("transporters")]
public List<TransporterData> Transporters { get; set; }
[JsonProperty("luas")]
public Dictionary<string, string> Luas { get; set; }
[JsonProperty("init_map")]
public uint InitMap { get; set; }//zone_id
[JsonProperty("zones")]
public List<ZoneData> Zones { get; set; }
// 默认构造函数
public MapData()
{
// 使用默认值初始化字段
Mapdata = new LoadingLevelPacket();
Objects = new List<ObjectData>();
Events = new List<EventData>();
Npcs = new List<NPCData>();
Transporters = new List<TransporterData>();
Luas = new Dictionary<string, string>();
InitMap = new uint();
Zones = new List<ZoneData>();
}
}
public class ObjectData
{
[JsonProperty("zone_id")]
public uint zone_id { get; set; }
[JsonProperty("is_active")]
public bool is_active { get; set; }
[JsonProperty("data")]
public ObjectSpawnPacket data { get; set; }
[JsonProperty("lua_data")]
public string lua_data { get; set; }
}
public class EventData
{
[JsonProperty("zone_id")]
public uint zone_id { get; set; }
[JsonProperty("is_active")]
public bool is_active { get; set; }
[JsonProperty("data")]
public EventSpawnPacket data { get; set; }
[JsonProperty("lua_data")]
public string lua_data { get; set; }
}
public class NPCData
{
[JsonProperty("zone_id")]
public uint zone_id { get; set; }
[JsonProperty("is_active")]
public bool is_active { get; set; }
[JsonProperty("data")]
public NPCSpawnPacket data { get; set; }
[JsonProperty("lua_data")]
public string lua_data { get; set; }
}
public class TransporterData
{
[JsonProperty("zone_id")]
public uint zone_id { get; set; }
[JsonProperty("is_active")]
public bool is_active { get; set; }
[JsonProperty("data")]
public TransporterSpawnPacket data { get; set; }
[JsonProperty("lua_data")]
public string lua_data { get; set; }
}
public class EnemySpawn
{
public string EnemyName { get; set; }
public uint SpawnCategory { get; set; }
// 默认构造函数
public EnemySpawn()
{
EnemyName = string.Empty;
SpawnCategory = 0;
}
// 带参构造函数
public EnemySpawn(string enemyName, uint spawnCategory)
{
EnemyName = enemyName;
SpawnCategory = spawnCategory;
}
}
public class ZoneData
{
[JsonProperty("name")]
public string Name { get; set; }
[JsonProperty("is_special_zone")]
public bool Is_special_zone { get; set; }
[JsonProperty("zone_id")]
public uint Zone_id { get; set; }
[JsonProperty("settings")]
public ZoneSettings Settings { get; set; }
[JsonProperty("default_location")]
public PSOLocation Default_location { get; set; }
[JsonProperty("enemies")]
public List<EnemySpawn> Enemies { get; set; }
[JsonProperty("chunks")]
public List<ZoneChunk> Chunks { get; set; }
// 默认构造函数
public ZoneData()
{
Name = string.Empty;
Is_special_zone = false;
Zone_id = new uint();
Settings = new ZoneSettings();
Default_location = new PSOLocation();
Enemies = new List<EnemySpawn>();
Chunks = new List<ZoneChunk>();
}
}
public class ZoneChunk
{
[JsonProperty("zone_id")]
public uint ZoneId { get; set; }
[JsonProperty("chunk_id")]
public uint ChunkId { get; set; }
[JsonProperty("enemy_spawn_type")]
public EnemySpawnType EnemySpawnType { get; set; }
[JsonProperty("enemy_spawn_points")]
public List<PSOLocation> EnemySpawnPoints { get; set; }
// 默认构造函数
public ZoneChunk()
{
ZoneId = new uint();
ChunkId = 0;
EnemySpawnType = new Disabled();
EnemySpawnPoints = new List<PSOLocation>();
}
}
public abstract class EnemySpawnType
{
// 基类方法,用来获取默认值
public static EnemySpawnType Default => new Disabled();
}
public class Disabled : EnemySpawnType { }
public class Automatic : EnemySpawnType
{
public uint Min { get; set; }
public uint Max { get; set; }
public Automatic(uint min, uint max)
{
Min = min;
Max = max;
}
}
public class AutomaticWithRespawn : EnemySpawnType
{
public uint Min { get; set; }
public uint Max { get; set; }
public TimeSpan RespawnTime { get; set; }
public AutomaticWithRespawn(uint min, uint max, TimeSpan respawnTime)
{
Min = min;
Max = max;
RespawnTime = respawnTime;
}
}
public class Manual : EnemySpawnType { }
}

View File

@ -1,46 +0,0 @@
using Newtonsoft.Json;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Json
{
public class QuestData
{
[JsonProperty("definition")]
public QuestDef QuestDefiniton { get; set; } = new QuestDef();
[JsonProperty("difficulties")]
public QuestDiff Difficulty { get; set; } = new QuestDiff();
[JsonProperty("map")]
public MapData Mapdata { get; set; } = new MapData();
[JsonProperty("enemies")]
public List<EnemyData> Enemies { get; set; } = new List<EnemyData>();
[JsonProperty("immediate_move")]
public bool ImmediateMove { get; set; } = false;
public QuestData()
{
QuestDefiniton = new QuestDef();
Difficulty = new QuestDiff();
Mapdata = new MapData();
Enemies = new List<EnemyData>();
ImmediateMove = false;
}
}
public class EnemyData
{
[JsonProperty("difficulty")]
public ushort Difficulty { get; set; } = 0;
[JsonProperty("mapid")]
public uint mapid { get; set; } = 0;
[JsonProperty("data")]
public EnemySpawnPacket data { get; set; } = new EnemySpawnPacket();
[JsonProperty("lua_data")]
public string lua_data { get; set; } = string.Empty;
}
}

View File

@ -1,367 +0,0 @@
using System;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Runtime.Serialization.Formatters.Binary;
using System.Text;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER
{
/// <summary>
/// Wrapper for Console's Write and WriteLine functions to add coloring as well as integrate it into the Console System
/// and add dumping to a log file.
/// </summary>
public static class Logger
{
private static readonly StreamWriter Writer = new StreamWriter("SERVER.log", true);
public static bool VerbosePackets = false;
private static void AddLine(ConsoleColor color, string text)
{
// Return if we don't have a ConsoleSystem created yet
if (ServerApp.ConsoleSystem == null) return;
ServerApp.ConsoleSystem.AddLine(color, text);
}
public static void Write(string text, params object[] args)
{
AddLine(ConsoleColor.White, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteObj(object obj)
{
if (obj == null)
{
WriteError("无法分析空的对象.");
return;
}
Type objType = obj.GetType();
Write($"当前分析对象: {objType.FullName}");
// Use reflection to get all public fields of the object
var fields = objType.GetFields(BindingFlags.Public | BindingFlags.Instance);
foreach (var field in fields)
{
var value = field.GetValue(obj);
// 检查值的类型并转换为十六进制字符串
string hex;
switch (value)
{
case byte b:
hex = b.ToString("X2"); // 2-digit hex for byte
break;
case short s:
hex = s.ToString("X4"); // 4-digit hex for short
break;
case int i:
hex = i.ToString("X8"); // 8-digit hex for int
break;
case long l:
hex = l.ToString("X16"); // 16-digit hex for long
break;
case uint ui:
hex = ui.ToString("X8"); // 8-digit hex for uint
break;
default:
hex = value.ToString(); // Default: fallback to regular ToString
break;
}
Write($"{field.Name}: {value} - 0x{hex}");
}
}
public static void WriteSend(string text, params object[] args)
{
AddLine(ConsoleColor.Green, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteRecv(string text, params object[] args)
{
AddLine(ConsoleColor.Yellow, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteInternal(string text, params object[] args)
{
AddLine(ConsoleColor.Cyan, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteCommand(Client client, string text, params object[] args)
{
if (client == null)
{
AddLine(ConsoleColor.Green, string.Format(text, args));
WriteFile(text, args);
}
else
WriteClient(client, text, args);
}
public static void WriteClient(Client client, string text, params object[] args)
{
var message = string.Format(text, args).Replace('\\', '/');
var packet = new SystemMessagePacket(message, SystemMessagePacket.MessageType.SystemMessage);
client.SendPacket(packet);
}
public static void WriteWarning(string text, params object[] args)
{
AddLine(ConsoleColor.Yellow, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteError(string text, params object[] args)
{
AddLine(ConsoleColor.Red, string.Format(text, args));
WriteFile(text, args);
}
public static void WriteException(string message, Exception ex)
{
var text = string.Empty;
text += string.Format("[ERR] {0} - {1}: {2}", message, ex.GetType(), ex);
if (ex.InnerException != null)
text += string.Format("\n[ERR] Inner Exception: {0}", ex.InnerException);
WriteFile(text);
AddLine(ConsoleColor.Red, text);
}
public static void WriteUnkHex(string text, byte[] array)
{
AddLine(ConsoleColor.Red, text);
// Calculate lines
var lines = 0;
for (var i = 0; i < array.Length; i++)
if ((i % 16) == 0)
lines++;
for (var i = 0; i < lines; i++)
{
var hexString = string.Empty;
// Address
hexString += string.Format("{0:X8} ", i * 16);
// Bytes
for (var j = 0; j < 16; j++)
{
if (j + (i * 16) >= array.Length)
break;
// Append the hex byte with an extra space for every 8 bytes
if (j % 8 == 0 && j > 0)
hexString += ' ';
hexString += string.Format("{0:X2} ", array[j + (i * 16)]);
}
// Spacing
while (hexString.Length < 16 * 4)
hexString += ' ';
// ASCII
for (var j = 0; j < 16; j++)
{
if (j + (i * 16) >= array.Length)
break;
var asciiChar = (char)array[j + (i * 16)];
if (asciiChar == (char)0x00)
asciiChar = '.';
hexString += asciiChar;
}
// Strip off unnecessary stuff
hexString = hexString.Replace('\a', ' '); // Alert beeps
hexString = hexString.Replace('\n', ' '); // Newlines
hexString = hexString.Replace('\r', ' '); // Carriage returns
hexString = hexString.Replace('\\', ' '); // Escape break
AddLine(ConsoleColor.Red, hexString);
WriteFile(hexString);
}
}
public static void WriteHex(string text, byte[] array)
{
AddLine(ConsoleColor.DarkCyan, text);
// Calculate lines
var lines = 0;
for (var i = 0; i < array.Length; i++)
if ((i % 16) == 0)
lines++;
for (var i = 0; i < lines; i++)
{
var hexString = string.Empty;
// Address
hexString += string.Format("{0:X8} ", i * 16);
// Bytes
for (var j = 0; j < 16; j++)
{
if (j + (i * 16) >= array.Length)
break;
// Append the hex byte with an extra space for every 8 bytes
if (j % 8 == 0 && j > 0)
hexString += ' ';
hexString += string.Format("{0:X2} ", array[j + (i * 16)]);
}
// Spacing
while (hexString.Length < 16 * 4)
hexString += ' ';
// ASCII
for (var j = 0; j < 16; j++)
{
if (j + (i * 16) >= array.Length)
break;
var asciiChar = (char)array[j + (i * 16)];
if (asciiChar == (char)0x00)
asciiChar = '.';
hexString += asciiChar;
}
// Strip off unnecessary stuff
hexString = hexString.Replace('\a', ' '); // Alert beeps
hexString = hexString.Replace('\n', ' '); // Newlines
hexString = hexString.Replace('\r', ' '); // Carriage returns
hexString = hexString.Replace('\\', ' '); // Escape break
AddLine(ConsoleColor.White, hexString);
WriteFile(hexString);
}
}
public static void WriteFile(string text, params object[] args)
{
if (args.Length > 0)
Writer.WriteLine(DateTime.Now + " - " + text, args);
else
Writer.WriteLine(DateTime.Now + " - " + text);
// Later we should probably only flush once every PosX amount of lines or on some other condition
Writer.Flush();
}
public static void WriteHeader()
{
Writer.WriteLine();
Writer.WriteLine("--------------------------------------------------");
Writer.WriteLine("\t\t" + DateTime.Now);
Writer.WriteLine("--------------------------------------------------");
}
// 方法:打印结构体的二进制数据
public static void WriteStructBinary(string text, object obj)
{
var byteArray = StructToByteArray(obj);
WriteHex(text, byteArray);
}
// 将结构体转换为字节数组
public static byte[] StructToByteArray(object obj)
{
int size = Marshal.SizeOf(obj);
byte[] byteArray = new byte[size];
IntPtr ptr = Marshal.AllocHGlobal(size);
try
{
Marshal.StructureToPtr(obj, ptr, false);
Marshal.Copy(ptr, byteArray, 0, size);
}
finally
{
Marshal.FreeHGlobal(ptr);
}
return byteArray;
}
// 用于将不同类型的值转换为字节流的方法
public static byte[] ConvertToByteArray(object value)
{
if (value == null)
{
return new byte[0]; // 如果 value 为 null返回空字节数组
}
if (value is byte b)
{
return new byte[] { b }; // byte 转为字节数组
}
else if (value is short s)
{
return BitConverter.GetBytes(s); // short 转为字节数组
}
else if (value is int i)
{
return BitConverter.GetBytes(i); // int 转为字节数组
}
else if (value is long l)
{
return BitConverter.GetBytes(l); // long 转为字节数组
}
else if (value is float f)
{
return BitConverter.GetBytes(f); // float 转为字节数组
}
else if (value is double d)
{
return BitConverter.GetBytes(d); // double 转为字节数组
}
else if (value is string str)
{
return Encoding.UTF8.GetBytes(str); // string 转为字节数组
}
else if (value is bool bval)
{
return BitConverter.GetBytes(bval); // bool 转为字节数组
}
else if (value is decimal dec)
{
return BitConverter.GetBytes(decimal.ToDouble(dec)); // decimal 转为字节数组
}
else
{
return SerializeObject(value); // 对象(如自定义类)转为字节数组
}
}
// 如果是自定义对象,使用二进制序列化转为字节流
public static byte[] SerializeObject(object value)
{
using (var memoryStream = new MemoryStream())
{
var formatter = new BinaryFormatter();
formatter.Serialize(memoryStream, value); // 将对象序列化
return memoryStream.ToArray(); // 获取字节流
}
}
}
}

View File

@ -1,137 +0,0 @@
using NLua;
using System;
namespace PSO2SERVER.LuaScriptEngine
{
public class LuaEngine
{
private Lua lua;
public LuaEngine()
{
// 初始化 LuaEngine 解释器
lua = new Lua();
}
// 执行 LuaEngine 脚本
public void ExecuteScript(string script)
{
try
{
// 执行传入的 LuaEngine 脚本
lua.DoString(script);
}
catch (Exception ex)
{
Console.WriteLine("Error executing LuaEngine script: " + ex.Message);
}
}
// 执行并获取 LuaEngine 脚本结果
public object ExecuteScriptWithResult(string script)
{
try
{
// 执行 LuaEngine 脚本并返回结果
return lua.DoString(script)[0];
}
catch (Exception ex)
{
Console.WriteLine("Error executing LuaEngine script: " + ex.Message);
return null;
}
}
// 将 C# 对象传递给 LuaEngine 环境
public void SetGlobalVariable(string name, object value)
{
lua[name] = value;
}
// 获取 LuaEngine 环境中的全局变量
public object GetGlobalVariable(string name)
{
return lua[name];
}
// 使用 MethodBase 注册方法
public void RegisterFunction(object obj, string luaFunctionName, string methodName)
{
var methodInfo = obj.GetType().GetMethod(methodName);
lua.RegisterFunction(luaFunctionName, obj, methodInfo);
}
// 注册委托函数到 Lua
public void RegisterFunction(string luaFunctionName, Delegate function)
{
// 使用反射将委托作为方法注册
var method = function.Method;
lua.RegisterFunction(luaFunctionName, null, method);
}
// 清理 Lua 环境
public void Close()
{
lua.Close();
}
// 清理 LuaEngine 解释器
public void Dispose()
{
lua.Dispose();
}
// 示例方法
public int AddNumbers(int a, int b)
{
return a + b;
}
// 示例方法
public void print(string text, params object[] args)
{
Logger.Write(text, args);
}
public void printobj(object obj)
{
Logger.WriteObj(obj);
}
// 注册函数
public void RegisterExampleFunction()
{
// 使用反射将方法注册到 Lua
RegisterFunction(this, "AddNumbers", "AddNumbers");
}
public void RegisterLogger()
{
// 使用反射将方法注册到 Lua
RegisterFunction(this, "print", "print");
RegisterFunction(this, "printobj", "printobj");
}
public void LusTest()
{
var engine = new LuaEngine();
engine.RegisterLogger();
// 注册 C# 函数到 Lua
engine.RegisterExampleFunction();
// Lua 脚本
string luaScript = @"
-- C#
result = AddNumbers(10, 20)
print('Result from C#: ' .. result)
printobj(result)
";
// 执行 Lua 脚本
engine.ExecuteScript(luaScript);
// 关闭 Lua 环境
engine.Close();
}
}
}

View File

@ -1,12 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.LuaScriptEngine
{
internal class LuaScript
{
}
}

View File

@ -1,98 +0,0 @@
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
// Class to represent PlayerStats
public class BattlePlayerStats
{
public uint MaxHp { get; set; } = 0;
public uint Hp { get; set; } = 0;
public uint Dex { get; set; } = 0;
public uint BaseMelPwr { get; set; } = 0;
public uint WeaponMelPwr { get; set; } = 0;
public uint BaseRngPwr { get; set; } = 0;
public uint WeaponRngPwr { get; set; } = 0;
public uint BaseTecPwr { get; set; } = 0;
public uint WeaponTecPwr { get; set; } = 0;
public uint BaseMelDef { get; set; } = 0;
public uint BaseRngDef { get; set; } = 0;
public uint BaseTecDef { get; set; } = 0;
// Constructor to initialize with default values
public BattlePlayerStats()
{
}
}
// Class to represent EnemyStats
public class BattleEnemyStats
{
public string Name { get; set; } = string.Empty;
public uint Level { get; set; } = 0;
public uint Exp { get; set; } = 0;
public PSOLocation Pos { get; set; } = new PSOLocation();
public uint MaxHp { get; set; } = 0;
public uint Hp { get; set; } = 0;
public uint Dex { get; set; } = 0;
public uint MaxMelPwr { get; set; } = 0;
public uint MinMelPwr { get; set; } = 0;
public uint MaxRngPwr { get; set; } = 0;
public uint MinRngPwr { get; set; } = 0;
public uint MaxTecPwr { get; set; } = 0;
public uint MinTecPwr { get; set; } = 0;
public uint MelDef { get; set; } = 0;
public uint RngDef { get; set; } = 0;
public uint TecDef { get; set; } = 0;
public List<EnemyHitbox> Hitboxes { get; set; } = new List<EnemyHitbox>();
// Constructor to initialize with default values
public BattleEnemyStats()
{
}
}
// Enum to represent the BattleResult type
public enum BattleResultType
{
Damaged,
Killed
}
// Class to represent the BattleResult
public class BattleResult
{
public BattleResultType ResultType { get; set; }
public DamageReceivePacket DmgPacket { get; set; }
public EnemyKilledPacket KillPacket { get; set; }
public uint ExpAmount { get; set; }
// Constructor for Damaged result
public BattleResult(DamageReceivePacket dmgPacket)
{
ResultType = BattleResultType.Damaged;
DmgPacket = dmgPacket;
}
// Constructor for Killed result
public BattleResult(DamageReceivePacket dmgPacket, EnemyKilledPacket killPacket, uint expAmount)
{
ResultType = BattleResultType.Killed;
DmgPacket = dmgPacket;
KillPacket = killPacket;
ExpAmount = expAmount;
}
}
}

View File

@ -1,30 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public class BlockInfo
{
public uint unk1 { get; set; }
public byte unk2 { get; set; }
public byte unk3 { get; set; }
public byte unk4 { get; set; }
public byte unk5 { get; set; }
public uint unk6 { get; set; }
public uint unk7 { get; set; }
public ushort unk8 { get; set; }
public ushort block_id { get; set; }
public string block_name { get; set; }// 0x20
public Ipv4Addr ip { get; set; } = new Ipv4Addr(new byte[] { 127, 0, 0, 1 });
public ushort port { get; set; }
public ushort unk10 { get; set; }
public ushort unk11 { get; set; }
public ushort[] unk12 { get; set; } = new ushort[3];
public float cur_capacity { get; set; } /// Block fullness (between 0 and 1).
}
}

View File

@ -1,181 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public class CharacterAdditionalStruct
{
}
public enum ShortLanguage : byte
{
Japanese,
English,
Unknown,
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct CharParam
{
public int CharacterID;
public int player_id;
public uint unk1;
public uint voice_type;
public ushort unk2;
public short voice_pitch;
}
public enum ClassType : byte
{
Hunter,
Ranger,
Force,
Fighter,
Gunner,
Techer,
Braver,
Bouncer,
Challenger,
Summoner,
BattleWarrior,
Hero,
Phantom,
Etole,
Luster,
Unknown = 0xFF,
}
// 每个枚举成员的值是通过位移操作计算的:
//Hunter1 << 0即 10x0001
//Ranger1 << 1即 20x0002
//Force1 << 2即 40x0004
//Fighter1 << 3即 80x0008
//Gunner1 << 4即 160x0010
//Techer1 << 5即 320x0020
//Braver1 << 6即 640x0040
//Bouncer1 << 7即 1280x0080
//Challenger1 << 8即 2560x0100
//Summoner1 << 9即 5120x0200
//BattleWarrior1 << 10即 10240x0400
//Hero1 << 11即 20480x0800
//Phantom1 << 12即 40960x1000
//Etole1 << 13即 81920x2000
//Luster1 << 14即 163840x4000
[Flags]
public enum ClassFlags : ushort
{
Hunter = 1 << 0,
Ranger = 1 << 1,
Force = 1 << 2,
Fighter = 1 << 3,
Gunner = 1 << 4,
Techer = 1 << 5,
Braver = 1 << 6,
Bouncer = 1 << 7,
Challenger = 1 << 8,
Summoner = 1 << 9,
BattleWarrior = 1 << 10,
Hero = 1 << 11,
Phantom = 1 << 12,
Etole = 1 << 13,
Luster = 1 << 14
}
[StructLayout(LayoutKind.Sequential)]
public struct JobEntry
{
/// Main level.
public ushort level;
public ushort level2; // SubClass level
/// Current EXP.
public uint exp;
}
[StructLayout(LayoutKind.Sequential)]
public struct Entries
{
public JobEntry
hunter
, ranger
, force
, fighter
, gunner
, techer
, braver
, bouncer
, Challenger
, Summoner
, BattleWarrior
, Hero
, Phantom
, Etole
, Luster
, unk16
, unk17
, unk18
, unk19
, unk20
, unk21
, unk22
, unk23
, unk24
;
}
public enum RunAnimation : ushort
{
Walking = 9,
Hovering = 11
}
public enum Race : ushort
{
Unknown = 0xFFFF,
Human = 0,
Newman,
Cast,
Dewman,
}
public enum Gender : ushort
{
Unknown = 0xFFFF,
Male = 0,
Female,
}
[StructLayout(LayoutKind.Sequential)]
public struct Figure
{
public ushort x, y, z; // Great naming, SEGA
}
public struct AccessoryData
{
public sbyte Value1;
public sbyte Value2;
public sbyte Value3;
}
public enum SkinColor : byte
{
RaceDefined,
Human,
Deuman,
Cast
}
[StructLayout(LayoutKind.Sequential)]
public struct HSVColor
{
public ushort hue, saturation, value;
}
}

View File

@ -1,101 +0,0 @@
using System;
using System.Runtime.InteropServices;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using PSO2SERVER.Protocol;
using PSO2SERVER.Database;
namespace PSO2SERVER.Models
{
public class CharacterStruct
{
[StructLayout(LayoutKind.Sequential)]
public unsafe struct LooksParam
{
public RunAnimation running_animation;
public Race race;
public Gender gender;
public ushort Muscule;
public Figure Body;
public Figure Arms;
public Figure Legs;
public Figure Chest;
public Figure FaceShape;
public Figure FaceParts;
public Figure Eyes;
public Figure NoseSize;
public Figure NoseHeight;
public Figure Mouth;
public Figure Ears;
public Figure Neck;
public Figure Waist;
public Figure Body2;
public Figure Arms2;
public Figure Legs2;
public Figure Chest2;
public Figure Neck2;
public Figure Waist2;
public fixed byte Unk1[0x20];
public fixed byte Unk2[0x0A];
public AccessoryData Acc1Location;
public AccessoryData Acc2Location;
public AccessoryData Acc3Location;
public AccessoryData Acc4Location;
public HSVColor UnkColor;
public HSVColor CostumeColor;
public HSVColor MainColor;
public HSVColor Sub1Color;
public HSVColor Sub2Color;
public HSVColor Sub3Color;
public HSVColor EyeColor;
public HSVColor HairColor;
public fixed byte Unk3[0x20];
public fixed byte Unk4[0x10];
public ushort CostumeId;
public ushort BodyPaint1;
public ushort StickerId;
public ushort RightEyeId;
public ushort EyebrowId;
public ushort EyelashId;
public ushort FaceId1;
public ushort FaceId2;
public ushort Facemakeup1Id;
public ushort HairstyleId;
public ushort Acc1Id;
public ushort Acc2Id;
public ushort Acc3Id;
public ushort Facemakeup2Id;
public ushort LegId;
public ushort ArmId;
public ushort Acc4Id;
public fixed byte Unk5[0x04];
public ushort BodyPaint2;
public ushort LeftEyeId;
public fixed byte Unk6[0x12];
public AccessoryData Acc1Size;
public AccessoryData Acc2Size;
public AccessoryData Acc3Size;
public AccessoryData Acc4Size;
public AccessoryData Acc1Rotation;
public AccessoryData Acc2Rotation;
public AccessoryData Acc3Rotation;
public AccessoryData Acc4Rotation;
public ushort Unk7;
public fixed byte Unk8[0x08];
public SkinColor SkinColorType;
public sbyte EyebrowThickness;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct JobParam
{
public ClassType mainClass;//1
public ClassType subClass;//1
public ushort unk2;//2
public ClassFlags enabledClasses;//2
public ushort unk3;//2
public Entries entries; //TODO: Make this a fixed array 24 * 8
public fixed ushort unk_maxlevel[15];//30
}
}
}

View File

@ -1,77 +0,0 @@
using System;
using System.Net.Sockets;
using System.Runtime.InteropServices;
namespace PSO2SERVER.Models
{
public struct PacketHeader
{
public UInt32 Size;
public byte Type;
public byte Subtype;
public byte Flags1;
public byte Flags2;
public PacketHeader(int size, byte type, byte subtype, byte flags1, byte flags2)
{
this.Size = (uint)size;
this.Type = type;
this.Subtype = subtype;
this.Flags1 = flags1;
this.Flags2 = flags2;
}
public PacketHeader(int size, byte type, byte subtype, PacketFlags flags1, PacketFlags flags2) : this(size, type, subtype, (byte)flags1, (byte)flags1)
{
}
public PacketHeader(byte type, byte subtype) : this(type, subtype, (byte)0)
{
}
public PacketHeader(byte type, byte subtype, byte flags1) : this(0, type, subtype, flags1, 0)
{
}
public PacketHeader(byte type, byte subtype, PacketFlags packetFlags) : this(type, subtype, (byte)packetFlags)
{
}
// ToBytes 方法的实现
public byte[] ToBytes()
{
byte[] bytes = new byte[sizeof(UInt32) + sizeof(byte) * 4]; // 计算结构体的总大小
int index = 0;
// 将结构体每个字段转换为字节
Array.Copy(BitConverter.GetBytes(Size), 0, bytes, index, sizeof(UInt32));
index += sizeof(UInt32);
bytes[index++] = Type;
bytes[index++] = Subtype;
bytes[index++] = Flags1;
bytes[index++] = Flags2;
return bytes;
}
}
/// Packet flags.
[Flags]
public enum PacketFlags : byte
{
/// 0x00
None = 0x00,
/// Set when the packet contains variable length data. 0x04
PACKED = 1 << 2,
/// 0x10
FLAG_10 = 1 << 4,
/// Set when the [`Packet::Movement`] has all fields set. 0x20
FULL_MOVEMENT = 1 << 5,
/// Set for all (?) of (0x04) packets. 0x40
OBJECT_RELATED = 1 << 6,
/// 0x44
PACKED_OBJECT_RELATED = PACKED | OBJECT_RELATED
}
}

View File

@ -1,997 +0,0 @@
using Newtonsoft.Json.Linq;
using Newtonsoft.Json;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Collections;
using System.Xml.Linq;
using System.IO;
using PSO2SERVER.Protocol;
namespace PSO2SERVER.Models
{
public class FixedList<T> : IEnumerable<T>
{
private List<T> _data;
private int _capacity;
public FixedList(int size)
{
if (size <= 0)
throw new ArgumentException("Size must be greater than zero.", nameof(size));
_data = new List<T>(size);
_capacity = size;
}
public FixedList(int size, T defaultValue)
{
if (size <= 0)
throw new ArgumentException("Size must be greater than zero.", nameof(size));
_data = new List<T>(size);
_capacity = size;
for (int i = 0; i < size; i++)
{
_data.Add(defaultValue);
}
}
public int Capacity => _capacity;
public int Length => _data.Count;
public void Add(T item)
{
if (_data.Count >= _capacity)
throw new InvalidOperationException("FixedList is full.");
_data.Add(item);
}
public void Insert(int index, T item)
{
if (index < 0 || index > _data.Count)
throw new ArgumentOutOfRangeException(nameof(index));
if (_data.Count >= _capacity)
throw new InvalidOperationException("FixedList is full.");
_data.Insert(index, item);
}
public void RemoveAt(int index)
{
if (index < 0 || index >= _data.Count)
throw new ArgumentOutOfRangeException(nameof(index));
_data.RemoveAt(index);
}
public void RemoveLast()
{
if (_data.Count == 0)
throw new InvalidOperationException("FixedList is empty.");
_data.RemoveAt(_data.Count - 1);
}
public void SetItem(int index, T value)
{
// 如果 index 超出当前长度,但没有超出最大容量,进行扩展
if (index >= _data.Count)
{
// 如果 index 没有超过最大容量
if (index < _capacity)
{
// 扩展 _data 列表,填充默认值
while (_data.Count <= index && _data.Count < _capacity)
{
_data.Add(default(T));
}
}
else
{
throw new ArgumentOutOfRangeException(nameof(index), "索引超出最大容量.");
}
}
// 赋值
_data[index] = value;
}
public void SetListItem(List<T> value)
{
// 截取最大容量的值,如果传入的值超过了 _capacity
if (value.Count > _capacity)
{
value = value.Take(_capacity).ToList();
}
// 更新已有的元素
for (var i = 0; i < _data.Count && i < value.Count; i++)
{
_data[i] = value[i];
}
// 如果 value 的长度超过当前 _data 长度并且未超过最大容量,则扩展 _data 长度
if (value.Count > _data.Count)
{
// 计算要增加的元素数量
int elementsToAdd = value.Count - _data.Count;
// 添加元素,确保不超过最大容量
for (var i = 0; i < elementsToAdd && _data.Count < _capacity; i++)
{
_data.Add(default(T)); // 填充默认值
}
// 将传入的 value 中剩余的元素(超出 _data 长度部分)复制到 _data
for (var i = _data.Count; i < value.Count; i++)
{
_data[i] = value[i];
}
}
}
public void SetAll(T newValue)
{
for (int i = 0; i < _data.Count; i++)
{
_data[i] = newValue;
}
}
public void Clear()
{
_data.Clear();
}
public T this[int index]
{
get
{
if (index < 0 || index >= _data.Count)
throw new IndexOutOfRangeException();
return _data[index];
}
set
{
if (index < 0 || index >= _data.Count)
throw new IndexOutOfRangeException();
_data[index] = value;
}
}
public override string ToString()
{
return $"[{string.Join(", ", _data)}]";
}
// 实现 IEnumerable<T> 接口
public IEnumerator<T> GetEnumerator()
{
return _data.GetEnumerator();
}
// 非泛型版本的 GetEnumerator
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
public class FixedSortedList<T> : IEnumerable<T> where T : IComparable<T>
{
private List<T> _list;
private int _capacity;
public FixedSortedList(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_list = new List<T>();
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count => _list.Count;
// 添加元素
public void Add(T item)
{
if (_list.Count >= _capacity)
throw new InvalidOperationException("List is full.");
_list.Add(item);
_list.Sort(); // 自动排序
}
// 获取元素
public T Get(int index)
{
if (index < 0 || index >= _list.Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _list[index];
}
// 清空列表
public void Clear()
{
_list.Clear();
}
public override string ToString()
{
return $"[{string.Join(", ", _list)}]";
}
// 实现 IEnumerable<T> 接口
public IEnumerator<T> GetEnumerator()
{
return _list.GetEnumerator();
}
// 非泛型版本的 GetEnumerator
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
public class FixedHashSet<T> : IEnumerable<T>
{
private HashSet<T> _set;
private int _capacity;
public FixedHashSet(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_set = new HashSet<T>();
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count => _set.Count;
// 添加元素
public bool Add(T item)
{
if (_set.Count >= _capacity)
throw new InvalidOperationException("HashSet is full.");
return _set.Add(item);
}
// 判断元素是否存在
public bool Contains(T item)
{
return _set.Contains(item);
}
// 清空集合
public void Clear()
{
_set.Clear();
}
public override string ToString()
{
return $"[{string.Join(", ", _set)}]";
}
// 实现 IEnumerable<T> 接口
public IEnumerator<T> GetEnumerator()
{
return _set.GetEnumerator();
}
// 非泛型版本的 GetEnumerator
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
public class FixedString
{
private string[] _array;
private int _capacity;
public FixedString(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new string[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的字符串
public string this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加字符串
public void Add(string item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var str in _array.Take(Count))
{
byte[] stringBytes = System.Text.Encoding.UTF8.GetBytes(str);
byteList.AddRange(stringBytes);
}
return byteList.ToArray();
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
public class FixedInt
{
private int[] _array;
private int _capacity;
public FixedInt(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new int[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的整数
public int this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加整数
public void Add(int item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] intBytes = BitConverter.GetBytes(num);
byteList.AddRange(intBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
public class FixedUlong
{
private ulong[] _array;
private int _capacity;
public FixedUlong(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new ulong[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 ulong 数值
public ulong this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 ulong 数值
public void Add(ulong item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] ulongBytes = BitConverter.GetBytes(num);
byteList.AddRange(ulongBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
//public class FixedByte
//{
// private byte[] _array;
// private int _capacity;
// public FixedByte(int capacity)
// {
// if (capacity <= 0)
// throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
// _array = new byte[capacity];
// _capacity = capacity;
// Count = 0; // 初始化 Count 为 0
// }
// public int Capacity => _capacity;
// // Count 现在是只读属性
// public int Count { get; private set; }
// // 获取或设置指定索引的字节
// public byte this[int index]
// {
// get
// {
// if (index < 0 || index >= Count)
// throw new ArgumentOutOfRangeException(nameof(index));
// return _array[index];
// }
// set
// {
// if (index < 0 || index >= Count)
// throw new ArgumentOutOfRangeException(nameof(index));
// _array[index] = value;
// }
// }
// // 添加字节
// public void Add(byte item)
// {
// if (Count >= _capacity)
// throw new InvalidOperationException("Array is full.");
// _array[Count++] = item;
// }
// // 清空数组
// public void Clear()
// {
// Array.Clear(_array, 0, _capacity);
// Count = 0;
// }
// // 将字节数组转换为字节数组(仅返回已使用的部分)
// public byte[] ToBytes()
// {
// return _array.Take(Count).ToArray();
// }
// public override string ToString()
// {
// return $"[{string.Join(", ", _array.Take(Count))}]";
// }
// // JSON 序列化和反序列化的自定义处理
// public class FixedByteConverter : JsonConverter<FixedByte>
// {
// public override FixedByte ReadJson(JsonReader reader, Type objectType, FixedByte existingValue, bool hasExistingValue, JsonSerializer serializer)
// {
// // 确保是 JSON 数组格式
// if (reader.TokenType == JsonToken.StartArray)
// {
// // 反序列化为字节数组
// var byteArray = serializer.Deserialize<byte[]>(reader);
// var fixedByte = new FixedByte(byteArray.Length);
// // 使用 Add 方法填充字节
// foreach (var byteValue in byteArray)
// {
// fixedByte.Add(byteValue);
// }
// return fixedByte;
// }
// else
// {
// throw new JsonSerializationException("Expected a JSON array.");
// }
// }
// public override void WriteJson(JsonWriter writer, FixedByte value, JsonSerializer serializer)
// {
// // 序列化时直接写入字节数组
// serializer.Serialize(writer, value.ToBytes());
// }
// }
//}
public class FixedDouble
{
private double[] _array;
private int _capacity;
public FixedDouble(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new double[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 double 类型数值
public double this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 double 类型数值
public void Add(double item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] doubleBytes = BitConverter.GetBytes(num);
byteList.AddRange(doubleBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
public class FixedFloat
{
private float[] _array;
private int _capacity;
public FixedFloat(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new float[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 float 类型数值
public float this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 float 类型数值
public void Add(float item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] floatBytes = BitConverter.GetBytes(num);
byteList.AddRange(floatBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
public class FixedLong
{
private long[] _array;
private int _capacity;
public FixedLong(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new long[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 long 类型数值
public long this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 long 类型数值
public void Add(long item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] longBytes = BitConverter.GetBytes(num);
byteList.AddRange(longBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
public class FixedUShort
{
private ushort[] _array;
private int _capacity;
public FixedUShort(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new ushort[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 ushort 类型数值
public ushort this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 ushort 类型数值
public void Add(ushort item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] ushortBytes = BitConverter.GetBytes(num);
byteList.AddRange(ushortBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
public class FixedUShortConverter : JsonConverter<FixedUShort>
{
public override FixedUShort ReadJson(JsonReader reader, Type objectType, FixedUShort existingValue, bool hasExistingValue, JsonSerializer serializer)
{
// 确保我们读取的是一个 JSON 数组
if (reader.TokenType != JsonToken.StartArray)
throw new JsonSerializationException("Expected StartArray token.");
JArray array = JArray.Load(reader);
var fixedUShort = new FixedUShort(array.Count); // 创建一个合适容量的 FixedUShort 对象
foreach (var item in array)
{
// 将每个元素添加到 FixedUShort 对象中
if (item.Type == JTokenType.Integer)
{
fixedUShort.Add((ushort)item);
}
else
{
throw new JsonSerializationException("Expected ushort values in the array.");
}
}
return fixedUShort;
}
public override void WriteJson(JsonWriter writer, FixedUShort value, JsonSerializer serializer)
{
writer.WriteStartArray();
for (int i = 0; i < value.Count; i++)
{
writer.WriteValue(value[i]);
}
writer.WriteEndArray();
}
}
}
public class FixedShort
{
private short[] _array;
private int _capacity;
public FixedShort(int capacity)
{
if (capacity <= 0)
throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
_array = new short[capacity];
_capacity = capacity;
}
public int Capacity => _capacity;
public int Count { get; private set; } = 0;
// 获取或设置指定索引的 ushort 类型数值
public short this[int index]
{
get
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
return _array[index];
}
set
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException(nameof(index));
_array[index] = value;
}
}
// 添加 ushort 类型数值
public void Add(short item)
{
if (Count >= _capacity)
throw new InvalidOperationException("Array is full.");
_array[Count++] = item;
}
// 清空数组
public void Clear()
{
Array.Clear(_array, 0, _capacity);
Count = 0;
}
// 将数组转换为字节数组
public byte[] ToBytes()
{
List<byte> byteList = new List<byte>();
foreach (var num in _array.Take(Count))
{
byte[] ushortBytes = BitConverter.GetBytes(num);
byteList.AddRange(ushortBytes);
}
return byteList.ToArray();
}
public override string ToString()
{
return $"[{string.Join(", ", _array.Take(Count))}]";
}
}
}

View File

@ -1,84 +0,0 @@
using System.Collections.Generic;
using System.Security.AccessControl;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Models
{
public class Flags
{
public enum FlagType : uint
{
/// Flag is account related.
Account,
/// Flag is character related.
Character,
}
private List<byte> flags = new List<byte>();
private List<uint> paramsList = new List<uint>();
public Flags() { flags = new List<byte>(); paramsList = new List<uint>(); }
public void Set(int id, byte val)
{
int index = id / 8;
byte bitIndex = (byte)(id % 8);
while (flags.Count <= index)
flags.Add(0);
flags[index] = SetBit(flags[index], bitIndex, val);
}
public byte Get(int id)
{
int index = id / 8;
byte bitIndex = (byte)(id % 8);
if (index >= flags.Count)
return 0;
return (byte)((flags[index] & (1 << bitIndex)) >> bitIndex);
}
public void SetParam(int id, uint val)
{
while (paramsList.Count <= id)
paramsList.Add(0);
paramsList[id] = val;
}
public uint GetParam(int id)
{
if (id >= paramsList.Count)
return 0;
return paramsList[id];
}
public AccountFlagsPacket ToAccountFlags()
{
var aflags = new AccountFlagsPacket();
aflags.Flags.SetListItem(flags);
aflags.Params.SetListItem(paramsList);
return aflags;
}
public CharacterFlagsPacket ToCharFlags()
{
var cflags = new CharacterFlagsPacket();
cflags.Flags.SetListItem(flags);
cflags.Params.SetListItem(paramsList);
return cflags;
}
private static byte SetBit(byte byteValue, byte index, byte val)
{
byteValue &= (byte)~(1 << index);
return (byte)(byteValue | ((val & 1) << index));
}
}
}

View File

@ -1,47 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public struct ItemName
{
public ItemId id;
public string en_name;
public string jp_name;
public string en_desc;
public string jp_desc;
}
public struct ItemParameters
{
public List<byte> pc_attrs;
public List<byte> vita_attrs;
public ItemAttributesPC attrs;
public List<ItemName> names;
}
public struct StorageInventory
{
public uint total_space;
public byte storage_id;
public bool is_enabled;
public bool is_purchased;
public byte storage_type;
public List<PSO2Items> items;
}
public struct AccountStorages
{
public ulong Storage_meseta;
public StorageInventory Default;
public StorageInventory Premium;
public StorageInventory Extend1;
}
internal class Inventory
{
}
}

View File

@ -1,905 +0,0 @@
using Newtonsoft.Json;
using PSO2SERVER.Json;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Security.AccessControl;
using System.Text;
using System.Threading.Tasks;
using UltimateOrb;
namespace PSO2SERVER.Models
{
// 表示 JSON 的根对象
public class ItemAttributesRootObject
{
[JsonProperty("PC")]
public ItemAttributesPC PC { get; set; }
[JsonProperty("VITA")]
public ItemAttributesVita VITA { get; set; }
}
// Base class for ItemAttributes
[Serializable]
public abstract class ItemAttributesBase
{
[JsonProperty("unk1")]
public uint Unk1 { get; set; }
[JsonProperty("unk2")]
public UInt128 Unk2 { get; set; }
}
// PC ItemAttributes (NA and JP client)
[Serializable]
public class ItemAttributesPC : ItemAttributesBase
{
[JsonProperty("weapons")]
public List<WeaponAttrs> Weapons { get; set; } = new List<WeaponAttrs>();
[JsonProperty("human_costumes")]
public List<HumanCostume> HumanCostumes { get; set; } = new List<HumanCostume>();
[JsonProperty("cast_parts")]
public List<CastPart> CastParts { get; set; } = new List<CastPart>();
[JsonProperty("consumables")]
public List<Consumable> Consumables { get; set; } = new List<Consumable>();
[JsonProperty("data5")]
public List<Data5> Data5 { get; set; } = new List<Data5>();
[JsonProperty("data6")]
public List<Unit> Data6 { get; set; } = new List<Unit>();
[JsonProperty("data7")]
public List<Data7> Data7 { get; set; } = new List<Data7>();
[JsonProperty("data8")]
public List<Data8> Data8 { get; set; } = new List<Data8>();
[JsonProperty("data9")]
public List<Data9> Data9 { get; set; } = new List<Data9>();
[JsonProperty("data10")]
public List<Data10> Data10 { get; set; } = new List<Data10>();
[JsonProperty("data11")]
public List<Data11> Data11 { get; set; } = new List<Data11>();
[JsonProperty("data12")]
public List<Data12> Data12 { get; set; } = new List<Data12>();
[JsonProperty("data13")]
public List<Data13> Data13 { get; set; } = new List<Data13>();
[JsonProperty("data14")]
public List<Data14> Data14 { get; set; } = new List<Data14>();
[JsonProperty("data15")]
public List<Data15> Data15 { get; set; } = new List<Data15>();
[JsonProperty("data16")]
public List<Data16> Data16 { get; set; } = new List<Data16>();
[JsonProperty("data17")]
public List<Data17> Data17 { get; set; } = new List<Data17>();
[JsonProperty("data18")]
[JsonConverter(typeof(FixedListConverter<ShortData>))]
public FixedList<ShortData> Data18 { get; set; } = new FixedList<ShortData>(406);
[JsonProperty("data19")]
public List<Data19> Data19 { get; set; } = new List<Data19>();
[JsonProperty("data20")]
public List<Data20> Data20 { get; set; } = new List<Data20>();
public static void LogWeapons(List<WeaponAttrs> weapons)
{
if (weapons == null || weapons.Count == 0) return;
Logger.Write($"Weapons Count: {weapons.Count}");
foreach (var weapon in weapons)
{
Logger.Write($"Weapon ID: {weapon.Id}, Range Damage: {weapon.RangeDmg}, Gender: {weapon.Gender}");
Logger.Write($"weapon.Unk8 Length: {weapon.Unk8.Length}");
foreach (var un8 in weapon.Unk8)
{
Logger.Write($"Weapon un8: {un8}");
}
}
}
// 输出 Data17 列表信息
public static void LogData17(List<Data17> data17List)
{
if (data17List == null || data17List.Count == 0) return;
Logger.Write($"Data17 Count: {data17List.Count}");
foreach (var data17 in data17List)
{
Logger.Write($"data17.unk Length: {data17.unk.Length}");
foreach (var value in data17.unk)
{
Logger.Write($"data17 value: {value}");
}
}
}
}
// Vita ItemAttributes (Vita client)
[Serializable]
public class ItemAttributesVita : ItemAttributesBase
{
[JsonProperty("weapons")]
public List<WeaponAttrs> Weapons { get; set; } = new List<WeaponAttrs>();
[JsonProperty("human_costumes")]
public List<HumanCostume> HumanCostumes { get; set; } = new List<HumanCostume>();
[JsonProperty("cast_parts")]
public List<CastPart> CastParts { get; set; } = new List<CastPart>();
[JsonProperty("consumables")]
public List<Consumable> Consumables { get; set; } = new List<Consumable>();
[JsonProperty("data5")]
public List<Data5> Data5 { get; set; } = new List<Data5>();
[JsonProperty("data6")]
public List<Unit> Data6 { get; set; } = new List<Unit>();
[JsonProperty("data7")]
public List<Data7> Data7 { get; set; } = new List<Data7>();
[JsonProperty("data8")]
public List<Data8> Data8 { get; set; } = new List<Data8>();
[JsonProperty("data9")]
public List<Data9> Data9 { get; set; } = new List<Data9>();
[JsonProperty("data10")]
public List<Data10> Data10 { get; set; } = new List<Data10>();
[JsonProperty("data11")]
public List<Data11> Data11 { get; set; } = new List<Data11>();
[JsonProperty("data12")]
public List<Data12> Data12 { get; set; } = new List<Data12>();
[JsonProperty("data13")]
public List<Data13> Data13 { get; set; } = new List<Data13>();
[JsonProperty("data14")]
public List<Data14> Data14 { get; set; } = new List<Data14>();
[JsonProperty("data15")]
public List<Data15> Data15 { get; set; } = new List<Data15>();
[JsonProperty("data16")]
public List<Data16> Data16 { get; set; } = new List<Data16>();
[JsonProperty("data17")]
public List<Data17> Data17 { get; set; } = new List<Data17>();
[JsonProperty("data18")]
[JsonConverter(typeof(FixedListConverter<ShortData>))]
public FixedList<ShortData> Data18 { get; set; } = new FixedList<ShortData>(406);
[JsonProperty("data19")]
public List<Data19Vita> Data19 { get; set; } = new List<Data19Vita>();
[JsonProperty("data20")]
public List<Data20> Data20 { get; set; } = new List<Data20>();
}
// Abstract base class that holds the common functionality of the Item, irrespective of the platform
[Serializable]
public class ItemAttr
{
public ItemAttributesBase Attributes { get; set; }
// Constructor to initialize an item with platform-specific attributes
public ItemAttr(ItemAttributesBase attributes)
{
Attributes = attributes;
}
// Method to display item attributes
public void DisplayAttributes()
{
switch (Attributes)
{
case ItemAttributesPC pcAttributes:
Console.WriteLine("PC Version Item Attributes:");
Console.WriteLine($"Weapons Count: {pcAttributes.Weapons.Count}");
// Add further attribute display logic for PC
break;
case ItemAttributesVita vitaAttributes:
Console.WriteLine("Vita Version Item Attributes:");
Console.WriteLine($"Weapons Count: {vitaAttributes.Weapons.Count}");
// Add further attribute display logic for Vita
break;
default:
Console.WriteLine("Unknown ItemAttributes");
break;
}
}
}
// Example of a simple class, could be for one of the referenced types
[Serializable]
public class WeaponAttrs
{
// Item category
[JsonProperty("id")]
public ushort Id { get; set; }
// Item ID
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public byte Unk1 { get; set; }
[JsonProperty("priority")]
public byte Priority { get; set; }
[JsonProperty("unk2")]
public byte Unk2 { get; set; }
[JsonProperty("priority2")]
public byte Priority2 { get; set; }
// Item rarity in stars
[JsonProperty("rarity")]
public byte Rarity { get; set; }
[JsonProperty("flags")]
public ushort Flags { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
// Range damage
[JsonProperty("range_dmg")]
public ushort RangeDmg { get; set; }
[JsonProperty("unk4")]
public byte Unk4 { get; set; }
// Melee damage
[JsonProperty("melee_dmg")]
public ushort MeleeDmg { get; set; }
[JsonProperty("unk5")]
public byte Unk5 { get; set; }
[JsonProperty("unk6")]
public uint Unk6 { get; set; }
// Force damage and equipable genders
[JsonProperty("force_dmg")]
public uint ForceDmg { get; set; }
[JsonProperty("gender")]
public string Gender { get; set; }
[JsonProperty("unk8")]
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> Unk8 { get; set; } = new FixedList<byte>(0x04, 0);
// Equipable races
[JsonProperty("race")]
public string Race { get; set; }
[JsonProperty("flags2")]
public byte Flags2 { get; set; }
// Equipable classes
[JsonProperty("class")]
public string Class { get; set; }
// Required stat value
[JsonProperty("req_stat")]
public ushort ReqStat { get; set; }
// Required stat type
[JsonProperty("req_stat_type")]
public string ReqStatType { get; set; }
[JsonProperty("unk9")]
public byte Unk9 { get; set; }
[JsonProperty("model")]
public ushort Model { get; set; }
[JsonProperty("unk10")]
public uint Unk10 { get; set; }
[JsonProperty("unk11")]
public ushort Unk11 { get; set; }
[JsonProperty("affix_flag")]
public ushort AffixFlag { get; set; }
[JsonProperty("unk12")]
public ushort Unk12 { get; set; }
}
[Serializable]
public class HumanCostume
{
/// <summary>
/// Item category.
/// </summary>
[JsonProperty("id")]
public ushort Id { get; set; }
/// <summary>
/// Item ID.
/// </summary>
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public ushort Unk1 { get; set; }
[JsonProperty("unk2")]
public ushort Unk2 { get; set; }
/// <summary>
/// Item rarity in stars.
/// </summary>
[JsonProperty("rarity")]
public byte Rarity { get; set; }
[JsonProperty("flags")]
public ushort Flags { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
/// <summary>
/// Equipable genders.
/// </summary>
[JsonProperty("gender_flags")]
[JsonConverter(typeof(GenderFlagsConverter))]
public GenderFlags GenderFlags { get; set; }
[JsonProperty("color_flags")]
public byte ColorFlags { get; set; }
/// <summary>
/// Equipable races.
/// </summary>
[JsonProperty("race_flags")]
[JsonConverter(typeof(RaceFlagsConverter))]
public RaceFlags RaceFlags { get; set; }
[JsonProperty("unk4")]
public byte Unk4 { get; set; }
[JsonProperty("model")]
public ushort Model { get; set; }
[JsonProperty("unk5")]
[JsonConverter(typeof(FixedUShort.FixedUShortConverter))]
public FixedUShort Unk5 { get; set; } = new FixedUShort(3);
}
[Serializable]
public class CastPart
{
/// <summary>
/// Item category.
/// </summary>
[JsonProperty("id")]
public ushort Id { get; set; }
/// <summary>
/// Item ID.
/// </summary>
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public ushort Unk1 { get; set; }
[JsonProperty("unk2")]
public ushort Unk2 { get; set; }
/// <summary>
/// Item rarity in stars.
/// </summary>
/// [JsonProperty("")]
[JsonProperty("rarity")]
public byte Rarity { get; set; }
[JsonProperty("flags")]
public ushort Flags { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
[JsonProperty("gender_flags")]
public byte GenderFlags { get; set; }
[JsonProperty("unk4")]
public byte Unk4 { get; set; }
[JsonProperty("race_flags")]
public byte RaceFlags { get; set; }
[JsonProperty("unk5")]
public byte Unk5 { get; set; }
[JsonProperty("unk6")]
public ushort Unk6 { get; set; }
[JsonProperty("model")]
public ushort Model { get; set; }
// 默认构造函数
public CastPart()
{
// 如果需要初始化某些属性,可以在这里进行
}
}
[Serializable]
public class Consumable
{
[JsonProperty("id")]
public ushort Id { get; set; }
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public ushort Unk1 { get; set; }
[JsonProperty("unk2")]
public ushort Unk2 { get; set; }
[JsonProperty("rarity")]
public byte Rarity { get; set; }
[JsonProperty("flags")]
public ushort Flags { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
[JsonProperty("max_qty")]
public byte MaxQty { get; set; }
[JsonProperty("unk4")]
public byte[] Unk4 { get; set; } = new byte[3];
[JsonProperty("unk5")]
public uint Unk5 { get; set; }
}
[Serializable]
public class Data5
{
[JsonProperty("id")]
public ushort Id { get; set; }
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public ushort Unk1 { get; set; }
[JsonProperty("unk2")]
public ushort Unk2 { get; set; }
[JsonProperty("unk6")]
public byte Unk6 { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("unk4")]
public byte Unk4 { get; set; }
[JsonProperty("unk5")]
public byte Unk5 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
[JsonProperty("unk")]
public byte[] Unk { get; set; } = new byte[0x30]; // Fixed length of 0x30 bytes
}
[Serializable]
public class Unit
{
/// <summary>
/// Item category.
/// </summary>
[JsonProperty("id")]
public ushort Id { get; set; }
/// <summary>
/// Item ID.
/// </summary>
[JsonProperty("subid")]
public ushort Subid { get; set; }
[JsonProperty("unk1")]
public ushort Unk1 { get; set; }
[JsonProperty("unk2")]
public ushort Unk2 { get; set; }
/// <summary>
/// Item rarity in stars.
/// </summary>
[JsonProperty("rarity")]
public byte Rarity { get; set; }
[JsonProperty("flags")]
public ushort Flags { get; set; }
[JsonProperty("unk3")]
public byte Unk3 { get; set; }
[JsonProperty("icon_list")]
public ushort IconList { get; set; }
[JsonProperty("icon_index")]
public ushort IconIndex { get; set; }
/// <summary>
/// Unit stats.
/// </summary>
[JsonProperty("stats")]
public UnitRes Stats { get; set; }
/// <summary>
/// Required stat type.
/// </summary>
[JsonProperty("req_stat_type")]
public StatType ReqStatType { get; set; }
[JsonProperty("unk4")]
public byte Unk4 { get; set; }
[JsonProperty("unk5")]
public byte Unk5 { get; set; }
[JsonProperty("unk6")]
public ushort Unk6 { get; set; }
[JsonProperty("unk7")]
public ushort Unk7 { get; set; }
[JsonProperty("unk8")]
public ushort Unk8 { get; set; }
[JsonProperty("unk9")]
public ushort Unk9 { get; set; }
/// <summary>
/// Required stat value.
/// </summary>
[JsonProperty("req_stat")]
public ushort ReqStat { get; set; }
/// <summary>
/// Unit attack values.
/// </summary>
[JsonProperty("atk")]
public UnitAtk Atk { get; set; }
[JsonProperty("unk10")]
public byte Unk10 { get; set; }
[JsonProperty("unk11")]
public byte Unk11 { get; set; }
[JsonProperty("unk12")]
public byte Unk12 { get; set; }
[JsonProperty("unk13")]
public ushort Unk13 { get; set; }
[JsonProperty("unk14")]
public uint Unk14 { get; set; }
[JsonProperty("unk15")]
public uint Unk15 { get; set; }
}
public class Data7
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x38, 0);
}
public class Data8
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x10, 0);
}
public class Data9
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x3C, 0);
}
public class Data10
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x24, 0);
}
public class Data11
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x10, 0);
}
public class Data12
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x24, 0);
}
public class Data13
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x50, 0);
}
public class Data14
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x7C, 0);
}
public class Data15
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x10, 0);
}
public class Data16
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x12, 0);
}
public class Data17
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x5A, 0);
}
public class Data18
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x08, 0);
}
public class ShortData
{
public List<Data18> unk { get; set; } = new List<Data18>();
}
public class Data19
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x54, 0);
}
public class Data19Vita
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x2C, 0);
}
public class Data20
{
[JsonConverter(typeof(FixedListConverter<byte>))]
public FixedList<byte> unk { get; set; } = new FixedList<byte>(0x1C, 0);
}
public struct GenderDmg
{
/// Force damage.
public ushort force_dmg;
/// Equipable genders.
public GenderFlags gender;
}
public enum StatType
{
// MEL power.
MELPwr = 0,
// RNG power.
RNGPwr = 1,
// TEC power.
TECPwr = 2,
// DEX.
DEX = 3,
// MEL defence.
MELDef = 4,
// RNG defence.
RNGDef = 5,
// TEC defence.
TECDef = 6
}
public static class StatTypeExtensions
{
public static StatType Default => StatType.MELPwr;
}
public class UnitRes
{
/// <summary>
/// TEC resistance.
/// </summary>
[JsonProperty("tec_res")]
public byte TecRes { get; set; }
/// <summary>
/// TEC defence.
/// </summary>
[JsonProperty("tec_def")]
public ushort TecDef { get; set; }
/// <summary>
/// RNG defence.
/// </summary>
[JsonProperty("rng_def")]
public ushort RngDef { get; set; }
/// <summary>
/// MEL defence.
/// </summary>
[JsonProperty("mel_def")]
public ushort MelDef { get; set; }
/// <summary>
/// Additional HP.
/// </summary>
[JsonProperty("hp")]
public ushort Hp { get; set; }
/// <summary>
/// Additional PP.
/// </summary>
[JsonProperty("pp")]
public byte Pp { get; set; }
/// <summary>
/// Dark resistance.
/// </summary>
[JsonProperty("dark_res")]
public byte DarkRes { get; set; }
/// <summary>
/// Light resistance.
/// </summary>
[JsonProperty("light_res")]
public byte LightRes { get; set; }
/// <summary>
/// Wind resistance.
/// </summary>
[JsonProperty("wind_res")]
public byte WindRes { get; set; }
/// <summary>
/// Lightning resistance.
/// </summary>
[JsonProperty("lightning_res")]
public byte LightningRes { get; set; }
/// <summary>
/// Ice resistance.
/// </summary>
[JsonProperty("ice_res")]
public byte IceRes { get; set; }
/// <summary>
/// Fire resistance.
/// </summary>
[JsonProperty("fire_res")]
public byte FireRes { get; set; }
/// <summary>
/// RNG resistance.
/// </summary>
[JsonProperty("rng_res")]
public byte RngRes { get; set; }
/// <summary>
/// MEL resistance.
/// </summary>
[JsonProperty("mel_res")]
public byte MelRes { get; set; }
}
public class UnitAtk
{
/// <summary>
/// Additional MEL attack.
/// </summary>
[JsonProperty("mel_atk")]
public ushort MelAtk { get; set; }
/// <summary>
/// Additional RNG attack.
/// </summary>
[JsonProperty("rng_atk")]
public ushort RngAtk { get; set; }
/// <summary>
/// Additional TEC attack.
/// </summary>
[JsonProperty("tec_atk")]
public ushort TecAtk { get; set; }
/// <summary>
/// Additional DEX.
/// </summary>
[JsonProperty("dex")]
public ushort Dex { get; set; }
[JsonProperty("unk_atk")]
public byte UnkAtk { get; set; }
}
public class GenderFlagsConverter : JsonConverter<GenderFlags>
{
public override GenderFlags ReadJson(JsonReader reader, Type objectType, GenderFlags existingValue, bool hasExistingValue, JsonSerializer serializer)
{
var value = reader.Value as string;
if (value != null)
{
GenderFlags genderFlags = GenderFlags.NONE;
// 按 '|' 分割字符串
var flags = value.Split('|');
foreach (var flag in flags)
{
var trimmedFlag = flag.Trim();
if (Enum.TryParse<GenderFlags>(trimmedFlag, out var parsedFlag))
{
genderFlags |= parsedFlag;
}
}
return genderFlags;
}
return GenderFlags.NONE;
}
public override void WriteJson(JsonWriter writer, GenderFlags value, JsonSerializer serializer)
{
writer.WriteValue(value.ToString());
}
}
[Flags]
/// Equipable genders.
public enum GenderFlags : byte
{
NONE = 0,
/// Males can equip.
MALE = 1 << 0,
/// Females can equip.
FEMALE = 1 << 1,
}
public class RaceFlagsConverter : JsonConverter<RaceFlags>
{
public override RaceFlags ReadJson(JsonReader reader, Type objectType, RaceFlags existingValue, bool hasExistingValue, JsonSerializer serializer)
{
var value = reader.Value as string;
if (value != null)
{
RaceFlags raceFlags = RaceFlags.NONE;
// 按 '|' 分割字符串
var flags = value.Split('|');
foreach (var flag in flags)
{
var trimmedFlag = flag.Trim();
if (Enum.TryParse<RaceFlags>(trimmedFlag, out var parsedFlag))
{
raceFlags |= parsedFlag;
}
}
return raceFlags;
}
return RaceFlags.NONE;
}
public override void WriteJson(JsonWriter writer, RaceFlags value, JsonSerializer serializer)
{
writer.WriteValue(value.ToString());
}
}
[Flags]
/// Equipable races.
public enum RaceFlags : byte
{
NONE = 0,
/// Humans can equip.
HUMAN = 1 << 0,
/// Newmans can equip.
NEWMAN = 1 << 1,
/// CASTs can equip.
CAST = 1 << 2,
/// Deumans can equip.
DEUMAN = 1 << 3,
}
}

View File

@ -1,37 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public enum ItemTypes : ushort
{
NoItem = 0,
Weapon,
Clothing,
Consumable,
Unknown,
Unit,
Camo = 10,
}
[Flags]
public enum ItemFlags
{
Locked = 0x01,
BoundToOwner = 0x02
}
public enum ItemElement
{
None,
Fire,
Ice,
Lightning,
Wind,
Light,
Dark
}
}

View File

@ -1,68 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public struct MailId
{
/// Mail ID.
public uint MailIdValue;
public uint Unk1;
public uint Unk2;
// Constructor for easier initialization
public MailId(uint mailId, uint unk1, uint unk2)
{
MailIdValue = mailId;
Unk1 = unk1;
Unk2 = unk2;
}
}
public unsafe struct MailHeader
{
/// Mail ID.
public uint MailIdValue;
public uint Unk2;
/// Sender player ID (?).
public uint UserId;
public fixed byte Unk3[0x14]; // byte[0x14]
public uint Unk4;
public uint Unk5;
/// Mail receive timestamp.
public TimeSpan ReceiveTime;
public uint Unk6;
/// Sender name (0x22 length).
public string Sender;
/// Mail subject (0x2A length).
public string Subject;
// Constructor for easier initialization
public MailHeader(uint mailId, uint unk2, uint userId, byte[] unk3, uint unk4, uint unk5, TimeSpan receiveTime, uint unk6, string sender, string subject)
{
MailIdValue = mailId;
Unk2 = unk2;
UserId = userId;
//Unk3 = 0;
Unk4 = unk4;
Unk5 = unk5;
ReceiveTime = receiveTime;
Unk6 = unk6;
Sender = sender.Length > 0x22 ? sender.Substring(0, 0x22) : sender.PadRight(0x22, '\0');
Subject = subject.Length > 0x2A ? subject.Substring(0, 0x2A) : subject.PadRight(0x2A, '\0');
}
}
[Flags]
public enum MailType : uint
{
ALL = 0x00000065,
SYSTEM = 0x0000000A,
CHAR = 0x00000009,
FRIENDANDGUILD = 0x00000001,
MAILSENDED = 0x00000006
}
}

View File

@ -1,78 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public unsafe struct Unk2Struct
{
public fixed uint Unk[0x40]; // Fixed length array equivalent in C#
}
public struct Mission
{
/*5 - main
1 - daily
2 - weekly
7 - tier */
/// Mission type.
public uint MissionType; // Mission type.
public uint StartDate; // Mission start timestamp.
public uint EndDate; // Mission end timestamp.
public uint Id; // Mission ID.
public uint Unk5;
public uint CompletionDate; // Last completion timestamp.
public uint Unk7;
public uint Unk8;
public uint Unk9;
public uint Unk10;
public uint Unk11;
public uint Unk12;
public uint Unk13;
public uint Unk14;
public uint Unk15;
}
public struct MissionPassItem
{
/// Reward ID (?).
public uint Id { get; set; }
/// Reward tier.
public uint Tier { get; set; }
/// Gold pass only flag.
public uint IsGold { get; set; }
public uint Unk4 { get; set; }
/// Group ID.
public uint Group { get; set; }
public uint Unk6 { get; set; }
public uint Unk7 { get; set; }
public uint Unk8 { get; set; }
/// Item data.
public PSO2Items Item { get; set; }
// Constructor for struct
public MissionPassItem(uint id, uint tier, uint isGold, uint unk4, uint group, uint unk6, uint unk7, uint unk8, PSO2Items item)
{
Id = id;
Tier = tier;
IsGold = isGold;
Unk4 = unk4;
Group = group;
Unk6 = unk6;
Unk7 = unk7;
Unk8 = unk8;
Item = item;
}
}
}

View File

@ -1,54 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Protocol;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public class NetInterface
{
/// <summary>
/// Interface status.
/// </summary>
public uint State { get; set; }
/// <summary>
/// Interface MAC address.
/// </summary>
public string Mac { get; set; } = new string('\0', 0x18); // 以字符串形式存储
public void ReadFromStream(PacketReader reader)
{
State = reader.ReadUInt32();
Mac = Encoding.ASCII.GetString(reader.ReadBytes(0x18)).TrimEnd('\0');
}
public override string ToString()
{
return $"状态: {State}, MAC: {Mac}";
}
public void UpdateNetInterface(int id, int accountid, NetInterface updatedInterface)
{
using (var db = new ServerEf())
{
var existingInterface = db.AccountsNetInterFaces
.FirstOrDefault(x => x.AccountId == accountid && x.id == id);
if (existingInterface != null)
{
// 更新其他字段
existingInterface.State = (int)updatedInterface.State;
existingInterface.Mac = updatedInterface.Mac;
// 保存更改
db.SaveChanges();
}
}
}
}
}

View File

@ -1,29 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public class Orders
{
public struct ClientOrder
{
public uint unk1;
public uint id;
public uint status;
public uint finish_date;
}
public struct OrderStatus
{
public uint unk1;
public uint unk2;
public uint unk3;
public uint unk4;
public uint unk5;
public uint unk6;
}
}
}

View File

@ -1,448 +0,0 @@
using PSO2SERVER.Protocol;
using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.InteropServices;
using System.Text;
using static PSO2SERVER.Models.CharacterStruct;
namespace PSO2SERVER.Models
{
[StructLayout(LayoutKind.Sequential)]
public unsafe struct ShortItemId
{
public byte ItemType;
public byte Id;
public ushort Subid;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct ItemId
{
public ushort ItemType;
public ushort Id;
public ushort Unk3;
public ushort Subid;
}
[StructLayout(LayoutKind.Explicit)]
public unsafe struct Items
{
[FieldOffset(0)]
public PSO2ItemWeapon Weapon;
[FieldOffset(0)]
public PSO2ItemClothing Clothing;
[FieldOffset(0)]
public PSO2ItemConsumable Consumable;
[FieldOffset(0)]
public PSO2ItemCamo Camo;
[FieldOffset(0)]
public PSO2ItemUnit Unit;
[FieldOffset(0)]
public fixed byte Unknown[0x28];
[FieldOffset(0)]
public fixed byte Noitem[0x28];
}
[StructLayout(LayoutKind.Sequential)]
public struct PSO2Items
{
public ulong uuid;
public ItemId id;
public Items data;
public byte[] ToByteArray()
{
using (var pkt = new PacketWriter())
{
pkt.Write(uuid);
pkt.WriteStruct(id);
pkt.WriteStruct(data);
return pkt.ToArray();
}
}
// 从字节流转换为结构体
public static PSO2Items FromByteArray(byte[] byteArray)
{
using (var reader = new PacketReader(byteArray))
{
PSO2Items items = new PSO2Items
{
uuid = reader.ReadUInt64(),
id = reader.ReadStruct<ItemId>(),
data = reader.ReadStruct<Items>()
};
return items;
}
}
public ulong GetGUID()
{
return uuid;
}
public void SetGUID(ulong guid)
{
uuid = guid;
}
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct PSO2ItemWeapon
{
/// Item flags.
public byte flags;
/// Item element.
public byte element;
/// Item force.
public byte force;
public byte grind;
public byte grindPercent;
public byte unk1;
public ushort unk2;
/// Item affix IDs.
public fixed ushort affixes[8];
/// Item potential.
public uint potential;
public byte extend;
public byte unk4;
public ushort unk5;
public uint unk6;
public uint unk7;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct PSO2ItemClothing
{
public ushort flags;
public fixed byte unk1[0x14];
public HSVColor Color;
public fixed byte unk2[0xA];
public ushort Unk3;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct PSO2ItemConsumable
{
public ushort flags;
public fixed byte unk1[0x24];
public ushort amount;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct PSO2ItemCamo
{
public byte unk1;
public byte unk2;
public byte unk3;
public fixed byte unk4[0x24];
public byte unk5;
}
[StructLayout(LayoutKind.Sequential)]
public unsafe struct PSO2ItemUnit
{
public byte flags;
public byte EnhLevel;
public byte EnhPercent;
public byte Unk1;
// 使用 fixed 数组来存储附加信息
public fixed ushort Affixes[8]; // Item affix IDs (0 to 4095)
public fixed byte unk4[0x7];
public uint Potential;
// 使用 fixed 数组来存储未知字段
public fixed byte Unk2[4];
public uint Unk3;
public ushort Unk4;
public ushort Unk5;
// 提供访问固定数组的属性
public Span<ushort> AffixSpan
{
get
{
fixed (ushort* p = Affixes)
{
return new Span<ushort>(p, 8);
}
}
}
}
public struct Campaign
{
/// Campaign ID.
public uint Id; // 对应 Rust 的 u32
/// Start timestamp.
public TimeSpan StartDate; // 对应 Rust 的 Duration
/// End timestamp.
public TimeSpan EndDate; // 对应 Rust 的 Duration
/// Campaign title (固定长度 0x3E).
public const int TitleLength = 0x3E;
public byte[] titleBytes;
public string Title
{
get => Encoding.ASCII.GetString(titleBytes).TrimEnd('\0');
set
{
titleBytes = new byte[TitleLength];
byte[] valueBytes = Encoding.ASCII.GetBytes(value);
Array.Copy(valueBytes, titleBytes, Math.Min(valueBytes.Length, TitleLength));
}
}
/// Campaign conditions (固定长度 0x102).
public const int ConditionsLength = 0x102;
public byte[] conditionsBytes;
public string Conditions
{
get => Encoding.ASCII.GetString(conditionsBytes).TrimEnd('\0');
set
{
conditionsBytes = new byte[ConditionsLength];
byte[] valueBytes = Encoding.ASCII.GetBytes(value);
Array.Copy(valueBytes, conditionsBytes, Math.Min(valueBytes.Length, ConditionsLength));
}
}
// 从数据流中读取 Campaign
public static Campaign FromStream(Stream stream)
{
using (BinaryReader reader = new BinaryReader(stream, Encoding.ASCII, true))
{
Campaign campaign = new Campaign
{
Id = reader.ReadUInt32(),
StartDate = TimeSpan.FromTicks(reader.ReadInt64()),
EndDate = TimeSpan.FromTicks(reader.ReadInt64()),
};
campaign.titleBytes = reader.ReadBytes(TitleLength);
campaign.conditionsBytes = reader.ReadBytes(ConditionsLength);
return campaign;
}
}
// 将 Campaign 写入数据流
public void WriteToStream(Stream stream)
{
using (BinaryWriter writer = new BinaryWriter(stream, Encoding.ASCII, true))
{
writer.Write(Id);
writer.Write(StartDate.Ticks);
writer.Write(EndDate.Ticks);
writer.Write(titleBytes);
writer.Write(conditionsBytes);
}
}
}
[StructLayout(LayoutKind.Sequential)]
public struct NamedId
{
public string Name { get; set; }
public ItemId Id { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct CampaignItemDefinition
{
/// <summary>
/// Campaign ID.
/// </summary>
public uint CampaignId; // Equivalent to u32
/// <summary>
/// Number of items in the following array.
/// </summary>
public uint ItemAmount; // Equivalent to u32
/// <summary>
/// Campaign items.
/// </summary>
public List<CampaignItem> Items; // Using List instead of fixed-size array
}
[StructLayout(LayoutKind.Sequential)]
public struct CampaignItem
{
/// <summary>
/// Item ID.
/// </summary>
public ItemId Id; // Assuming ItemId is a type, use it as-is
/// <summary>
/// Item amount.
/// </summary>
public uint Amount; // Equivalent to u32
/// <summary>
/// Unknown field.
/// </summary>
public uint Unk; // Equivalent to u32
}
[StructLayout(LayoutKind.Sequential)]
public struct StorageInfo
{
/// <summary>
/// Total space in the storage.
/// </summary>
public uint TotalSpace; // Equivalent to u32
/// <summary>
/// Used space in the storage.
/// </summary>
public uint UsedSpace; // Equivalent to u32
/// <summary>
/// Storage ID.
/// </summary>
public byte StorageId; // Equivalent to u8
/// <summary>
/// Storage type (?).
/// </summary>
public byte StorageType; // Equivalent to u8
/// <summary>
/// Is the storage locked (?).
/// </summary>
public byte IsLocked; // Equivalent to u8
/// <summary>
/// Is the storage enabled (?).
/// </summary>
public byte IsEnabled; // Equivalent to u8
}
[StructLayout(LayoutKind.Sequential)]
public struct MaterialStorageItem
{
/// <summary>
/// Item category.
/// </summary>
public ushort Id; // Equivalent to u16
/// <summary>
/// Item ID.
/// </summary>
public ushort Subid; // Equivalent to u16
/// <summary>
/// Item amount.
/// </summary>
public ushort Amount; // Equivalent to u16
/// <summary>
/// Unknown field.
/// </summary>
public ushort Unk4; // Equivalent to u16
}
[StructLayout(LayoutKind.Sequential)]
public struct UUIDAmount
{
public ulong Uuid { get; set; }
public ushort Amount { get; set; }
public ushort Unk { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct EquipedItem
{
public PSO2Items Item { get; set; }
public uint Unk { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct MoveStorageItemRequest
{
public ulong Uuid { get; set; }
public byte Amount { get; set; }
public byte Unk { get; set; }
public ushort StorageId { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct NewStorageItem
{
public PSO2Items Item { get; set; }
public uint StorageId { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct NewInventoryItem
{
public PSO2Items Item { get; set; }
public ushort Amount { get; set; }
public ushort IsNew { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct UpdatedItem
{
public ulong Uuid { get; set; }
public uint NewAmount { get; set; }
public uint StorageId { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct UpdatedInventoryItem
{
public ulong Uuid { get; set; }
public ushort NewAmount { get; set; }
public ushort Moved { get; set; }
}
[StructLayout(LayoutKind.Sequential)]
public struct UpdatedStorageItem
{
public ulong Uuid { get; set; }
public ushort NewAmount { get; set; }
public ushort Moved { get; set; }
public uint StorageId { get; set; }
}
public static class AffixUtils
{
public static ushort[] ReadPackedAffixes(Stream reader)
{
byte[] packed = new byte[12];
reader.Read(packed, 0, packed.Length);
ushort[] affixes = new ushort[8];
for (int i = 0; i < 4; i++)
{
affixes[i * 2] = BitConverter.ToUInt16(new byte[] { packed[i * 3], (byte)((packed[i * 3 + 2] & 0xF0) >> 4) }, 0);
affixes[i * 2 + 1] = BitConverter.ToUInt16(new byte[] { packed[i * 3 + 1], (byte)(packed[i * 3 + 2] & 0xF) }, 0);
}
return affixes;
}
public static void WritePackedAffixes(ushort[] affixes, Stream writer)
{
byte[] packed = new byte[12];
for (int i = 0; i < 4; i++)
{
byte[] affix1 = BitConverter.GetBytes(affixes[i * 2]);
byte[] affix2 = BitConverter.GetBytes(affixes[i * 2 + 1]);
packed[i * 3] = affix1[0];
packed[i * 3 + 1] = affix2[0];
packed[i * 3 + 2] = (byte)((affix1[1] << 4) | (affix2[1] & 0xF));
}
writer.Write(packed, 0, packed.Length);
}
}
}

View File

@ -1,89 +0,0 @@
using Newtonsoft.Json;
using PSO2SERVER.Json;
using PSO2SERVER.Protocol;
namespace PSO2SERVER.Models
{
// 对象类型枚举
public enum ObjectType : ushort
{
Unknown = 0x0000,
/// 玩家对象。
Player = 0x0004,
/// 区域对象。
Map = 0x0005,
/// 大部分的对象和NPC。
Object = 0x0006,
/// 一些可破坏的物体(例如一些树木)。
StaticObject = 0x0007,
/// 任务对象。11
Quest = 0x000B,
/// 队伍对象。13
Party = 0x000D,
/// 世界地图池中的对象。16
World = 0x0010,
/// 非玩家可控的伙伴。22
APC = 0x0016,
/// 未定义的类型。
Undefined = 0xFFFF
}
// 对象头部结构体
public struct ObjectHeader
{
/// 对象的ID。
[JsonProperty("id")]
public uint ID { get; set; }
[JsonProperty("unk")]
public uint Padding { get; set; }
/// 对象的类型。
[JsonProperty("entity_type")]
[JsonConverter(typeof(ObjectTypeConverter))]
public ObjectType ObjectType { get; set; }
/// 对象所在的区域ID。玩家对象没有设置这个字段。
[JsonProperty("map_id")]
public ushort MapID { get; set; }
// 只包含ID和类型的构造函数
public ObjectHeader(uint id, ObjectType objectType)
{
ID = id;
Padding = 0;
ObjectType = objectType;
MapID = 0;
}
// 包含ID、类型和区域ID的构造函数
public ObjectHeader(uint id, ObjectType objectType, ushort mapid)
{
ID = id;
Padding = 0;
ObjectType = objectType;
MapID = mapid;
}
// 从数据流中读取数据并填充到当前结构体
public void ReadObjectHeaderFromStream(PacketReader reader)
{
ID = reader.ReadUInt32(); // 读取对象ID
Padding = reader.ReadUInt32(); // 读取填充部分
ObjectType = (ObjectType)reader.ReadUInt16(); // 读取对象类型
MapID = reader.ReadUInt16(); // 读取区域ID
}
// 将当前结构体的数据写入到数据流
public void WriteObjectHeaderToStream(PacketWriter writer)
{
writer.Write(ID); // 写入对象ID
writer.Write(Padding); // 写入填充部分
writer.Write((ushort)ObjectType); // 写入对象类型
writer.Write(MapID); // 写入区域ID
}
// ToString 方法,用来返回 ObjectHeader 的字符串表示
public override string ToString()
{
return $"ObjectHeader [ID={ID}, Padding={Padding}, ObjectType={ObjectType}, MapID={MapID}]";
}
}
}

View File

@ -1,150 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using System.Runtime.Remoting.Messaging;
using System.Text;
using PSO2SERVER.Database;
using PSO2SERVER.Protocol;
using PSO2SERVER.Zone;
namespace PSO2SERVER.Models
{
public class PSOObject
{
public struct PSOObjectThing
{
public uint Data;
public PSOObjectThing(uint data)
{
Data = data;
}
}
public ObjectHeader Header { get; set; }
public PSOLocation Position { get; set; }
public string Name { get; set; }
public uint ThingFlag { get; set; }
public PSOObjectThing[] Things { get; set; }
public virtual byte[] GenerateSpawnBlob()
{
PacketWriter writer = new PacketWriter();
Header.WriteObjectHeaderToStream(writer);
writer.WritePosition(Position);
writer.Seek(2, SeekOrigin.Current); // Padding I guess...
writer.WriteFixedLengthASCII(Name, 0x34);
writer.Write(ThingFlag);
writer.Write(Things.Length);
foreach (PSOObjectThing thing in Things)
{
writer.WriteStruct(thing);
}
return writer.ToArray();
}
internal static PSOObject FromPacketBin(byte[] v)
{
PacketReader reader = new PacketReader(v);
PSOObject obj = new PSOObject();
reader.ReadStruct<PacketHeader>(); //Skip over header
obj.Header = reader.ReadObjectHeader();
obj.Position = reader.ReadEntityPosition();
reader.ReadUInt16(); // Seek 2
obj.Name = reader.ReadFixedLengthAscii(0x34);
obj.ThingFlag = reader.ReadUInt32();
uint thingCount = reader.ReadUInt32();
obj.Things = new PSOObjectThing[thingCount];
for (int i = 0; i < thingCount; i++)
{
obj.Things[i] = reader.ReadStruct<PSOObjectThing>();
}
return obj;
}
internal static PSOObject FromDBObject(GameObject dbObject)
{
PSOObject psoObj = new PSOObject();
psoObj.Header = new ObjectHeader((uint)dbObject.ObjectID, ObjectType.Object);
psoObj.Name = dbObject.ObjectName;
psoObj.Position = new PSOLocation(dbObject.RotX, dbObject.RotY, dbObject.RotZ, dbObject.RotW, dbObject.PosX, dbObject.PosY, dbObject.PosZ);
int thingCount = dbObject.ObjectFlags.Length / 4; // I hope this will work
psoObj.Things = new PSOObjectThing[thingCount];
for(int i = 0; i < psoObj.Things.Length; i++)
{
psoObj.Things[i] = new PSOObjectThing(BitConverter.ToUInt32(dbObject.ObjectFlags, 4 * i)); // This should work?
}
return psoObj;
}
}
public class PSONPC : PSOObject
{
public unsafe struct NPC {
public ObjectHeader objHeader;
public PSOLocation objPosition;
public ushort unk1;
public string objName;
public uint unk2;
public fixed byte unk3[0x0C];
public ushort unk4;
public ushort unk5;
public uint unk6;
public uint unk7;
public uint unk8;
public uint unk9;
public uint unk10;
public uint unk11;
public uint unk12;
public string unk13; //Ascii
}
public override byte[] GenerateSpawnBlob()
{
NPC npc = new NPC
{
objHeader = Header,
objPosition = Position,
objName = Name,
unk8 = 1101004800,
unk12 = 1,
unk13 = "",
};
PacketWriter writer = new PacketWriter();
writer.WriteObjectHeader(Header);
writer.WritePosition(Position);
writer.Write((UInt16)0);
writer.WriteFixedLengthASCII(Name, 0x20);
writer.Write(0); // Padding?
writer.Write(new byte[0xC]); // Unknown, usually zero
writer.Write((UInt16)0);
writer.Write((UInt16)0);
writer.Write((UInt32)0);
writer.Write((UInt32)0);
writer.Write((UInt32)1101004800); // Always this
writer.Write((UInt32)0);
writer.Write((UInt32)0);
writer.Write((UInt32)0);
writer.Write((UInt32)1);
writer.WriteMagic(1, 0x9FCD, 0xE7);
writer.Write((UInt32)0);
return writer.ToArray();
}
}
}

View File

@ -1,205 +0,0 @@
using Newtonsoft.Json;
using PSO2SERVER.Json;
using PSO2SERVER.Protocol;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using static PSO2SERVER.Models.PSOPalette;
namespace PSO2SERVER.Models
{
public class SubPalette
{
[JsonProperty("items")]
[JsonConverter(typeof(FixedListConverter<PalettePA>))]
// Items in the subpalette.
public FixedList<PalettePA> Items { get; set; } = new FixedList<PalettePA>(0x0C, new PalettePA());
// 从字节流中读取数据
public void ReadFromStream(PacketReader reader)
{
// 假设 Items 长度固定为 12
for (int i = 0; i < Items.Length; i++)
{
// 读取每个 PalettePA
var palettePA = new PalettePA();
palettePA.ReadFromStream(reader);
Items.SetItem(i, palettePA);
}
}
// 写入数据到字节流
public void WriteToStream(PacketWriter writer)
{
// 遍历 FixedList将每个 PalettePA 写入字节流
foreach (var item in Items)
{
item.WriteToStream(writer);
}
}
}
public class PalettePA
{
[JsonProperty("id")]
/// PA ID.
public byte ID { get; set; } = 0;
[JsonProperty("category")]
/// PA category.
public byte Category { get; set; } = 0;
[JsonProperty("unk")]
public byte Unk { get; set; } = 0;
[JsonProperty("level")]
/// PA level.
public byte Level { get; set; } = 0;
public PalettePA()
{
}
public PalettePA(byte id, byte category, byte unk, byte level)
{
ID = id;
Category = category;
Unk = unk;
Level = level;
}
public void ReadFromStream(PacketReader reader)
{
ID = reader.ReadByte();
Category = reader.ReadByte();
Unk = reader.ReadByte();
Level = reader.ReadByte();
}
public void WriteToStream(PacketWriter writer)
{
writer.Write(ID);
writer.Write(Category);
writer.Write(Unk);
writer.Write(Level);
}
}
public class WeaponPalette
{
public ulong Uuid { get; set; } = 0;
public uint Unk1 { get; set; } = 0;
public PalettePA Unk2 { get; set; } = new PalettePA();
public PalettePA Unk3 { get; set; } = new PalettePA();
public PalettePA Unk4 { get; set; } = new PalettePA();
public uint[] Unk { get; set; } = new uint[3];
public uint PetId { get; set; } = 0;
public List<PalettePA> Skills { get; set; } = new List<PalettePA>(6);
public WeaponPalette()
{
}
public void ReadFromStream(PacketReader reader)
{
Uuid = reader.ReadUInt64();
Unk1 = reader.ReadUInt32();
Unk2 = new PalettePA();
Unk2.ReadFromStream(reader);
Unk3 = new PalettePA();
Unk3.ReadFromStream(reader);
Unk4 = new PalettePA();
Unk4.ReadFromStream(reader);
for (int i = 0; i < Unk.Length; i++)
{
Unk[i] = reader.ReadUInt32();
}
PetId = reader.ReadUInt32();
for (int i = 0; i < Skills.Count; i++)
{
Skills[i].ReadFromStream(reader);
}
}
public void WriteToStream(PacketWriter writer)
{
writer.Write(Uuid);
writer.Write(Unk1);
Unk2.WriteToStream(writer);
Unk3.WriteToStream(writer);
Unk4.WriteToStream(writer);
foreach (var value in Unk)
{
writer.Write(value);
}
writer.Write(PetId);
foreach (var skill in Skills)
{
skill.WriteToStream(writer);
}
}
}
public class PSOPalette
{
public uint CurPalette { get; set; } = 0;
public uint CurSubpalette { get; set; } = 0;
public uint CurBook { get; set; } = 0;
public FixedList<WeaponPalette> Palettes { get; set; } = new FixedList<WeaponPalette>(6, new WeaponPalette());
public FixedList<SubPalette> Subpalettes { get; set; } = new FixedList<SubPalette>(6, new SubPalette());
public List<uint> DefaultPas { get; set; } = new List<uint>();
public PSOPalette()
{
Palettes = new FixedList<WeaponPalette>(6, new WeaponPalette());
Subpalettes = new FixedList<SubPalette>(6, new SubPalette());
DefaultPas = new List<uint>();
}
public void ReadFromStream(PacketReader reader)
{
CurPalette = reader.ReadUInt32();
CurSubpalette = reader.ReadUInt32();
CurBook = reader.ReadUInt32();
for (int i = 0; i < Palettes.Length; i++)
{
Palettes[i].ReadFromStream(reader);
}
for (int i = 0; i < Subpalettes.Length; i++)
{
Subpalettes[i].ReadFromStream(reader);
}
int defaultPasCount = reader.ReadInt32(); // 假设先读入数量
DefaultPas.Clear();
for (int i = 0; i < defaultPasCount; i++)
{
DefaultPas.Add(reader.ReadUInt32());
}
}
public void WriteToStream(PacketWriter writer)
{
writer.Write(CurPalette);
writer.Write(CurSubpalette);
writer.Write(CurBook);
for (int i = 0; i < Palettes.Length; i++)
{
Palettes[i].WriteToStream(writer);
}
for (int i = 0; i < Subpalettes.Length; i++)
{
Subpalettes[i].WriteToStream(writer);
}
foreach (var value in DefaultPas)
{
writer.Write(value);
}
}
}
}

View File

@ -1,460 +0,0 @@
using System;
using System.Diagnostics.Eventing.Reader;
using System.Runtime.InteropServices;
using Newtonsoft.Json;
using PSO2SERVER.Json;
using PSO2SERVER.Protocol;
using PSO2SERVER.Protocol.Handlers;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Models
{
public class QuestDef
{
/// <summary>
/// 任务日期
/// </summary>
[JsonProperty("date")]
[JsonConverter(typeof(AsciiStringConverter))]
public string Date { get; set; } = new string(' ', 32); // 0x20 length string
/// <summary>
/// 任务对象
/// </summary>
[JsonProperty("quest_obj")]
public ObjectHeader QuestObj;
/// <summary>
/// 任务名称的ID
/// </summary>
[JsonProperty("name_id")]
public uint NameId;
/// <summary>
/// 未知字段3数组大小为27
/// </summary>
[JsonProperty("unk3")]
public uint[] Unk3 = new uint[27];
/// <summary>
/// 未知字段4
/// </summary>
[JsonProperty("unk4")]
public ushort Unk4;
/// <summary>
/// 未知字段5
/// </summary>
[JsonProperty("unk5")]
public byte Unk5;
/// <summary>
/// 未知字段6
/// </summary>
[JsonProperty("unk6")]
public byte Unk6;
/// <summary>
/// 未知字段7数组大小为20
/// </summary>
[JsonProperty("unk7")]
public uint[] Unk7 = new uint[20];
/// <summary>
/// 未知字段8数组大小为3
/// </summary>
[JsonProperty("unk8")]
public ushort[] Unk8 = new ushort[3];
/// <summary>
/// 任务时长
/// </summary>
[JsonProperty("length")]
public byte Length;
/// <summary>
/// 任务的队伍类型
/// </summary>
[JsonProperty("party_type")]
[JsonConverter(typeof(QuestPartyTypeConverter))]
public QuestPartyType PartyType;
/// <summary>
/// 可用的难度
/// </summary>
[JsonProperty("difficulties")]
public QuestDifficultyType Difficulties;
/// <summary>
/// 已完成的难度
/// </summary>
[JsonProperty("difficulties_completed")]
public QuestDifficultyType DifficultiesCompleted;
/// <summary>
/// 未知字段9
/// </summary>
[JsonProperty("unk9")]
public byte Unk9;
/// <summary>
/// 所需等级
/// </summary>
[JsonProperty("req_level")]
public byte ReqLevel;
/// <summary>
/// 所需副职业等级
/// </summary>
[JsonProperty("sub_class_req_level")]
public byte SubClassReqLevel;
/// <summary>
/// 敌人等级
/// </summary>
[JsonProperty("enemy_level")]
public byte EnemyLevel;
/// <summary>
/// 未知字段10
/// </summary>
[JsonProperty("unk10")]
public byte Unk10;
/// <summary>
/// 任务类型
/// </summary>
[JsonProperty("quest_type")]
public QuestType QuestType;
/// <summary>
/// 未知字段11数组大小为6
/// </summary>
[JsonProperty("unk11")]
public byte[] Unk11 = new byte[6];
/// <summary>
/// 未知字段12
/// </summary>
[JsonProperty("unk12")]
public ushort Unk12;
/// <summary>
/// 未知字段13数组大小为2
/// </summary>
[JsonProperty("unk13")]
public uint[] Unk13 = new uint[2];
/// <summary>
/// 未知字段14
/// </summary>
[JsonProperty("unk14")]
public ushort Unk14;
/// <summary>
/// 未知字段15数组大小为2
/// </summary>
[JsonProperty("unk15")]
public byte[] Unk15 = new byte[2];
// 任务相关的其他字段
// public QuestThing[] Unk15 = new QuestThing[16];
/// <summary>
/// 未知字段16大小为0x320字节
/// </summary>
[JsonProperty("unk16")]
public byte[] Unk16 = new byte[0x320];
// 从数据流中读取数据并填充到当前结构体
public void ReadFromStream(PacketReader reader)
{
//TODO
}
// 将当前结构体的数据写入到数据流
public void WriteToStream(PacketWriter pkt)
{
pkt.WriteFixedLengthASCII(Date, 0x20);
pkt.WriteObjectHeader(QuestObj);
pkt.Write(NameId);
pkt.WriteIntArray(Unk3);
pkt.Write(Unk4);
pkt.Write(Unk5);
pkt.Write(Unk6);
pkt.WriteIntArray(Unk7);
pkt.WriteShortArray(Unk8);
pkt.Write(Length);
pkt.Write((byte)PartyType);
pkt.Write((byte)Difficulties);
pkt.Write((byte)DifficultiesCompleted);
pkt.Write(Unk9);
pkt.Write(ReqLevel);
pkt.Write(SubClassReqLevel);
pkt.Write(EnemyLevel);
pkt.Write(Unk10);
pkt.Write((byte)QuestType);
pkt.Write(Unk11);
pkt.Write(Unk12);
pkt.WriteIntArray(Unk13);
pkt.Write(Unk14);
pkt.Write(Unk15);
pkt.Write(Unk16);
}
}
public class QuestDiff
{
/// <summary>
/// PartyQuest date.
/// </summary>
[JsonProperty("date")]
[JsonConverter(typeof(AsciiStringConverter))]
public string Date { get; set; } = new string(' ', 32); // 0x20 length string
/// <summary>
/// PartyQuest object.
/// </summary>
[JsonProperty("quest_obj")]
public ObjectHeader QuestObj { get; set; }
/// <summary>
/// ID of the quest name.
/// </summary>
[JsonProperty("name_id")]
public uint NameId { get; set; }
/// <summary>
/// Planet ID.
/// </summary>
[JsonProperty("planet")]
public byte Planet { get; set; }
/// <summary>
/// Area ID.
/// </summary>
[JsonProperty("area")]
public byte Area { get; set; }
/// <summary>
/// Unknown field.
/// </summary>
[JsonProperty("unk1")]
public byte Unk1 { get; set; }
/// <summary>
/// Unknown field.
/// </summary>
[JsonProperty("unk2")]
public byte Unk2 { get; set; }
/// <summary>
/// Difficulty list.
/// </summary>
[JsonProperty("diffs")]
public QuestDifficultyEntry[] Diffs { get; set; } = new QuestDifficultyEntry[8];
// 从数据流中读取数据并填充到当前结构体
public void ReadFromStream(PacketReader reader)
{
//TODO
}
// 将当前结构体的数据写入到数据流
public void WriteToStream(PacketWriter pkt)
{
pkt.WriteFixedLengthASCII(Date, 0x20);
pkt.WriteObjectHeader(QuestObj);
pkt.Write(NameId);
pkt.Write(Planet);
pkt.Write(Area);
pkt.Write(Unk1);
pkt.Write(Unk2);
pkt.WriteStructArray(Diffs);
}
}
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public unsafe struct QuestDefiniton
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
public string dateOrSomething;
public ObjectHeader quest_obj;
public uint questNameid;
//27个uint
public uint field_30;
public uint field_34;
public uint field_38;
public uint field_3C;
public uint field_40;
public uint field_44;
public uint field_48;
public uint field_4C;
public uint field_50;
public uint field_54;
public uint field_58;
public uint field_5C;
public uint field_60;
public uint field_64;
public uint field_68;
public uint field_6C;
public uint field_70;
public uint field_74;
public uint field_78;
public uint field_7C;
public uint field_80;
public uint field_84;
public uint field_88;
public uint field_8C;
public uint field_90;
public uint field_94;
public uint field_98;
public UInt16 field_9C;
public byte field_9E;
public byte field_9F;
//20个uint
public uint field_A0;
public uint field_A4;
public uint field_A8;
public uint field_AC;
public uint field_B0;
public uint field_B4;
public uint field_B8;
public uint field_BC;
public uint field_C0;
public uint field_C4;
public uint field_C8;
public uint field_CC;
public uint field_D0;
public uint field_D4;
public uint field_D8;
public uint field_DC;
public uint field_E0;
public uint field_E4;
public uint field_E8; // Maybe a flags
public uint field_EC;
public ushort field_F0;
public ushort field_F2;
public QuestBitfield1 questBitfield1;
/// Length of the quest.
public QuestEstimatedTime playTime;
/// Party type of the quest.
public QuestPartyType partyType;
/// Available difficulties.
public QuestDifficultyType difficulties;
/// Completed difficulties.
public QuestDifficultyType difficultiesCompleted;
public byte field_FA;
/// Required level.
public byte req_level;
/// Required sub level.
public byte sub_class_req_level;
/// Enemy level.
public byte enemy_level;
public byte field_FE;
/// Type of the quest.
public QuestType quest_type;
public byte field_100;
public byte field_101;
public byte field_102;
public byte field_103;
public byte field_104;
public byte field_105;
public ushort field_106;
public uint field_108;
public uint field_10C;
public ushort field_110;
public byte field_112;
public byte field_113;
public fixed byte field_114[0x320];
}
public class PartyQuest
{
public int seed { get; set; }
public string name { get; set; }
public QuestDefiniton questDef { get; set; }
public QuestData Quest { get; set; }
public ushort Diff { get; set; } = 0;
public PartyQuest()
{
}
public PartyQuest(string name)
{
this.name = name;
}
public PartyQuest(string name, QuestData quest)
{
this.name = name;
Quest = quest;
}
}
public struct QuestThing
{
public uint field_0;
public uint field_4;
public byte field_8;
public byte field_9;
public ushort field_A;
}
//Size: 308 bytes, confirmed in unpacker
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public unsafe struct QuestDifficulty
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 24)]
public string dateOrSomething;
public ObjectHeader quest_obj;
public uint name_id;
public byte area;
public byte planet;
public byte unk1;
public byte unk2;
public QuestDifficultyEntry difficulty1;
public QuestDifficultyEntry difficulty2;
public QuestDifficultyEntry difficulty3;
public QuestDifficultyEntry difficulty4;
public QuestDifficultyEntry difficulty5;
public QuestDifficultyEntry difficulty6;
public QuestDifficultyEntry difficulty7;
public QuestDifficultyEntry difficulty8;
}
//Size: 32, confirmed in ctor TODO
public struct QuestDifficultyEntry
{
public byte ReqLevel;
public byte SubClassReqLevel;
public byte MonsterLevel;
public byte Unk1;
public uint AbilityAdj;
public uint DmgLimit;
public uint TimeLimit;
public uint TimeLimit2;
public uint SuppTarget;
public uint Unk2;
public uint Enemy1;
public uint Unk3;
public uint Enemy2;
public uint Unk4;
}
}

View File

@ -1,211 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public enum QuestType : byte
{
Unk0 = 0, // 0x00
Extreme = 1, // 0x01
ARKS = 3, // 0x03
LimitedTime = 4, // 0x04
ExtremeDebug = 5, // 0x05
Blank1 = 6, // 0x06
NetCafe = 8, // 0x08
WarmingDebug = 9, // 0x09
Blank2 = 10, // 0x0A
Advance = 11, // 0x0B
Expedition = 12, // 0x0C
FreeDebug = 13, // 0x0D
ArksDebug = 14, // 0x0E
Challenge = 16, // 0x10
Urgent = 17, // 0x11
UrgentDebug = 18, // 0x12
TimeAttack = 19, // 0x13
TimeDebug = 20, // 0x14
ArksDebug2 = 21, // 0x15
ArksDebug3 = 22, // 0x16
ArksDebug4 = 23, // 0x17
ArksDebug5 = 24, // 0x18
ArksDebug6 = 25, // 0x19
ArksDebug7 = 26, // 0x1A
ArksDebug8 = 27, // 0x1B
ArksDebug9 = 28, // 0x1C
ArksDebug10 = 29, // 0x1D
Blank3 = 30, // 0x1E
Recommended = 32, // 0x20
Ultimate = 33, // 0x21
UltimateDebug = 34, // 0x22
AGP = 35, // 0x23
Bonus = 36, // 0x24
StandardTraining = 37,// 0x25
HunterTraining = 38, // 0x26
RangerTraining = 39, // 0x27
ForceTraining = 40, // 0x28
FighterTraining = 41, // 0x29
GunnerTraining = 42, // 0x2A
TechterTraining = 43, // 0x2B
BraverTraining = 44, // 0x2C
BouncerTraining = 45, // 0x2D
SummonerTraining = 46,// 0x2E
AutoAccept = 47, // 0x2F
Ridroid = 48, // 0x30
CafeAGP = 49, // 0x31
BattleBroken = 50, // 0x32
BusterDebug = 51, // 0x33
Poka12 = 52, // 0x34
StoryEP1 = 55, // 0x37
Buster = 56, // 0x38
HeroTraining = 57, // 0x39
Amplified = 58, // 0x3A
DarkBlastTraining = 61,// 0x3D
Endless = 62, // 0x3E
Blank4 = 64, // 0x40
PhantomTraining = 65, // 0x41
AISTraining = 66, // 0x42
DamageCalculation = 68,// 0x44
EtoileTraining = 69, // 0x45
Divide = 70, // 0x46
Stars1 = 71, // 0x47
Stars2 = 72, // 0x48
Stars3 = 73, // 0x49
Stars4 = 74, // 0x4A
Stars5 = 75, // 0x4B
Stars6 = 76, // 0x4C
}
[Flags]
public enum AvailableQuestType : ulong
{
EXTREME = 1UL << 1,
STORY_EP1 = 1UL << 2,
ARKS = 1UL << 3,
LIMITED_TIME = 1UL << 4,
EXTREME_DEBUG = 1UL << 5,
BLANK1 = 1UL << 6,
STORY_EP2 = 1UL << 7,
NET_CAFE = 1UL << 8,
WARMING_DEBUG = 1UL << 9,
BLANK2 = 1UL << 10,
ADVANCE = 1UL << 11,
EXPEDITION = 1UL << 12,
FREE_DEBUG = 1UL << 13,
ARKS_DEBUG = 1UL << 14,
STORY_DEBUG = 1UL << 15,
CHALLENGE = 1UL << 16,
URGENT = 1UL << 17,
URGENT_DEBUG = 1UL << 18,
TIME_ATTACK = 1UL << 19,
TIME_DEBUG = 1UL << 20,
ARKS_DEBUG2 = 1UL << 21,
ARKS_DEBUG3 = 1UL << 22,
ARKS_DEBUG4 = 1UL << 23,
ARKS_DEBUG5 = 1UL << 24,
ARKS_DEBUG6 = 1UL << 25,
ARKS_DEBUG7 = 1UL << 26,
ARKS_DEBUG8 = 1UL << 27,
ARKS_DEBUG9 = 1UL << 28,
ARKS_DEBUG10 = 1UL << 29,
BLANK3 = 1UL << 30,
STORY_EP3 = 1UL << 31,
RECOMMENDED = 1UL << 32,
ULTIMATE = 1UL << 33,
ULTIMATE_DEBUG = 1UL << 34,
AGP = 1UL << 35,
BONUS = 1UL << 36,
UNK1 = 1UL << 37,
STANDARD_TRAINING = 1UL << 38,
HUNTER_TRAINING = 1UL << 39,
RANGER_TRAINING = 1UL << 40,
FORCE_TRAINING = 1UL << 41,
FIGHTER_TRAINING = 1UL << 42,
GUNNER_TRAINING = 1UL << 43,
TECHTER_TRAINING = 1UL << 44,
BRAVER_TRAINING = 1UL << 45,
BOUNCER_TRAINING = 1UL << 46,
SUMMONER_TRAINING = 1UL << 47,
AUTO_ACCEPT = 1UL << 48,
RIDROID = 1UL << 49,
NET_CAFE_AGP = 1UL << 50,
BATTLE_BROKEN = 1UL << 51,
BUSTER_DEBUG = 1UL << 52,
POKA12 = 1UL << 53,
UNK2 = 1UL << 54,
UNK3 = 1UL << 55,
BUSTER = 1UL << 56,
HERO_TRAINING = 1UL << 57,
AMPLIFIED = 1UL << 58,
UNK4 = 1UL << 59,
UNK5 = 1UL << 60,
DARK_BLAST_TRAINING = 1UL << 61,
ENDLESS = 1UL << 62,
UNK6 = 1UL << 63,
BLANK4 = 1UL << 64,
PHANTOM_TRAINING = 1UL << 65,
AIS_TRAINING = 1UL << 66,
UNK7 = 1UL << 67,
DAMAGE_CALC = 1UL << 68,
ETOILE_TRAINING = 1UL << 69,
DIVIDE = 1UL << 70,
STARS1 = 1UL << 71,
STARS2 = 1UL << 72,
STARS3 = 1UL << 73,
STARS4 = 1UL << 74,
STARS5 = 1UL << 75,
STARS6 = 1UL << 76,
UNK8 = 1UL << 77
}
public enum QuestEstimatedTime : byte
{
Short = 1,
Medium,
Long
}
[Flags]
public enum QuestDifficultyType : byte
{
NORMAL = 1 << 0, // 1 0x01 (0000 0001)
HARD = 1 << 1, // 2 0x02 (0000 0010)
VERY_HARD = 1 << 2, // 4 0x04 (0000 0100)
SUPER_HARD = 1 << 3, // 8 0x08 (0000 1000)
EX_HARD = 1 << 4, // 16 0x10 (0001 0000)
ULTRA_HARD = 1 << 5, // 32 0x20 (0010 0000)
Dummy2 = 1 << 6, // 64 0x40 (0100 0000)
Dummy3 = 1 << 7 // 128 0x80 (1000 0000)
}
public enum QuestPartyType : byte
{
Solo,
/// Only one party can join (up to 4 players).
SingleParty,
/// Multiple parties can join (up to 12 players).
MultiParty,
}
[Flags]
public enum QuestBitfield1 : ushort
{
MatterObjectiveQuest = 0x0001,
ClientOrderOnQuest = 0x0008,
NewQuest = 0x0100,
ClientOrder = 0x0800,
UnknownLevel = 0x1000
}
}

View File

@ -1,207 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
/// <summary>
/// Revealed minimap regions
/// </summary>
public class RevealedRegions
{
// 内部实现:
// 该结构体由一个大小为[10]的字节数组组成,表示一个 8x10 的已揭示区域网格
// (即 10 列 8 行。每一位bit表示一个区域是否已揭示。
// 位是从右到左索引的(即从 LSB 到 MSB字节是从左到右索引的arr[0]表示区域A1-8
// 获取该数组中的某个值可以通过如下方式:
//
// 示例代码:
// bool GetBit(byte[] arr, int row, int col)
// {
// // 确保行和列的范围有效
// if (row < 0 || row >= 8 || col < 0 || col >= 10)
// throw new ArgumentOutOfRangeException();
//
// // 计算对应位置的位偏移
// int offset = row * 10 + col;
// int byteOffset = offset / 8;
// int bitOffset = offset % 8;
//
// // 获取相应的位值
// return (arr[byteOffset] >> bitOffset & 1) == 1;
// }
// 使用一个长度为 10 的字节数组,表示 8 行 10 列的区域位图。
// Lsb0 表示最低有效位在字节的最右侧。
public byte[] zones = new byte[10];
// 默认构造函数
public RevealedRegions()
{
}
// 构造函数 - 用指定数据初始化
public RevealedRegions(byte[] data)
{
// 如果传入的 unk2 长度小于 10则填充剩余部分为 0
if (data.Length <= 10)
{
zones = data.Concat(new byte[10 - data.Length]).ToArray();
}
else
{
// 如果传入的 unk2 长度大于 10则截取前 10 字节
zones = data.Take(10).ToArray();
}
}
/// <summary>
/// 获取指定行列位置的位值
/// </summary>
/// <param name="row">行索引0 到 7</param>
/// <param name="col">列索引0 到 9</param>
/// <returns>该位置的位值true = 已揭示false = 未揭示)</returns>
public bool GetBit(int row, int col)
{
// 确保行列值在有效范围内
if (row < 0 || row >= 8 || col < 0 || col >= 10)
{
throw new ArgumentOutOfRangeException();
}
// 计算对应位置的偏移
int offset = row * 10 + col;
int byteOffset = offset / 8;
int bitOffset = offset % 8;
// 获取并返回该位置的位值
return (zones[byteOffset] >> bitOffset & 1) == 1;
}
/// <summary>
/// 设置指定行列位置的位值
/// </summary>
/// <param name="row">行索引0 到 7</param>
/// <param name="col">列索引0 到 9</param>
/// <param name="value">设置的值true = 设置为已揭示false = 设置为未揭示)</param>
public void SetBit(int row, int col, bool value)
{
// 确保行列值在有效范围内
if (row < 0 || row >= 8 || col < 0 || col >= 10)
{
throw new ArgumentOutOfRangeException();
}
// 计算对应位置的偏移
int offset = row * 10 + col;
int byteOffset = offset / 8;
int bitOffset = offset % 8;
// 设置对应位置的位值
if (value)
{
// 设置为1
zones[byteOffset] |= (byte)(1 << bitOffset);
}
else
{
// 设置为0
zones[byteOffset] &= (byte)~(1 << bitOffset);
}
}
// 读取数据
public static RevealedRegions Read(BinaryReader reader)
{
try
{
byte[] data = reader.ReadBytes(10);
if (data.Length != 10)
throw new InvalidDataException("Failed to read 10 bytes for zones data.");
return new RevealedRegions(data);
}
catch (Exception ex)
{
throw new PacketError("RevealedRegions", "zone_bits", ex);
}
}
// 写入数据
public void Write(BinaryWriter writer)
{
try
{
writer.Write(zones);
}
catch (Exception ex)
{
throw new PacketError("RevealedRegions", "zone_bits", ex);
}
}
// 索引操作,获取指定行的 10 位数据
public BitSlice GetRow(int index)
{
if (index < 0 || index >= 8)
throw new ArgumentOutOfRangeException();
byte[] row = new byte[10];
Array.Copy(zones, index * 10, row, 0, 10);
return new BitSlice(row);
}
// 用于调试输出
public override string ToString()
{
StringBuilder sb = new StringBuilder();
sb.Append("[");
for (int i = 0; i < 8; i++)
{
sb.Append(GetRow(i).ToString());
if (i < 7) sb.Append(", ");
}
sb.Append("]");
return sb.ToString();
}
// Helper class to represent a slice of bits
public class BitSlice
{
private byte[] data;
public BitSlice(byte[] data)
{
this.data = data;
}
public override string ToString()
{
StringBuilder sb = new StringBuilder();
foreach (byte b in data)
{
sb.Append(Convert.ToString(b, 2).PadLeft(8, '0'));
}
return sb.ToString();
}
}
// Error handling class for packet-related errors
public class PacketError : Exception
{
public string PacketName { get; }
public string FieldName { get; }
public new Exception InnerException { get; }
public PacketError(string packetName, string fieldName, Exception innerException)
: base($"Error in packet '{packetName}', field '{fieldName}'", innerException)
{
PacketName = packetName;
FieldName = fieldName;
InnerException = innerException;
}
}
}
}

View File

@ -1,32 +0,0 @@
using System;
using System.Runtime.InteropServices;
namespace PSO2SERVER.Models
{
public enum ShipStatus : ushort
{
Unknown = 0,
Online,
Busy,
Full,
Offline,
Undefined = 0xFFFF
}
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
public unsafe struct ShipEntry
{
public UInt32 number;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
public string name;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
public byte[] ip;
public UInt32 zero;
public ShipStatus status;
public UInt16 order;
public UInt32 unknown;
}
}

View File

@ -1,290 +0,0 @@
using System;
using System.Collections.Generic;
using Newtonsoft.Json;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public enum AttackType
{
Mel = 0,
Rng = 1,
Tec = 2
}
public class LevelStats
{
public ulong ExpToNext { get; set; } = 0;
public float Hp { get; set; } = 0;
public float Pp { get; set; } = 0;
public float MelPow { get; set; } = 0;
public float RngPow { get; set; } = 0;
public float TecPow { get; set; } = 0;
public float Dex { get; set; } = 0;
public float MelDef { get; set; } = 0;
public float RngDef { get; set; } = 0;
public float TecDef { get; set; } = 0;
}
public class StatMultipliers
{
public sbyte Hp { get; set; } = 0;
public sbyte MelPow { get; set; } = 0;
public sbyte RngPow { get; set; } = 0;
public sbyte TecPow { get; set; } = 0;
public sbyte Dex { get; set; } = 0;
public sbyte MelDef { get; set; } = 0;
public sbyte RngDef { get; set; } = 0;
public sbyte TecDef { get; set; } = 0;
}
public class ClassStatsStored
{
public ClassType Class { get; set; }
public List<LevelStats> Stats { get; set; } = new List<LevelStats>();
}
public class PlayerStats
{
public List<List<LevelStats>> Stats { get; set; } = new List<List<LevelStats>>();
public List<StatMultipliers> Modifiers { get; set; } = new List<StatMultipliers>();
}
public class RaceModifierStored
{
public StatMultipliers HumanMale { get; set; } = new StatMultipliers();
public StatMultipliers HumanFemale { get; set; } = new StatMultipliers();
public StatMultipliers NewmanMale { get; set; } = new StatMultipliers();
public StatMultipliers NewmanFemale { get; set; } = new StatMultipliers();
public StatMultipliers CastMale { get; set; } = new StatMultipliers();
public StatMultipliers CastFemale { get; set; } = new StatMultipliers();
public StatMultipliers DeumanMale { get; set; } = new StatMultipliers();
public StatMultipliers DeumanFemale { get; set; } = new StatMultipliers();
}
public class EnemyLevelBaseStats
{
public uint Level { get; set; } = 0;
public float Exp { get; set; } = 0;
public float Hp { get; set; } = 0;
public float MaxMelDmg { get; set; } = 0;
public float MinMelDmg { get; set; } = 0;
public float MaxRngDmg { get; set; } = 0;
public float MinRngDmg { get; set; } = 0;
public float MaxTecDmg { get; set; } = 0;
public float MinTecDmg { get; set; } = 0;
public float MelDef { get; set; } = 0;
public float RngDef { get; set; } = 0;
public float TecDef { get; set; } = 0;
public float Dex { get; set; } = 0;
}
public class EnemyHitbox
{
public string Name { get; set; } = string.Empty;
public uint HitboxId { get; set; } = 0;
public float DamageMul { get; set; } = 1.0f;
public float MelMul { get; set; } = 1.0f;
public float RngMul { get; set; } = 1.0f;
public float TecMul { get; set; } = 1.0f;
public float FireMul { get; set; } = 1.0f;
public float IceMul { get; set; } = 1.0f;
public float ThunderMul { get; set; } = 1.0f;
public float WindMul { get; set; } = 1.0f;
public float LightMul { get; set; } = 1.0f;
public float DarkMul { get; set; } = 1.0f;
}
public class EnemyBaseStats
{
public List<EnemyLevelBaseStats> Levels { get; set; } = new List<EnemyLevelBaseStats>();
}
public class EnemyStats
{
public List<EnemyLevelBaseStats> Levels { get; set; } = new List<EnemyLevelBaseStats>();
public List<EnemyHitbox> Hitboxes { get; set; } = new List<EnemyHitbox>();
}
public class NamedEnemyStats
{
public string Name { get; set; } = string.Empty;
public EnemyStats Stats { get; set; } = new EnemyStats();
}
public class AllEnemyStats
{
public EnemyBaseStats Base { get; set; } = new EnemyBaseStats();
public Dictionary<string, EnemyStats> Enemies { get; set; } = new Dictionary<string, EnemyStats>();
}
public class AttackStatsReadable
{
public string AttackName { get; set; } = string.Empty;
public string DamageName { get; set; } = string.Empty;
public AttackType AttackType { get; set; } = AttackType.Mel;
public AttackType DefenseType { get; set; } = AttackType.Mel;
public DamageTypeReadable Damage { get; set; } = new DamageTypeReadable();
}
public class AttackStats
{
public uint AttackId { get; set; } = 0;
public uint DamageId { get; set; } = 0;
public AttackType AttackType { get; set; } = AttackType.Mel;
public AttackType DefenseType { get; set; } = AttackType.Mel;
public DamageType Damage { get; set; } = new DamageType();
}
public class DamageTypeReadable
{
public static DamageTypeReadable Default => new DamageTypeReadable { Mul = 1.0f };
public float Mul { get; set; } = 1.0f;
public string Name { get; set; } = string.Empty;
}
public class DamageType
{
public float Mul { get; set; } = 1.0f;
public Tuple<uint, float> Pa { get; set; } = new Tuple<uint, float>(0, 1.0f);
public DamageType() { }
public DamageType(float mul)
{
Mul = mul;
}
public DamageType(Tuple<uint, float> pa)
{
Pa = pa;
}
}
public class DamageTypeConverter
{
public static DamageType ConvertToDamageType(DamageTypeReadable value)
{
return new DamageType(value.Mul);
}
}
public struct EXPReceiver
{
// 玩家获得经验的对象
public ObjectHeader Object; // 如果 ObjectHeader 是结构体,则可以这样使用;如果是类,需要修改或替代
public byte Unk1; // 未知字段 1
public byte Unk2; // 未知字段 2
public byte[] Unk3; // 6 字节的数组(未知数据),这部分需要额外处理
// 主职业获得的经验
public ulong Gained;
// 主职业的总经验
public ulong Total;
// 主职业的新子等级(?
public ushort Level2;
// 主职业的新等级
public ushort Level;
// 主职业
public ClassType Class; // 假设 ClassType 是一个枚举类型或者结构体
public byte[] Pad1; // 填充字节数组3 字节),需要额外处理
// 副职业获得的经验
public ulong GainedSub;
// 副职业的总经验
public ulong TotalSub;
// 副职业的新子等级(?
public ushort Level2Sub;
// 副职业的新等级
public ushort LevelSub;
// 副职业
public ClassType Subclass; // 假设 ClassType 是一个枚举类型或者结构体
public byte[] Pad2; // 填充字节数组3 字节),需要额外处理
// 结构体构造函数,用于初始化数组和其他默认值
public EXPReceiver(ObjectHeader objectHeader, byte unk1, byte unk2, byte[] unk3, ulong gained, ulong total,
ushort level2, ushort level, ClassType classType, byte[] pad1, ulong gainedSub,
ulong totalSub, ushort level2Sub, ushort levelSub, ClassType subclass, byte[] pad2)
{
Object = objectHeader;
Unk1 = unk1;
Unk2 = unk2;
Unk3 = unk3 ?? new byte[6]; // 默认值为6字节数组
Gained = gained;
Total = total;
Level2 = level2;
Level = level;
Class = classType;
Pad1 = pad1 ?? new byte[3]; // 默认值为3字节数组
GainedSub = gainedSub;
TotalSub = totalSub;
Level2Sub = level2Sub;
LevelSub = levelSub;
Subclass = subclass;
Pad2 = pad2 ?? new byte[3]; // 默认值为3字节数组
}
}
public enum UserState
{
LoggingIn, // User is logging in, nothing is set up.
NewUsername, // User is logged in, but no username was set, only user id is set.
CharacterSelect, // User is logged in, account stuff is set up, but no character info.
PreInGame, // User has selected the character, but map and party aren't set up yet.
InGame // User is in the game, character, map, party are set up.
}
public static class UserStateExtensions
{
// 处理不同状态的逻辑
public static string GetStateMessage(this UserState state)
{
switch (state)
{
case UserState.LoggingIn:
return "User is logging in...";
case UserState.NewUsername:
return "User needs to set a username.";
case UserState.CharacterSelect:
return "User is selecting a character.";
case UserState.PreInGame:
return "User is preparing to enter the game.";
case UserState.InGame:
return "User is in the game!";
default:
return "Unknown state.";
}
}
// 比较两个状态,判断当前状态是否在另一个状态之前
public static bool IsBefore(this UserState currentState, UserState otherState)
{
return currentState < otherState;
}
// 比较两个状态,判断当前状态是否在另一个状态之后
public static bool IsAfter(this UserState currentState, UserState otherState)
{
return currentState > otherState;
}
public static void ToLogger(this UserState state)
{
Logger.Write(GetStateMessage(state));
}
}
}

View File

@ -1,100 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Models
{
public struct Duration : IEquatable<Duration>, IComparable<Duration>
{
// The number of seconds in the duration
public long Seconds { get; }
// The number of nanoseconds, should always be between 0 and 999,999,999
public int Nanoseconds { get; }
// Constructor to initialize a Duration with seconds and nanoseconds
public Duration(long seconds, int nanoseconds)
{
if (nanoseconds < 0 || nanoseconds >= 1_000_000_000)
{
throw new ArgumentOutOfRangeException(nameof(nanoseconds), "Nanoseconds must be between 0 and 999,999,999.");
}
Seconds = seconds;
Nanoseconds = nanoseconds;
}
// A default zero-length duration
public static Duration Zero => new Duration(0, 0);
// Adds two durations together
public static Duration operator +(Duration left, Duration right)
{
long totalSeconds = left.Seconds + right.Seconds;
int totalNanoseconds = left.Nanoseconds + right.Nanoseconds;
// Handle carry over for nanoseconds
if (totalNanoseconds >= 1_000_000_000)
{
totalSeconds++;
totalNanoseconds -= 1_000_000_000;
}
return new Duration(totalSeconds, totalNanoseconds);
}
// Subtracts one duration from another
public static Duration operator -(Duration left, Duration right)
{
long totalSeconds = left.Seconds - right.Seconds;
int totalNanoseconds = left.Nanoseconds - right.Nanoseconds;
// Handle borrow for nanoseconds
if (totalNanoseconds < 0)
{
totalSeconds--;
totalNanoseconds += 1_000_000_000;
}
return new Duration(totalSeconds, totalNanoseconds);
}
// Compare two durations
public int CompareTo(Duration other)
{
if (Seconds < other.Seconds) return -1;
if (Seconds > other.Seconds) return 1;
return Nanoseconds.CompareTo(other.Nanoseconds);
}
// Check if two durations are equal
public bool Equals(Duration other)
{
return Seconds == other.Seconds && Nanoseconds == other.Nanoseconds;
}
// Override ToString() for custom string representation
public override string ToString()
{
return $"{Seconds}s {Nanoseconds}ns";
}
// Static method to create a Duration from milliseconds
public static Duration FromMilliseconds(long milliseconds)
{
long seconds = milliseconds / 1000;
int nanoseconds = (int)((milliseconds % 1000) * 1_000_000);
return new Duration(seconds, nanoseconds);
}
// Static method to create a Duration from seconds and nanoseconds
public static Duration FromSeconds(long seconds, int nanoseconds)
{
return new Duration(seconds, nanoseconds);
}
}
}

View File

@ -1,76 +0,0 @@
using System;
public struct Ipv4Addr : IEquatable<Ipv4Addr>
{
private byte[] octets;
// 构造函数
public Ipv4Addr(byte[] octets)
{
if (octets == null || octets.Length != 4)
throw new ArgumentException("IPv4 address must have exactly 4 octets.");
this.octets = octets;
}
// 允许访问 octets 数组
public byte[] Octets => octets;
// 重写 Equals 方法
public override bool Equals(object obj)
{
return obj is Ipv4Addr other && Equals(other);
}
// 实现 IEquatable<T> 接口
public bool Equals(Ipv4Addr other)
{
for (int i = 0; i < 4; i++)
{
if (this.octets[i] != other.octets[i])
return false;
}
return true;
}
// 重写 GetHashCode 方法
public override int GetHashCode()
{
int hash = 17;
foreach (var octet in octets)
{
hash = hash * 31 + octet;
}
return hash;
}
// 提供一个用于显示的字符串格式
public override string ToString()
{
return string.Join(".", octets);
}
// 静态方法来解析字符串形式的 IPv4 地址
public static Ipv4Addr Parse(string ipString)
{
var parts = ipString.Split('.');
if (parts.Length != 4)
throw new ArgumentException("Invalid IPv4 address format.");
var octets = new byte[4];
for (int i = 0; i < 4; i++)
{
octets[i] = byte.Parse(parts[i]);
}
return new Ipv4Addr(octets);
}
// 重载 == 和 != 操作符
public static bool operator ==(Ipv4Addr left, Ipv4Addr right)
{
return left.Equals(right);
}
public static bool operator !=(Ipv4Addr left, Ipv4Addr right)
{
return !(left == right);
}
}

View File

@ -1,32 +0,0 @@
using System;
using System.Net;
using System.Net.Sockets;
public class PortChecker
{
public static bool IsPortListening(int port)
{
TcpListener listener = null;
try
{
listener = new TcpListener(IPAddress.Any, port);
listener.Start();
return false; // Port is free
}
catch (SocketException)
{
return true; // Port is in use
}
finally
{
listener?.Stop(); // Ensure listener is stopped
}
}
public static void Main()
{
int portToCheck = 8080; // Replace with the port you want to check
bool isListening = IsPortListening(portToCheck);
Console.WriteLine($"Port {portToCheck} is {(isListening ? "in use" : "free")}");
}
}

View File

@ -1,109 +0,0 @@
using System.Net.Sockets;
namespace PSO2SERVER.Network
{
public class SocketClient
{
public delegate void ConnectionLostDelegate();
public delegate void DataReceivedDelegate(byte[] data, int size);
private readonly byte[] _readBuffer, _writeBuffer;
private readonly SocketServer _server;
private int _writePosition = 0;
public SocketClient()
{
_server = new SocketServer(0);
Socket = new TcpClient();
_readBuffer = new byte[1024 * 16];
_writeBuffer = new byte[1024 * 1024]; // too high? too low? not sure
}
public SocketClient(SocketServer server, TcpClient socket)
{
_server = server;
Socket = socket;
_readBuffer = new byte[1024 * 16];
_writeBuffer = new byte[1024 * 1024]; // too high? too low? not sure
}
public TcpClient Socket { get; private set; }
public event DataReceivedDelegate DataReceived;
public event ConnectionLostDelegate ConnectionLost;
public bool NeedsToWrite { get { return (_writePosition > 0); } }
public bool OnReadable()
{
try
{
var read = Socket.Client.Receive(_readBuffer);
if (read == 0)
{
// Connection failed, presumably
ConnectionLost();
_server.NotifyConnectionClosed(this);
return false;
}
DataReceived(_readBuffer, read);
return true;
}
catch (SocketException)
{
ConnectionLost();
_server.NotifyConnectionClosed(this);
return false;
}
}
public bool OnWritable()
{
try
{
var write = Socket.Client.Send(_writeBuffer, 0, _writePosition, SocketFlags.None);
if (write == 0)
{
// Connection failed, presumably
ConnectionLost();
_server.NotifyConnectionClosed(this);
return false;
}
System.Array.Copy(_writeBuffer, write, _writeBuffer, 0, _writePosition - write);
_writePosition -= write;
return true;
}
catch (SocketException)
{
ConnectionLost();
_server.NotifyConnectionClosed(this);
return false;
}
}
public void Write(byte[] blob)
{
if ((_writePosition + blob.Length) > _writeBuffer.Length)
{
// Buffer exceeded!
throw new System.Exception("too much data in write queue");
}
System.Array.Copy(blob, 0, _writeBuffer, _writePosition, blob.Length);
_writePosition += blob.Length;
}
public void Close()
{
ConnectionLost();
_server.NotifyConnectionClosed(this);
Socket.Close();
}
}
}

View File

@ -1,106 +0,0 @@
using System;
using System.Collections.Generic;
using System.Net;
using System.Net.Sockets;
namespace PSO2SERVER.Network
{
public class SocketServer
{
public delegate void NewClientDelegate(SocketClient client);
private readonly List<SocketClient> _clients = new List<SocketClient>();
private readonly TcpListener _listener;
private readonly List<Socket> _readableSockets = new List<Socket>();
private readonly List<Socket> _writableSockets = new List<Socket>();
private readonly Dictionary<Socket, SocketClient> _socketMap = new Dictionary<Socket, SocketClient>();
private int _port;
public SocketServer(int port)
{
_port = port;
_listener = new TcpListener(IPAddress.Any, port);
_listener.Start();
}
public IList<SocketClient> Clients
{
get { return _clients.AsReadOnly(); }
}
public event NewClientDelegate NewClient;
public void Run()
{
try
{
// Compile a list of possibly-readable sockets
_readableSockets.Clear();
_readableSockets.Add(_listener.Server);
_writableSockets.Clear();
foreach (var client in _clients)
{
_readableSockets.Add(client.Socket.Client);
if (client.NeedsToWrite)
_writableSockets.Add(client.Socket.Client);
}
Socket.Select(_readableSockets, _writableSockets, null, 1000000);
foreach (var socket in _readableSockets)
{
if (socket == _listener.Server)
{
var c = new SocketClient(this, _listener.AcceptTcpClient());
Logger.WriteInternal("[HI!] 新的客户端接入!");
_clients.Add(c);
_socketMap.Add(c.Socket.Client, c);
NewClient(c);
// 获取接入客户端的 IP 和端口
var clientEndPoint = c.Socket.Client.RemoteEndPoint as IPEndPoint;
if (clientEndPoint != null)
{
string clientIp = clientEndPoint.Address.ToString();
int clientPort = clientEndPoint.Port;
// 输出客户端的 IP 和端口
Logger.WriteInternal($"[HI!] 新的客户端接入! IP: {clientIp}, 端口: {clientPort}");
}
else
{
// New connection
Logger.WriteInternal("[HI!] 新的客户端接入!");
}
}
else
{
// Readable data
if (socket.Connected)
_socketMap[socket].OnReadable();
}
}
foreach (var socket in _writableSockets)
if (socket.Connected)
_socketMap[socket].OnWritable();
}
catch (Exception ex)
{
Logger.WriteException("A socket error occurred", ex);
}
}
internal void NotifyConnectionClosed(SocketClient client)
{
Logger.Write("断开客户端连接.");
_socketMap.Remove(client.Socket.Client);
_clients.Remove(client);
}
}
}

View File

@ -1,145 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Newtonsoft.Json;
using PSO2SERVER.Database;
using PSO2SERVER.Models;
using PSO2SERVER.Zone;
namespace PSO2SERVER.Object
{
class ObjectManager
{
private static readonly ObjectManager instance = new ObjectManager();
private Dictionary<String, Dictionary<ulong, PSOObject>> zoneObjects = new Dictionary<string, Dictionary<ulong, PSOObject>>();
private Dictionary<ulong, PSOObject> allTheObjects = new Dictionary<ulong, PSOObject>();
private ObjectManager() { }
public static ObjectManager Instance
{
get
{
return instance;
}
}
public PSOObject[] GetObjectsForZone(string zone)
{
if (zone == "tpmap") // Return empty object array for an tp'd map for now (We spawn in a teleporter manually)
{
return new PSOObject[0];
}
if (!zoneObjects.ContainsKey(zone))
{
Dictionary<ulong, PSOObject> objects = new Dictionary<ulong, PSOObject>();
// Collect from db
using (var db = new ServerEf())
{
var dbObjects = from dbo in db.GameObjects
where dbo.ZoneName == zone
select dbo;
foreach(var dbObject in dbObjects)
{
var newObject = PSOObject.FromDBObject(dbObject);
objects.Add(newObject.Header.ID, newObject);
allTheObjects.Add(newObject.Header.ID, newObject);
//Logger.WriteInternal("[OBJ] 从数据库中载入对象 {0} 所属区域 {1}.", newObject.name, zone);
}
}
// Fallback
if (objects.Count < 1 && Directory.Exists("Resources\\objects\\" + zone))
{
Logger.WriteWarning("[OBJ] 数据库中没有为该区域定义的对象 {0}, 回退到文件系统!", zone);
var objectPaths = Directory.GetFiles("Resources\\objects\\" + zone);
Array.Sort(objectPaths);
foreach (var path in objectPaths)
{
if (Path.GetExtension(path) == ".bin")
{
var newObject = PSOObject.FromPacketBin(File.ReadAllBytes(path));
objects.Add(newObject.Header.ID, newObject);
allTheObjects.Add(newObject.Header.ID, newObject);
Logger.WriteInternal("[OBJ] BIN文件系统载入对象 ID {0} 名称 {1} 坐标: ({2}, {3}, {4})", newObject.Header.ID, newObject.Name, newObject.Position.PosX,
newObject.Position.PosY, newObject.Position.PosZ);
}
else if (Path.GetExtension(path) == ".json")
{
var newObject = JsonConvert.DeserializeObject<PSOObject>(File.ReadAllText(path));
objects.Add(newObject.Header.ID, newObject);
allTheObjects.Add(newObject.Header.ID, newObject);
Logger.WriteInternal("[OBJ] JSON文件系统载入对象 ID {0} 名称 {1} 坐标: ({2}, {3}, {4})", newObject.Header.ID, newObject.Name, newObject.Position.PosX,
newObject.Position.PosY, newObject.Position.PosZ);
}
}
}
zoneObjects.Add(zone, objects);
}
return zoneObjects[zone].Values.ToArray();
}
internal PSONPC[] GetNpcSForZone(string zone)
{
List<PSONPC> npcs = new List<PSONPC>();
using (var db = new ServerEf())
{
var dbNpcs = from n in db.NPCs
where n.ZoneName == zone && n.is_active == 1
select n;
foreach (NPC npc in dbNpcs)
{
PSONPC dNpc = new PSONPC();
dNpc.Header = new ObjectHeader((uint)npc.EntityID, ObjectType.Object);
dNpc.Position = new PSOLocation(npc.RotX, npc.RotY, npc.RotZ, npc.RotW, npc.PosX, npc.PosY, npc.PosZ);
dNpc.Name = npc.NPCName;
npcs.Add(dNpc);
if (!zoneObjects[zone].ContainsKey(dNpc.Header.ID))
{
zoneObjects[zone].Add(dNpc.Header.ID, dNpc);
}
if (!allTheObjects.ContainsKey(dNpc.Header.ID))
allTheObjects.Add(dNpc.Header.ID, dNpc);
}
}
return npcs.ToArray();
}
internal PSOObject getObjectByID(string zone, uint ID)
{
//FIXME: This has been commented out because we were getting object errors with possible shared objects? That or it was just object 1 which is an edge case.
//if(!zoneObjects.ContainsKey(zone) || !zoneObjects[zone].ContainsKey(ID))
//{
// throw new Exception(String.Format("Object ID {0} does not exist in {1}!", ID, zone));
//}
//return zoneObjects[zone][ID];
return getObjectByID(ID);
}
internal PSOObject getObjectByID(uint ID)
{
if (!allTheObjects.ContainsKey(ID))
{
Logger.WriteWarning("[OBJ] 客户端请求的对象 {0} 服务端未解析. 等待分析.", ID);
return new PSOObject() { Header = new ObjectHeader(ID, ObjectType.Object), Name = "Unknown" };
}
return allTheObjects[ID];
}
}
}

View File

@ -1,217 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PSO2SERVER.Party
{
public class Party
{
public string Name;
public Client Leader;
public List<Client> members;
public PartyQuest currentQuest;
public PartySettingsPacket partySetting;
public string questname;
public List<Tuple<int, PartyColor>> PartyColors = new List<Tuple<int, PartyColor>>();
public Party(string name, Client leader)
{
Name = name;
Leader = leader;
members = new List<Client>();
currentQuest = new PartyQuest();
partySetting = new PartySettingsPacket();
questname = string.Empty;
addClientToParty(Leader);
}
public void addClientToParty(Client c)
{
if (members.Count < 1)
{
c.SendPacket(new PartyInitPacket(new List<Client> { c }));
}
else
{
// ???
}
members.Add(c);
c.currentParty = this;
}
public void removeClientFromParty(Client c)
{
if (!members.Contains(c))
{
Logger.WriteWarning("[PTY] 客户端 {0} 尝试离开 {1}, 但他不在队伍 {1} 中!", c._account.Username, Name);
return;
}
members.Remove(c);
//TODO do stuff like send the "remove from party" packet.
}
public bool hasClientInParty(Client c)
{
return members.Contains(c);
}
public Client getPartyHost()
{
return Leader;
}
public int getSize()
{
return members.Count;
}
public List<Client> getMembers()
{
return members;
}
// 方法:根据 id 添加颜色
public PartyColor AddColor(int id)
{
// 颜色预定义集合
PartyColor[] colorOptions = new PartyColor[] { PartyColor.Red, PartyColor.Blue, PartyColor.Green, PartyColor.Yellow };
foreach (var color in colorOptions)
{
// 检查 PartyColors 列表中是否已有该颜色
if (this.PartyColors.Any(c => c.Item2 == color))
{
continue; // 如果已存在,跳过
}
// 向 PartyColors 列表中添加新的颜色
this.PartyColors.Add(new Tuple<int, PartyColor>(id, color));
return color; // 返回添加的颜色
}
// 如果所有颜色都已存在,返回 PartyColor.Red
return PartyColor.Red;
}
// 根据 ID 获取颜色
public PartyColor GetColor(int id)
{
// 查找与给定 id 匹配的颜色
var foundColor = this.PartyColors
.FirstOrDefault(c => c.Item1 == id); // 获取第一个匹配的项
// 如果找到了匹配的颜色,返回颜色,否则返回默认颜色
return foundColor.Equals(default(Tuple<int, PartyColor>)) ? PartyColor.Red : foundColor.Item2;
}
}
public enum PartyColor : byte
{
Red,
Green,
Yellow,
Blue,
}
[Serializable]
public unsafe struct PartyEntry
{
/// 玩家对象的头部信息包含ID、类型、区域ID等
public ObjectHeader id;
/// 玩家昵称。
public string nickname;
/// 玩家角色名称。
public string char_name;
/// 主职业的等级。
public byte level;
/// 副职业的等级。
public byte sublevel;
/// 玩家主职业类型(`ClassType` 枚举类型)。
public ClassType mainClass;
/// 玩家副职业类型(`ClassType` 枚举类型)。
public ClassType subClass;
/// 玩家在队伍中的颜色标识。
public PartyColor color;
/// 7字节的未知数据通常是填充或者未定义用途
public fixed byte unk1[7];
/// 一个未知的 uint 类型数据,具体用途不明。
public uint unk2;
/// 玩家HP体力的三个数值。具体原因为什么有三个值目前不明。
public fixed uint hp[3];
/// 玩家所在的地图ID。
public ushort mapid;
/// 另一个未知的 ushort 类型数据。
public ushort unk3;
/// 12字节的未知数据。
public fixed byte unk4[0x0C];
/// 3个未知的 uint 数组数据。
public fixed uint unk5[0x03];
/// 未知的字符串,可能是其他信息(如玩家状态或其他描述信息)。
public string unk6;
// PSO VITAPlayStation Vita相关字段已注释掉可能与Vita版本相关。
public string unk10;
// 玩家角色的 ASCII 字符串相关数据,已注释掉。
public string unk7; // ASCII 字符串
// 玩家语言设置(`ShortLanguage` 枚举类型),已注释掉。
public ShortLanguage lang;
// 3字节的未知数据已注释掉。
public fixed byte unk9[3];
}
[Serializable]
public unsafe struct PartyInfo
{
public fixed byte unk1[0x0C];
public ObjectHeader party_object; // Assuming ObjectHeader is another class
public string name; // Name of the party 0xE7E8, 0xFF
public fixed byte unk2[0x09];
public fixed byte unk3[0x03];
public uint unk4; // 32-bit unsigned integer
public uint invite_time; // Time when the player was invited
public uint unk6; // 32-bit unsigned integer
}
[Flags]
public enum PartyFlags : byte
{
/// Is the party only for friends.
FRIENDS_ONLY = 1 << 0, // 1
/// Is the party only for alliance members.
ALLIANCE_ONLY = 1 << 1, // 2
/// Limit multiplayer requests from other parties.
LIMIT_OTHERS = 1 << 2, // 4
/// Is the party only for a single run.
SINGLE_RUN = 1 << 3, // 8
/// Is the party actively looking for members.
OPEN = 1 << 4, // 16
/// Is the party voice chat focused.
VC_FOCUS = 1 << 6, // 64
// This flag would represent an invalid state (all bits set).
_ = byte.MaxValue, // 255 (all flags set)
}
}

View File

@ -1,79 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PSO2SERVER.Party
{
public class PartyManager
{
public static readonly PartyManager instance = new PartyManager();
public Dictionary<Party, string> parties = new Dictionary<Party, string>(); // Key: Party, Value: name? (for now)
public static PartyManager Instance
{
get
{
return instance;
}
}
public PartyManager()
{
Logger.WriteInternal("[PTY] PartyManager 初始化完成.");
}
public Party GetCurrentPartyForClient(Client c)
{
foreach(Party p in parties.Keys) //TODO: Filter this on a per-block basis?
{
if (p.hasClientInParty(c))
return p;
}
return null;
}
public void CreateNewParty(Client c)
{
if (GetCurrentPartyForClient(c) != null)
return; // For now
parties.Add(new Party(c._account.Username, c), c._account.Username);
}
public void AddPlayerToParty(Client c, Party p)
{
if (!parties.ContainsKey(p))
return;
if (p.getSize() >= 4) // For now
return;
p.addClientToParty(c);
}
public void RemovePlayerToParty(Client c, Party p)
{
if (!parties.ContainsKey(p))
return;
//TODO: Later just transfer owner like the real servers.
if (c == p.getPartyHost())
{
foreach(Client cl in p.getMembers())
{
p.removeClientFromParty(cl);
}
parties.Remove(p);
}
else
{
p.removeClientFromParty(c);
}
}
}
}

View File

@ -1,288 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Protocol.Handlers;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Reflection;
using System.Security.Cryptography;
using System.Threading;
using System.Threading.Tasks;
namespace PSO2SERVER
{
internal class ServerApp
{
public static ServerApp Instance { get; private set; }
// Will be using these around the app later [KeyPhact]
public const string ServerName = "梦幻之星OL2 服务端";
public const string ServerShortName = "PSO2";
public const string ServerAuthor = "Sancaros (https://github.com/Sancaros/PSO2SERVER)";
public const string ServerCopyright = "(C) 2024 Sancaros.";
public const string ServerLicense = "All licenced under AGPL.";
public const string ServerVersion = "v0.1.4";
public const string ServerVersionName = "Sancaros";
public const int ServerShipProtNums = 10;
public const int ServerShipProt = 12000;
public const int ServerShipListProtNums_JP = 10;
public const int ServerShipListProt_JP = 12099;
public const int ServerShipListProtNums_NA = 10;
public const int ServerShipListProt_NA = 13001;
public const string ServerBlockBalanceName = "Test";
public const int ServerBlockBalanceProt = 12205;
public const string ServerSettingsKey = "Resources\\settings.txt";
public const string ServerMemoryPacket = "Resources\\setMemoryPacket.bin"; //TODO
// 密钥BLOB格式
public const string ServerPrivateKeyBlob = "key\\privateKey.blob";
public const string ServerPublicKeyBlob = "key\\publicKey.blob";
public const string ServerSEGAKeyBlob = "key\\SEGAKey.blob";
// 密钥PEM格式 来自Schthack
public const string ServerPrivatePem = "key\\privateKey.pem";
public const string ServerSEGAPem = "key\\SEGAKey.pem";
public static IPAddress BindAddress = IPAddress.Parse("127.0.0.1");
public static Config Config;
public static ConsoleSystem ConsoleSystem;
public List<QueryServer> QueryServers = new List<QueryServer>();
public Server Server;
private static Assembly ResolveAssembly(object sender, ResolveEventArgs args)
{
string libDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "lib");
string assemblyPath = Path.Combine(libDir, new AssemblyName(args.Name).Name + ".dll");
if (File.Exists(assemblyPath))
{
return Assembly.LoadFrom(assemblyPath);
}
return null;
}
public static void Main(string[] args)
{
// 设置 AssemblyResolve 事件处理程序
AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(ResolveAssembly);
Config = new Config();
ConsoleSystem = new ConsoleSystem { Thread = new Thread(ConsoleSystem.StartThread) };
ConsoleSystem.Thread.Start();
// Setup function exit handlers to guarentee Exit() is run before closing
Console.CancelKeyPress += Exit;
AppDomain.CurrentDomain.ProcessExit += Exit;
try
{
for (var i = 0; i < args.Length; i++)
{
switch (args[i].ToLower())
{
case "-b":
case "--bind-address":
if (++i < args.Length)
{
var value = args[i];
try
{
if (IPAddress.TryParse(value, out IPAddress ipAddress))
{
// IP address is valid
BindAddress = ipAddress;
}
else
{
// Not an IP address, try resolving as a domain name
var addresses = Dns.GetHostAddresses(value);
if (addresses.Length > 0)
{
// Prefer IPv4 addresses over IPv6
ipAddress = addresses.FirstOrDefault(addr => addr.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork) ?? addresses[0];
BindAddress = ipAddress; // Use the first resolved IP address
}
else
{
Logger.WriteError($"No IP addresses found for domain: {value}");
}
}
}
catch (Exception ex)
{
Logger.WriteError($"Error resolving domain {value}: {ex.Message}");
}
}
break;
case "-s":
case "--size":
var splitArgs = args[++i].Split(',');
var width = int.Parse(splitArgs[0]);
var height = int.Parse(splitArgs[1]);
if (width < ConsoleSystem.Width)
{
Logger.WriteWarning("[ARG] Capping console width to {0} columns", ConsoleSystem.Width);
width = ConsoleSystem.Width;
}
if (height < ConsoleSystem.Height)
{
Logger.WriteWarning("[ARG] Capping console height to {0} rows", ConsoleSystem.Height);
height = ConsoleSystem.Height;
}
ConsoleSystem.SetSize(width, height);
break;
}
}
}
catch (Exception ex)
{
Logger.WriteException("An error has occurred while parsing command line parameters", ex);
}
// Check for settings.txt [AIDA]
if (!File.Exists(ServerSettingsKey))
{
// If it doesn't exist, throw an error and quit [AIDA]
Logger.WriteError("[ERR] 载入 {0} 文件错误. 按任意键退出.", ServerSettingsKey);
Console.ReadKey();
Environment.Exit(0);
}
Instance = new ServerApp();
Instance.GenerateKeys();
// Fix up startup message [KeyPhact]
Logger.WriteHeader();
Logger.Write(ServerName + " - " + ServerVersion + " (" + ServerVersionName + ")");
Logger.Write("作者 " + ServerAuthor);
//Logger.Write(ServerLicense);
Thread.Sleep(1000);
//System.Data.Entity.Database.SetInitializer(new System.Data.Entity.DropCreateDatabaseIfModelChanges<ServerEf>());
_ = Instance.StartAsync();
}
public void GenerateKeys()
{
// Process private key files
KeyLoader.ProcessKeyFiles(ServerPrivatePem, ServerPrivateKeyBlob, true, File.Exists(ServerPrivatePem));
// Process SEGA public key files
KeyLoader.ProcessKeyFiles(ServerSEGAPem, ServerSEGAKeyBlob, false, File.Exists(ServerSEGAPem));
// Process general RSA keys
using (RSACryptoServiceProvider rsa = new RSACryptoServiceProvider())
{
// Process private and public RSA keys
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPrivateKeyBlob, true);
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPublicKeyBlob, false);
}
}
public async Task StartAsync()
{
var startTime = DateTime.Now; // 记录启动开始时间
Server = new Server();
Server.LuaEngine.RegisterLogger();
//Server.LuaEngine.LusTest();
await InitializeConfigurationAsync();
await InitializeDatabaseAsync();
await InitializeQueryServers(QueryMode.AuthList, "认证", 11000, 3, 1000);
await InitializeQueryServers(QueryMode.AuthList, "认证", 13099, 1, 0);
await InitializeQueryServers(QueryMode.AuthList, "认证", ServerShipProt, ServerShipProtNums, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12100, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12193, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12194, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", ServerShipListProt_JP, ServerShipListProtNums_JP, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_NA", ServerShipListProt_NA, ServerShipListProtNums_NA, 1);
Logger.WriteInternal("服务器启动完成 " + DateTime.Now);
var endTime = DateTime.Now; // 记录启动结束时间
var duration = endTime - startTime; // 计算启动耗时
Logger.WriteInternal($"服务器启动耗时: {duration.TotalSeconds} 秒"); // 记录启动耗时
Server.Run();
}
private async Task InitializeConfigurationAsync()
{
await Task.Run(() =>
{
Config.Load();
PacketHandlers.LoadPacketHandlers();
});
}
private async Task InitializeDatabaseAsync()
{
try
{
Logger.WriteInternal("[DBC] 载入数据库...");
using (var db = new ServerEf())
{
await Task.Run(() =>
{
db.TestDatabaseConnection();
//db.SetupDB();
});
Logger.WriteInternal("[DBC] 数据库初始化完成。");
}
}
catch (Exception ex)
{
Logger.WriteException("[DBC] 数据库初始化异常", ex);
throw; // 重新抛出异常,或者根据需要处理它
}
}
public async Task InitializeQueryServers(QueryMode queryMode, string portname, int port, int portnums, int add)
{
await Task.Run(() =>
{
if (portnums <= 0)
portnums = 1;
if (portnums > 0)
{
for (var i = 0; i < portnums; i++)
{
QueryServers.Add(new QueryServer(queryMode, portname, port + (add * i)));
}
}
});
}
private static void Exit(object sender, EventArgs e)
{
// Save the configuration
Config.Save();
}
}
}

View File

@ -1,25 +0,0 @@
using System.Reflection;
// Information about this assembly is defined by the following attributes.
// Change them to the values specific to your project.
[assembly: AssemblyTitle("PSO2SERVER")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("")]
[assembly: AssemblyProduct("")]
[assembly: AssemblyCopyright("Sancaros")]
[assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")]
// The assembly version has the format "{Major}.{Minor}.{Build}.{Revision}".
// The form "{Major}.{Minor}.*" will automatically update the build and revision,
// and "{Major}.{Minor}.{Build}.*" will update just the revision.
[assembly: AssemblyVersion("1.0.*")]
// The following attributes are used to specify the signing key for the assembly,
// if desired. See the Mono documentation for more information about signing.
//[assembly: AssemblyDelaySign(false)]
//[assembly: AssemblyKeyFile("")]

View File

@ -1,16 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
// [PacketHandlerAttr(0x0E, 0x19)]
// public class _0E_19_UNK : PacketHandler
// {
// public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
// {
// //var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
// //Logger.WriteHex(info, data);
// }
// }
}

View File

@ -1,41 +0,0 @@
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x03)]
public class InitialLoad : PacketHandler
{
/// (0x03, 0x03) Initial Load (?).
///
/// (C -> S) Sent when the client loads for the first time in the session.
///
/// Response to: [`Packet::LoadingScreenTransition`] (?).
///
/// Respond with: lobby map setup.
// Ninji note: 3-3 may not be the correct place to do this
// Once we have better state tracking, we should make sure that
// 3-3 only does anything at the points where the client is supposed
// to be sending it, etc etc
// This seems to only ever be called once after logging in, yet is also handled by 11-3E in other places
// Moved the actual handling into 11-3E until I can actually confirm this
// Just insantiate a new CharacterSpawn and push it through until then
// - Kyle
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
// Spawn Accounts
new CharacterSpawn().HandlePacket(context, flags, data, position, size);
}
#endregion
}
}

View File

@ -1,20 +0,0 @@
using System;
using System.IO;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Database;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x0C)]
public class ServerPong : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
Logger.Write("[HI!] 收到 {0} Ping回应 ", context._account.Username);
}
#endregion
}
}

View File

@ -1,60 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x10)]
public class MapLoaded : PacketHandler
{
public unsafe struct MapLoadedPacket
{
/// Loaded zone object.
public ObjectHeader MapObject;
/// Unknown data, 32 bytes.
public fixed byte Unk[0x20];
}
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<MapLoadedPacket>();
Logger.Write($"MapObject {pkt.MapObject.ToString()}");
//TODO 这里才是完整的让角色在地图生成的地方
context.SendPacket(new LobbyMonitorPacket(1));
if(context.Character == null)
{
Logger.WriteError("Character should be loaded here");
return;
}
// Unlock Controls
context.SendPacket(new UnlockControlsPacket()); // Inital spawn only, move this!
context.SendPacket(new LoadingScreenRemovePacket());
}
#endregion
}
}

View File

@ -1,41 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Zone;
using static PSO2SERVER.Protocol.Handlers.TeleportCasinoToLobby;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x12)]
public class CampshipTeleport : PacketHandler
{
public struct ToCampshipPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
public uint unk4;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.currentParty.currentQuest == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<ToCampshipPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3} unk4 {pkt.unk4}");
var instanceName = String.Format("{0}-{1}", context.currentParty.currentQuest.name, context._account.Nickname);
ZoneManager.Instance.NewInstance(instanceName, new Map("campship", 150, 0, Map.MapType.Campship, 0));
// todo: add next map
ZoneManager.Instance.AddMapToInstance(instanceName, new Map("area1", 311, -1, Map.MapType.Other, (Map.MapFlags)0x6) { GenerationArgs = new Map.GenParam((uint)new Random().Next(), 2, 3)});
Map campship = ZoneManager.Instance.MapFromInstance("campship", instanceName);
campship.SpawnClient(context, new PSOLocation(0, 1, 0, 0, 0, 0, 0), context.currentParty.currentQuest.name);
}
}
}

View File

@ -1,42 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Zone;
using static PSO2SERVER.Protocol.Handlers.TeleportCasinoToLobby;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x16)]
public class CampshipTeleportDown : PacketHandler
{
public struct CampshipDownPacket
{
public uint zone_id;
public uint unk2;
public uint unk3;
public uint unk4;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.currentParty.currentQuest == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<CampshipDownPacket>();
Logger.Write($"zone_id {pkt.zone_id} unk2 {pkt.unk2} unk3 {pkt.unk3} unk4 {pkt.unk4}");
// TODO: WTF terribad hax?
if (context.CurrentLocation.PosZ >= 20)
{
var instanceName = String.Format("{0}-{1}", context.currentParty.currentQuest.name, context._account.Nickname);
Map forest = ZoneManager.Instance.MapFromInstance("area1", instanceName);
forest.SpawnClient(context, new PSOLocation(0, 1, 0, -0, -37, 0.314f, 145.5f));
}
}
}
}

View File

@ -1,52 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using static PSO2SERVER.Protocol.Handlers.TeleportLobbyToCafe;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x34)]
public class TeleportCasinoToLobby : PacketHandler
{
/// (0x03, 0x34) Move Casino -> Lobby.
public struct CasinoToLobbyPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
public uint unk4;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<CasinoToLobbyPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3} unk4 {pkt.unk4}");
if(pkt.unk3 != 0x10)
{
Logger.WriteWarning("[WRN] Packet 0x3 0x34's first value was not 0x10! Investigate.");
}
PSOLocation destination;
if(pkt.unk4 == 0) // Gate area
{
destination = new PSOLocation(0f, 1f, 0f, 0f, -0.22f, 2.4f, 198.75f);
}
else // Shop area
{
destination = new PSOLocation(0f, 1f, 0f, 20f, 0.20f, 1.23f, -175.25f);
}
Map lobbyMap = ZoneManager.Instance.MapFromInstance("lobby", "lobby");
lobbyMap.SpawnClient(context, destination, "lobby");
}
}
}

View File

@ -1,44 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using static PSO2SERVER.Protocol.Handlers.TeleportLobbyToCafe;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x35)]
public class TeleportLobbyToCasino : PacketHandler
{
/// (0x03, 0x35) Move Lobby -> Casino.
public struct CasinoTransportPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
}
///[<--] 0x03 - 0x35 (TeleportLobbyToCasino) (Flags None) (20 字节)
/// unk1 0 unk2 0 unk3 16
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<CasinoTransportPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3}");
// Dunno what these are yet.
context.SendPacket(new Unk110APacket((uint)context._account.AccountId));
context.SendPacket(new Unk1E0CPacket(101));
Map casinoMap = ZoneManager.Instance.MapFromInstance("casino", "lobby");
casinoMap.SpawnClient(context, casinoMap.GetDefaultLocation());
}
}
}

View File

@ -1,53 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using static PSO2SERVER.Protocol.Handlers.TeleportCasinoToLobby;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x38)]
public class TeleportBridgeToLobby : PacketHandler
{
/// (0x03, 0x38) Move Bridge -> Lobby. TODO
public struct BridgeToLobbyPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
public uint zone_id;
public uint unk4;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data);
var pkt = reader.ReadStruct<BridgeToLobbyPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3} zone_id {pkt.zone_id} unk4 {pkt.unk4}");
if(pkt.unk3 != 0x10)
{
Logger.WriteWarning("[WRN] Packet 0x3 0x34's first value was not 0x10! Investigate.");
}
PSOLocation destination;
if(pkt.zone_id == 0) // Gate area
{
destination = new PSOLocation(0f, 1f, 0f, 0f, -0.22f, 2.4f, 198.75f);
}
else // Shop area
{
destination = new PSOLocation(0f, 1f, 0f, 20f, 0.20f, 1.23f, -175.25f);
}
Map lobbyMap = ZoneManager.Instance.MapFromInstance("lobby", "lobby");
lobbyMap.SpawnClient(context, destination, "lobby");
}
}
}

View File

@ -1,43 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using static PSO2SERVER.Protocol.Handlers.TeleportCasinoToLobby;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x39)]
public class TeleportLobbyToBridge : PacketHandler
{
/// (0x03, 0x39) Move Lobby -> Bridge.
public struct BridgeTransportPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<BridgeTransportPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3}");
// Dunno what these are yet.
context.SendPacket(new Unk110APacket((uint)context._account.AccountId));
context.SendPacket(new Unk1E0CPacket(101));
Map bridgeMap = ZoneManager.Instance.MapFromInstance("bridge", "lobby");
bridgeMap.SpawnClient(context, bridgeMap.GetDefaultLocation());
}
}
}

View File

@ -1,53 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using static PSO2SERVER.Protocol.Handlers.TeleportBridgeToLobby;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x3B)]
public class TeleportCafeToLobby : PacketHandler
{
/// (0x03, 0x3B) Move Cafe -> Lobby.
public struct CafeToLobbyPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
public uint zone_id;
public uint unk4;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data);
var pkt = reader.ReadStruct<CafeToLobbyPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3} zone_id {pkt.zone_id} unk4 {pkt.unk4}");
if(pkt.unk3 != 0x10)
{
Logger.WriteWarning("[WRN] Packet 0x3 0x34's first value was not 0x10! Investigate.");
}
PSOLocation destination;
if(pkt.zone_id == 0) // Gate area
{
destination = new PSOLocation(0f, 1f, 0f, 0f, -0.22f, 2.4f, 198.75f);
}
else // Shop area
{
destination = new PSOLocation(0f, 1f, 0f, 20f, 0.20f, 1.23f, -175.25f);
}
Map lobbyMap = ZoneManager.Instance.MapFromInstance("lobby", "lobby");
lobbyMap.SpawnClient(context, destination, "lobby");
}
}
}

View File

@ -1,50 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x03, 0x3C)]
public class TeleportLobbyToCafe : PacketHandler
{
/// (0x03, 0x3C) Move Lobby -> Cafe.
public struct CafeTransportPacket
{
public uint unk1;
public uint unk2;
public uint unk3;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<CafeTransportPacket>();
Logger.Write($"unk1 {pkt.unk1} unk2 {pkt.unk2} unk3 {pkt.unk3}");
// Dunno what these are yet.
context.SendPacket(new Unk110APacket((uint)context._account.AccountId));
context.SendPacket(new Unk1E0CPacket(101));
Map dstMap = ZoneManager.Instance.MapFromInstance("cafe", "lobby");
if(dstMap != null)
{
dstMap.SpawnClient(context, dstMap.GetDefaultLocation());
}
else
{
Logger.WriteError("cafe 区域不存在");
return;
}
}
}
}

View File

@ -1,175 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Runtime.InteropServices;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x07)]
public class MovementHandler : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
// This packet is "Compressed" basically.
reader.ReadBytes(6); // Get past the junk
// For simplicity's sake, read the 3 flag bytes into a big int
byte[] flagBytes = reader.ReadBytes(3);
uint dataFlags = flagBytes[0];
dataFlags |= (uint)(flagBytes[1] << 8);
dataFlags |= (uint)(flagBytes[2] << 16);
PackedData theFlags = (PackedData)dataFlags;
// Debug
//Logger.WriteInternal("[移动] Movement 数据包来自 {0} 包含 {1} 数据.", context.Character.name, theFlags);
// TODO: Maybe do this better someday
MovementPacket.FullMovementData dstData = new MovementPacket.FullMovementData();
if (theFlags.HasFlag(PackedData.ENT1_ID))
{
dstData.entity1.ID = (uint)reader.ReadUInt64();
}
if (theFlags.HasFlag(PackedData.ENT1_TYPE))
{
dstData.entity1.ObjectType = (ObjectType)reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT1_A))
{
dstData.entity1.MapID = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT2_ID))
{
dstData.entity1.ID = (uint)reader.ReadUInt64();
}
if (theFlags.HasFlag(PackedData.ENT2_TYPE))
{
dstData.entity1.ObjectType = (ObjectType)reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.ENT2_A))
{
dstData.entity1.MapID = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.TIMESTAMP))
{
dstData.timestamp = reader.ReadUInt32();
context.MovementTimestamp = dstData.timestamp;
}
if (theFlags.HasFlag(PackedData.ROT_X))
{
dstData.rotation.x = reader.ReadUInt16();
context.CurrentLocation.RotX = Helper.FloatFromHalfPrecision(dstData.rotation.x);
}
if (theFlags.HasFlag(PackedData.ROT_Y))
{
dstData.rotation.y = reader.ReadUInt16();
context.CurrentLocation.RotY = Helper.FloatFromHalfPrecision(dstData.rotation.y);
}
if (theFlags.HasFlag(PackedData.ROT_Z))
{
dstData.rotation.z = reader.ReadUInt16();
context.CurrentLocation.RotZ = Helper.FloatFromHalfPrecision(dstData.rotation.z);
}
if (theFlags.HasFlag(PackedData.ROT_W))
{
dstData.rotation.w = reader.ReadUInt16();
context.CurrentLocation.RotW = Helper.FloatFromHalfPrecision(dstData.rotation.w);
}
if (theFlags.HasFlag(PackedData.CUR_X))
{
dstData.currentPos.x = reader.ReadUInt16();
context.CurrentLocation.PosX = Helper.FloatFromHalfPrecision(dstData.currentPos.x);
}
if (theFlags.HasFlag(PackedData.CUR_Y))
{
dstData.currentPos.y = reader.ReadUInt16();
context.CurrentLocation.PosY = Helper.FloatFromHalfPrecision(dstData.currentPos.y);
}
if (theFlags.HasFlag(PackedData.CUR_Z))
{
dstData.currentPos.z = reader.ReadUInt16();
context.CurrentLocation.PosZ = Helper.FloatFromHalfPrecision(dstData.currentPos.z);
}
if (theFlags.HasFlag(PackedData.UNKNOWN4))
{
dstData.Unknown2 = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.UNK_X))
{
dstData.unknownPos.x = reader.ReadUInt16();
context.LastLocation.PosX = Helper.FloatFromHalfPrecision(dstData.unknownPos.x);
}
if (theFlags.HasFlag(PackedData.UNK_Y))
{
dstData.unknownPos.y = reader.ReadUInt16();
context.LastLocation.PosY = Helper.FloatFromHalfPrecision(dstData.unknownPos.y);
}
if (theFlags.HasFlag(PackedData.UNK_Z))
{
dstData.unknownPos.z = reader.ReadUInt16();
context.LastLocation.PosZ = Helper.FloatFromHalfPrecision(dstData.unknownPos.z);
}
if (theFlags.HasFlag(PackedData.UNKNOWN5))
{
dstData.Unknown3 = reader.ReadUInt16();
}
if (theFlags.HasFlag(PackedData.UNKNOWN6))
{
if (theFlags.HasFlag(PackedData.UNKNOWN7))
{
dstData.Unknown4 = reader.ReadByte();
}
else
{
dstData.Unknown4 = reader.ReadUInt32();
}
}
//Logger.WriteInternal("[移动] 玩家 {0} 移动中 (坐标: X{1}, Y{2}, Z{3})", context.Character.name, context.CurrentLocation.PosX,
//context.CurrentLocation.PosY, context.CurrentLocation.PosZ);
foreach (var c in Server.Instance.Clients)
{
if (c.Character == null || c == context || c.CurrentMap != context.CurrentMap)
continue;
//c.SendPacket(0x04, 0x07, flags, data);
c.SendPacket(new MovementPacket(dstData));
}
}
#endregion
}
[Flags]
public enum PackedData : Int32
{
ENT1_ID = 1,
ENT1_TYPE = 2,
ENT1_A = 4,
ENT2_ID = 8,
ENT2_TYPE = 0x10,
ENT2_A = 0x20,
TIMESTAMP = 0x40,
ROT_X = 0x80,
ROT_Y = 0x100,
ROT_Z = 0x200,
ROT_W = 0x400,
CUR_X = 0x800,
CUR_Y = 0x1000,
CUR_Z = 0x2000,
UNKNOWN4 = 0x4000,
UNK_X = 0x8000,
UNK_Y = 0x10000,
UNK_Z = 0x20000,
UNKNOWN5 = 0x40000,
UNKNOWN6 = 0x80000,
UNKNOWN7 = 0x100000
}
}

View File

@ -1,57 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x08)]
public class MovementActionHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
reader.ReadObjectHeader(); // Skip blank entity header.
var preformer = reader.ReadObjectHeader(); // Preformer
byte[] preData = reader.ReadBytes(40);
string command = reader.ReadAscii(0x922D, 0x45);
byte[] rest = reader.ReadBytes(4);
uint thingCount = reader.ReadMagic(0x922D, 0x45);
byte[] things;
PacketWriter thingWriter = new PacketWriter();
for (int i = 0; i < thingCount; i++)
{
thingWriter.Write(reader.ReadBytes(4));
}
things = thingWriter.ToArray();
byte[] final = reader.ReadBytes(4);
//Logger.WriteInternal("[动作] {0} 发送动作 {1}", context.Character.name, command);
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
//PacketWriter output = new PacketWriter();
//output.WriteStruct(new ObjectHeader((uint)context._account.AccountId, ObjectType.Accounts));
//output.WriteStruct(preformer);
//output.Write(preData);
//output.WriteAscii(command, 0x4315, 0x7A);
//output.Write(rest);
//output.WriteMagic(thingCount, 0x4315, 0x7A);
//output.Write(things);
//output.Write(final);
//c.SendPacket(0x4, 0x80, 0x44, output.ToArray());
c.SendPacket(new MovementActionServerPacket(context._account.AccountId, preformer, preData
, command, rest, thingCount, things, final));
}
}
}
}

View File

@ -1,28 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Zone;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x13)]
public class _04_13_UNK : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.currentParty.currentQuest == null)
return;
// TODO: WTF terribad hax?
if (context.CurrentLocation.PosZ >= 20)
{
var instanceName = String.Format("{0}-{1}", context.currentParty.currentQuest.name, context._account.Nickname);
Map forest = ZoneManager.Instance.MapFromInstance("area1", instanceName);
forest.SpawnClient(context, new PSOLocation(0, 1, 0, -0, -37, 0.314f, 145.5f));
}
}
}
}

View File

@ -1,144 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Models;
using PSO2SERVER.Object;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x14)]
public class ObjectInteract : PacketHandler
{
public class InteractPacket
{
// Fields
public byte[] Unk1 { get; set; } // [u8; 0xC]
public ObjectHeader Object1 { get; set; }
public byte[] Unk2 { get; set; } // [u8; 0x4]
public ObjectHeader Object3 { get; set; }
public byte[] Object4 { get; set; } // [u8; 0x10]
public string Action { get; set; } // AsciiString
// Constructor
public InteractPacket()
{
Unk1 = new byte[0x0C]; // Initialize arrays with appropriate sizes
Unk2 = new byte[0x04];
Object4 = new byte[0x10];
Action = string.Empty;
}
// Deserialize constructor (for example purposes, if needed)
public InteractPacket(byte[] unk1, ObjectHeader object1, byte[] unk2, ObjectHeader object3, byte[] object4, string action)
{
Unk1 = unk1;
Object1 = object1;
Unk2 = unk2;
Object3 = object3;
Object4 = object4;
Action = action;
}
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data, position, size);
var pkt = new InteractPacket
{
Unk1 = reader.ReadBytes(0x0C),
Object1 = reader.ReadObjectHeader(),
Unk2 = reader.ReadBytes(4),
Object3 = reader.ReadObjectHeader(),
Object4 = reader.ReadBytes(0x10),
Action = reader.ReadAscii(0xD711, 0xCA)
};
//Logger.Write($"PKT:Object1 {pkt.Object1.ToString()} Object3 {pkt.Object3.ToString()} Action {pkt.Action}");
PSOObject srcObj;
if(pkt.Object1.ObjectType == ObjectType.Object)
{
srcObj = ObjectManager.Instance.getObjectByID(context.CurrentMap.Name, pkt.Object1.ID);
}
else if(pkt.Object1.ObjectType == ObjectType.Player)
{
srcObj = new PSOObject
{
Header = pkt.Object1,
Name = "Accounts"
};
}
else
{
srcObj = null;
}
//Logger.WriteInternal("[OBJ] {0} (ID {1}) <{2}> --> Ent {3} (ID {4})", srcObj.name, srcObj.player.ID, command, (ObjectType)dstObject.ObjectType, dstObject.ID);
// TODO: Delete this code and do this COMPLETELY correctly!!!
if (pkt.Action == "Transfer" && context.CurrentMap.Name == "lobby")
{
// Try and get the teleport definition for the object...
using (var db = new ServerEf())
{
db.Configuration.AutoDetectChangesEnabled = true;
var teleporterEndpoint = db.Teleports.Find("lobby", (int)pkt.Object1.ID);
if (teleporterEndpoint == null)
{
Logger.WriteError("[OBJ] 传输 {0} 于 {1} 没有包含一个有效目的地!", srcObj.Header.ID, "lobby");
// Teleport Accounts to default point
context.SendPacket(new TeleportTransferPacket(srcObj, new PSOLocation(0f, 1f, 0f, -0.000031f, -0.417969f, 0.000031f, 134.375f)));
// Unhide player
context.SendPacket(new ObjectActionPacket(pkt.Object3, pkt.Object1, new ObjectHeader(), new ObjectHeader(), "Forwarded"));
}
else
{
PSOLocation endpointLocation = new PSOLocation()
{
RotX = teleporterEndpoint.RotX,
RotY = teleporterEndpoint.RotY,
RotZ = teleporterEndpoint.RotZ,
RotW = teleporterEndpoint.RotW,
PosX = teleporterEndpoint.PosX,
PosY = teleporterEndpoint.PosY,
PosZ = teleporterEndpoint.PosZ,
};
// Teleport Accounts
context.SendPacket(new TeleportTransferPacket(srcObj, endpointLocation));
// Unhide player
context.SendPacket(new ObjectActionPacket(pkt.Object3, pkt.Object1, new ObjectHeader(), new ObjectHeader(), "Forwarded"));
}
}
}
if (pkt.Action == "READY")
{
context.SendPacket(new ObjectActionPacket(new ObjectHeader((uint)context._account.AccountId, ObjectType.Player), srcObj.Header, srcObj.Header,
new ObjectHeader(), "FavsNeutral"));
context.SendPacket(new ObjectActionPacket(new ObjectHeader((uint)context._account.AccountId, ObjectType.Player), srcObj.Header, srcObj.Header,
new ObjectHeader(), "AP")); // Short for Appear, Thanks Zapero!
}
if (pkt.Action == "Sit")
{
foreach (var client in Server.Instance.Clients)
{
if (client.Character == null || client == context)
continue;
client.SendPacket(new ObjectActionPacket(new ObjectHeader((uint)client._account.AccountId, ObjectType.Player), srcObj.Header,
new ObjectHeader(pkt.Object3.ID, ObjectType.Player), new ObjectHeader(), "SitSuccess"));
}
}
}
}
}

View File

@ -1,36 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x3C)]
public class ActionUpdateHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
reader.ReadObjectHeader(); // Read the blank
ObjectHeader actor = reader.ReadObjectHeader(); // Read the actor
byte[] rest = reader.ReadBytes(32); // TODO Map this out and do stuff with it!
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
//PacketWriter writer = new PacketWriter();
//writer.WriteStruct(new ObjectHeader((uint)c._account.AccountId, ObjectType.Accounts));
//writer.WriteStruct(actor);
//writer.Write(rest);
//c.SendPacket(0x4, 0x81, 0x40, writer.ToArray());
c.SendPacket(new ActionUpdateServerPacket(c._account.AccountId, actor, rest));
}
}
}
}

View File

@ -1,43 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x71)]
public class MovementEndHandler : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
PacketReader reader = new PacketReader(data);
MovementPacket.FullMovementData movData = reader.ReadStruct<MovementPacket.FullMovementData>();
if (movData.entity1.ID == 0 && movData.entity2.ID != 0)
movData.entity1 = movData.entity2;
movData.timestamp = 0;
//Logger.WriteInternal("[移动] 玩家 {0} 停止移动 (坐标:{1}, {2}, {3})", context.Character.name,
// Helper.FloatFromHalfPrecision(movData.currentPos.x), Helper.FloatFromHalfPrecision(movData.currentPos.y),
// Helper.FloatFromHalfPrecision(movData.currentPos.z));
foreach (var c in Server.Instance.Clients)
{
if (c == context || c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
c.SendPacket(new MovementEndPacket(movData));
}
}
#endregion
}
}

View File

@ -1,52 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using PSO2SERVER.Models;
using PSO2SERVER.Party;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Zone;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x04, 0x75)]
public class ActionEnd : PacketHandler
{
public ObjectHeader unk1 { get; set; } = new ObjectHeader();
/// Object that was performing an action.
public ObjectHeader performer { get; set; } = new ObjectHeader();
public uint unk2 { get; set; } = 0;
public ObjectHeader unk3 { get; set; } = new ObjectHeader();
public ObjectHeader unk4 { get; set; } = new ObjectHeader();
public byte[] unk5 { get; set; } = new byte[0x04];
public string action { get; set; } = string.Empty;//Ascii
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
unk1 = reader.ReadObjectHeader();
performer = reader.ReadObjectHeader();
unk2 = reader.ReadUInt32();
unk3 = reader.ReadObjectHeader();
unk4 = reader.ReadObjectHeader();
unk5 = reader.ReadBytes(4);
action = reader.ReadAscii(0x83EF, 0x40);
// 打印各个字段的内容
Logger.Write("unk1: " + unk1); // 假设 ObjectHeader 重载了 ToString() 方法
Logger.Write("performer: " + performer); // 同上
Logger.Write("unk2: " + unk2);
Logger.Write("unk3: " + unk3);
Logger.Write("unk4: " + unk4);
Logger.Write("unk5: " + BitConverter.ToString(unk5)); // 打印 byte 数组
Logger.Write("action: " + action); // 打印 ASCII 字符串
foreach (Client cl in PartyManager.instance.GetCurrentPartyForClient(context).getMembers())
{
cl.SendPacket(new ActionEndPacket(unk1, performer, unk2, unk3, unk4, unk5, action));
}
}
}
}

View File

@ -1,38 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x06, 0x01)]
public class DealDamage : PacketHandler
{
public unsafe struct DealDamagePacket
{
/// Object that inflicted the damage.
public ObjectHeader Inflicter;
/// Object that received the damage.
public ObjectHeader Target;
public uint Attack_id;
public ulong Unk2;
/// Hitbox ID (?).
public uint Hitbox_id;
/// Hit x position.
public Half X_pos;
/// Hit y position.
public Half Y_pos;
/// Hit z position.
public Half Z_pos;
public ushort unk4;
public ulong unk5;
public fixed byte unk6[0x18];
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
}
}
}

View File

@ -1,69 +0,0 @@
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System.Collections.Generic;
using System.IO;
using System.Linq;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x07, 0x00)]
public class ChatHandler : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.Character == null)
return;
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var obj = reader.ReadObjectHeader();
var channel = reader.ReadByte();
var unk3 = reader.ReadByte();
var unk4 = reader.ReadInt16();
reader.BaseStream.Seek(0x4, SeekOrigin.Current);
var message = reader.ReadUtf16(0x9D3F, 0x44);
if (message.StartsWith(ServerApp.Config.CommandPrefix))
{
var valid = false;
// Iterate commands
foreach (var command in ServerApp.ConsoleSystem.Commands)
{
var full = message.Substring(1); // Strip the command chars
var args = full.Split(' ');
if (command.Names.Any(name => args[0].ToLower() == name.ToLower()))
{
command.Run(args, args.Length, full, context);
valid = true;
Logger.WriteCommand(null, "[CMD] {0} 发送指令 {1}", context._account.Username, full);
}
if (valid)
break;
}
if (!valid)
Logger.WriteClient(context, "[CMD] {0} - 指令不存在", message.Split(' ')[0].Trim('\r'));
}
else
{
Logger.Write("[CHT] <{0}> 频道{1}说 {2}", context.Character.Name, channel, message);
foreach (var c in Server.Instance.Clients)
{
if (c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
c.SendPacket(new ChatPacket((uint)context._account.AccountId, channel, message));
}
}
}
#endregion
}
}

View File

@ -1,23 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x07, 0x37)]
public class _07_37_UNK : PacketHandler
{
public uint unk { get; set; } = 0;
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
unk = reader.ReadUInt32();
Logger.WriteObj(unk);
}
}
}

View File

@ -1,16 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x07, 0x3D)]
public class _07_3D_UNK : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
}
}
}

View File

@ -1,48 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x09)]
public class MinimapRevealRequest : PacketHandler
{
// 定义 MinimapRevealRequestPacket 数据包结构
public struct MinimapRevealRequestPacket
{
// 未知字段32位无符号整数
public uint unk1;
// 玩家进入的区域块的ID
public uint chunk_id;
// 地图上区域块的列坐标
public uint map_column;
// 地图上区域块的行坐标
public uint map_row;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
if (context.Character == null)
return;
var reader = new PacketReader(data, position, size);
// 从接收到的字节数组中解析数据到结构体
MinimapRevealRequestPacket packet = new MinimapRevealRequestPacket
{
unk1 = reader.ReadUInt32(), // 从数据包中解析unk1
chunk_id = reader.ReadUInt32(), // 从数据包中解析chunk_id
map_column = reader.ReadUInt32(), // 从数据包中解析map_column
map_row = reader.ReadUInt32() // 从数据包中解析map_row
};
// 打印解析后的数据
Logger.Write($"解析的地图数据包: unk1 = {packet.unk1}, Chunk ID = {packet.chunk_id}, Column = {packet.map_column}, Row = {packet.map_row}");
}
}
}

View File

@ -1,38 +0,0 @@
using PSO2SERVER.Json;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x15)]
public class QuestCounterAvailableHander : PacketHandler
{
public struct AvailableQuestsRequestPacket
{
public uint unk1;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context == null)
return;
string jsonFilePath4 = "data\\quests\\Story Quests\\EP1\\700000 - An Encounter with Xion\\data.json";
var quest = JsonRead.DeserializeJson<QuestData>(jsonFilePath4);
var quests = new List<QuestData>();
quests.Add(quest);
context.SendPacket(new QuestAvailablePacket(quests));
}
}
}

View File

@ -1,60 +0,0 @@
using PSO2SERVER.Json;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x17)]
public class QuestCategoryRequest : PacketHandler
{
public uint unk1 { get; set; }
public QuestType category { get; set; }
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0}字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
unk1 = reader.ReadUInt32();
category = (QuestType)reader.ReadUInt32();
Logger.Write($"unk1 = {unk1} category = {category}");
string jsonFilePath4 = "data\\quests\\Story Quests\\EP1\\700000 - An Encounter with Xion\\data.json";
var quest = JsonRead.DeserializeJson<QuestData>(jsonFilePath4);
var quests = new List<QuestData>();
quests.Add(quest);
//// What am I doing
//QuestDefiniton[] defs = new QuestDefiniton[1];
//for (int i = 0; i < defs.Length; i++)
//{
// defs[i].dateOrSomething = "2012/01/05";
// defs[i].quest_obj = new ObjectHeader(0x20, ObjectType.PartyQuest);
// defs[i].questNameid = 30010;
// defs[i].playTime = QuestEstimatedTime.Short;
// defs[i].partyType = QuestPartyType.SingleParty;
// defs[i].difficulties = QuestDifficultyType.NORMAL | QuestDifficultyType.HARD | QuestDifficultyType.VERY_HARD | QuestDifficultyType.SUPER_HARD;
// defs[i].req_level = 1;
// // Not sure why but these need to be set for the quest to be enabled
// defs[i].quest_type = (QuestType)0xF1;
// defs[i].field_101 = 1;
//}
context.SendPacket(new QuestCategoryPacket(quests));
context.SendPacket(new QuestCategoryStopperPacket());
}
}
}

View File

@ -1,79 +0,0 @@
using Org.BouncyCastle.Bcpg;
using PSO2SERVER.Json;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using System.Collections.Generic;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x19)]
public class QuestDifficultyRequestHandler : PacketHandler
{
/// List of objects of requested quests.
public List<ObjectHeader> Quests { get; set; } = new List<ObjectHeader>();
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0}字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var questCount = reader.ReadMagic(0xA36E, 0x10);
for (uint i = 0; i < questCount; i++)
{
var qobj = new ObjectHeader();
qobj.ReadObjectHeaderFromStream(reader);
Quests.Add(qobj);
// 打印每个任务的 ID 和类型,方便调试
Logger.Write($"任务 ID: {qobj.ID}, 任务类型: {qobj.ObjectType}");
}
string jsonFilePath4 = "data\\quests\\Story Quests\\EP1\\700000 - An Encounter with Xion\\data.json";
var quest = JsonRead.DeserializeJson<QuestData>(jsonFilePath4);
var quests = new List<QuestData>();
quests.Add(quest);
//QuestDifficulty[] diffs = new QuestDifficulty[1];
//for (int i = 0; i < diffs.Length; i++)
//{
// diffs[i].dateOrSomething = "2012/01/05";
// diffs[i].quest_obj = new ObjectHeader(0x20, ObjectType.PartyQuest);
// diffs[i].name_id = 30010;
// diffs[i].area = 0x01;
// diffs[i].planet = 0x03;
// diffs[i].unk1 = 0x03;
// diffs[i].unk2 = 0x00;
// diffs[i].difficulty1 = new QuestDifficultyEntry();
// diffs[i].difficulty2 = new QuestDifficultyEntry();
// diffs[i].difficulty3 = new QuestDifficultyEntry();
// diffs[i].difficulty4 = new QuestDifficultyEntry();
// diffs[i].difficulty5 = new QuestDifficultyEntry();
// diffs[i].difficulty6 = new QuestDifficultyEntry();
// diffs[i].difficulty7 = new QuestDifficultyEntry();
// diffs[i].difficulty8 = new QuestDifficultyEntry();
//}
context.SendPacket(new QuestDifficultyPacket(quests));
// [K873] I believe this is the correct packet, but it causes an infinite send/recieve loop, we're probably just missing something else
context.SendPacket(new QuestDifficultySendFinishedPacket());
//// 现在可以对任务列表Quests进行处理
//// 例如,可以记录任务的信息或进行其他处理
//foreach (var quest in Quests)
//{
// if(quest.ObjectType == ObjectType.PartyQuest)
// {
// }
//}
}
}
}

View File

@ -1,36 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x20)]
public class AcceptQuest : PacketHandler
{
public ObjectHeader quest_obj { get; set; }
public ushort diff { get; set; }
public ushort unk1 { get; set; }
public uint[] unk2 { get; set; } = new uint[7];
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
try
{
quest_obj = reader.ReadObjectHeader();
diff = reader.ReadUInt16();
unk1 = reader.ReadUInt16();
for (int i = 0; i < 7; i++) { unk2[i] = reader.ReadUInt32(); }
}
catch (Exception ex)
{
Logger.WriteError("Error while processing AcceptQuest packet: " + ex.Message);
}
Logger.Write($"PartyQuest Object: {quest_obj}, Difficulty: {diff}, unk1: {unk1}, unk2: {string.Join(", ", unk2)}");
}
}
}

View File

@ -1,22 +0,0 @@
using PSO2SERVER.Protocol.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0x30)]
public class QuestCounterHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
//Not sure what this does yet
context.SendPacket(new Unk4901Packet());
context.SendPacket(new Unk0E65Packet());
context.SendPacket(new Unk0B22Packet());
}
}
}

View File

@ -1,33 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Party;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0B, 0xCD)]
public class AcceptStoryQuestHandler : PacketHandler
{
public struct AcceptStoryQuestPacket
{
public uint name_id;
public uint unk;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<AcceptStoryQuestPacket>();
Logger.Write("任务编号: " + pkt.name_id + " unk: " + pkt.unk);
PartyManager.Instance.CreateNewParty(context);
// 告诉客户端切换到加载界面
//context.SendPacket(new LoadingScreenTransitionPacket());
}
}
}

View File

@ -1,84 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Party;
using PSO2SERVER.Json;
using System.Collections.Generic;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0E, 0x0C)]
public class NewPartySettings : PacketHandler
{
public string name { get; set; } = string.Empty;
public string password { get; set; } = string.Empty;
public string comments { get; set; } = string.Empty;
public string questname { get; set; } = string.Empty;
public byte min_level { get; set; } = 0;
public byte max_level { get; set; } = 0;
public byte playstyle { get; set; } = 0;
public PartyFlags partyFlags { get; set; } = 0;
public ulong unk { get; set; } = 0;
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0}字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
name = reader.ReadUtf16(0x11CB, 0x98);
password = reader.ReadUtf16(0x11CB, 0x98);
comments = reader.ReadUtf16(0x11CB, 0x98);
questname = reader.ReadUtf16(0x11CB, 0x98);
min_level = reader.ReadByte();
max_level = reader.ReadByte();
playstyle = reader.ReadByte();
partyFlags = (PartyFlags)reader.ReadByte();
unk = reader.ReadUInt64();
// 打印输出
Logger.Write($"name: {name}");
Logger.Write($"Password: {password}");
Logger.Write($"Comments: {comments}");
Logger.Write($"PartyQuest name: {questname}");
Logger.Write($"Min Level: {min_level}");
Logger.Write($"Max Level: {max_level}");
Logger.Write($"Playstyle: {playstyle}");
Logger.Write($"Party Flags: {partyFlags}"); // 如果 PartyFlags 是枚举类型
Logger.Write($"Unknown Value: {unk}");
context.currentParty.partySetting = new PartySettingsPacket
{
name = name,
password = password,
comments = comments,
min_level = min_level,
max_level = max_level,
playstyle = playstyle,
flags = partyFlags,
unk = unk,
};
context.SendPacket(new PartySettingsPacket(context.currentParty.partySetting));
if (questname != "")
{
context.currentParty.questname = questname;
//TODO questname
string jsonFilePath4 = "data\\quests\\Story Quests\\EP1\\700000 - An Encounter with Xion\\data.json";
var quest = JsonRead.DeserializeJson<QuestData>(jsonFilePath4);
var quests = new List<QuestData>();
quests.Add(quest);
context.currentParty.currentQuest.Quest = quest;
context.SendPacket(new SetQuestInfoPacket(quest.QuestDefiniton, 0, (uint)context._account.AccountId));
context.SendPacket(new PartySetQuestPacket(0x753A, 0, quest, (uint)context._account.AccountId));
}
}
}
}

View File

@ -1,22 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0E, 0x19)]
public class ChatStatusHandler : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
var reader = new PacketReader(data);
var obj = reader.ReadObjectHeader();
var status = reader.ReadUInt32();
context.SendPacket(new ChatStatusPacket(obj.ID, status));
}
}
}

View File

@ -1,25 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0E, 0x28)]
public class PlayerIsBusyState : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.Character == null)
return;
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
foreach (var c in Server.Instance.Clients)
{
if (c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
c.SendPacket(new NewBusyStatePacket(context._account.AccountId, BusyState.Busy));
}
}
}
}

View File

@ -1,26 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0E, 0x29)]
public class PlayerIsNotBusyState : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context.Character == null)
return;
//var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
//Logger.WriteHex(info, data);
foreach (var c in Server.Instance.Clients)
{
if (c.Character == null || c.CurrentMap != context.CurrentMap)
continue;
c.SendPacket(new NewBusyStatePacket(context._account.AccountId, BusyState.NotBusy));
}
}
}
}

View File

@ -1,37 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0F, 0x01)]
public class ItemPickupRequest : PacketHandler
{
public struct ItemPickupRequestPacket
{
/// Item drop ID.
public uint DropId { get; set; }
/// Unknown field.
public uint Unk { get; set; }
// Constructor for convenience
public ItemPickupRequestPacket(uint dropId, uint unk)
{
DropId = dropId;
Unk = unk;
}
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<ItemPickupRequestPacket>();
context.SendPacket(new ItemPickupResponsePacket(context._account.AccountId, pkt.DropId, 1, pkt.Unk));
}
}
}

View File

@ -1,27 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0F, 0x1C)]
public class GetItemDescription : PacketHandler
{
public struct GetItemDescriptionPacket
{
/// Item ID which description is requested.
public ItemId Item { get; set; }
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<GetItemDescriptionPacket>();
Logger.WriteObj(pkt.Item);
}
}
}

View File

@ -1,16 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0F, 0x28)]
public class _0F_28_UNK : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
}
}
}

View File

@ -1,16 +0,0 @@
using System;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0F, 0xDA)]
public class _0F_DA_UNK : PacketHandler
{
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
}
}
}

View File

@ -1,37 +0,0 @@
using System;
using System.Collections.Generic;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x0F, 0xE0)]
public class MoveToMatStorageRequest : PacketHandler
{
public struct MoveToMatStorageRequestPacket
{
/// Information about items being moved.
public List<MaterialStorageItem> Items;
}
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
var items_count = reader.ReadMagic(0x9087, 0xEA);
var pkt = new MoveToMatStorageRequestPacket();
for (int i = 0; i < items_count; i++)
{
var item = reader.ReadStruct<MaterialStorageItem>();
pkt.Items.Add(item);
}
//TODO
var newinv = new List<UpdatedInventoryItem>();
context.SendPacket(new MoveToMatStoragePacket(newinv, pkt.Items));
}
}
}

View File

@ -1,250 +0,0 @@
using System.IO;
using System.Linq;
using System.Text.RegularExpressions;
using PSO2SERVER.Database;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Models;
using System.Text;
using System.Collections.Generic;
using System;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x11, 0x00)]
public class SegaIDLogin : PacketHandler
{
public uint Unk1 { get; set; }
public uint Unk2 { get; set; }
public uint Unk3 { get; set; }
public byte[] VerId { get; set; } = new byte[0x20];
public List<NetInterface> Interfaces { get; set; } = new List<NetInterface>();
public byte[] Unk4 { get; set; } = new byte[0x90];
public byte[] Unk5 { get; set; } = new byte[0x10];
public Language TextLang { get; set; }
public Language VoiceLang { get; set; }
public Language TextLang2 { get; set; }
public Language LangLang { get; set; }
public string LanguageCode { get; set; } = new string('\0', 0x10);
public uint Unk6 { get; set; }
public uint Unk7 { get; set; }
public uint Magic1 { get; set; }
public byte[] Unk8 { get; set; } = new byte[0x20];
public byte[] Unk9 { get; set; } = new byte[0x44];
public string Username { get; set; } = new string('\0', 0x40);
public string Password { get; set; } = new string('\0', 0x40);
public uint Unk10 { get; set; }
public string Unk11 { get; set; } = string.Empty;
public List<NetInterface> ReadNetInterfaces(PacketReader reader, uint count)
{
var interfaces = new List<NetInterface>();
for (uint i = 0; i < count; i++)
{
var netInterface = new NetInterface();
netInterface.ReadFromStream(reader);
interfaces.Add(netInterface);
}
return interfaces;
}
public void SaveNetInterfacesToDatabase(int AccountId, string Username, List<NetInterface> interfaces)
{
using (var db = new ServerEf())
{
// 使用 AddRange 来批量插入数据
var newInterfaces = interfaces.Select(netInterface => new AccountNetInterFace
{
AccountId = AccountId,
Username = Username,
State = (int)netInterface.State, // 确保转换类型正确
Mac = netInterface.Mac
}).ToList();
db.AccountsNetInterFaces.AddRange(newInterfaces);
db.SaveChanges(); // 批量保存更改
}
}
// 假设你有一个读取到的实例
public void PrintInterfaces(List<NetInterface> interfaces)
{
foreach (var netInterface in interfaces)
{
Logger.Write(netInterface.ToString());
}
}
public void ReadFromStream(PacketReader reader)
{
Unk1 = reader.ReadUInt32();
Unk2 = reader.ReadUInt32();
Unk3 = reader.ReadUInt32();
VerId = reader.ReadBytes(0x20);
var macCount = reader.ReadMagic(0x5E6, 107);
// Assuming Interfaces is populated somehow
// e.g. Interfaces = ReadNetInterfaces(reader);
Interfaces = ReadNetInterfaces(reader, macCount);
// Read the fixed length fields
reader.BaseStream.Seek(0x14, SeekOrigin.Current);
Unk4 = reader.ReadBytes(0x90);
reader.BaseStream.Seek(0x10, SeekOrigin.Current);
Unk5 = reader.ReadBytes(0x10);
reader.BaseStream.Seek(0x10, SeekOrigin.Current);
TextLang = (Language)reader.ReadUInt32(); // Adjust based on actual type
VoiceLang = (Language)reader.ReadUInt32(); // Adjust based on actual type
TextLang2 = (Language)reader.ReadUInt32(); // Adjust based on actual type
LangLang = (Language)reader.ReadUInt32(); // Adjust based on actual type
reader.BaseStream.Seek(0x8, SeekOrigin.Current);
LanguageCode = reader.ReadFixedLengthUtf16(0x10);
Unk6 = reader.ReadUInt32();
Unk7 = reader.ReadUInt32();
Magic1 = reader.ReadUInt32();
Unk8 = reader.ReadBytes(0x20);
Unk9 = reader.ReadBytes(0x44);
// Read Username and Password
reader.BaseStream.Seek(0x104, SeekOrigin.Current);
Username = reader.ReadFixedLengthAscii(0x40);
reader.BaseStream.Seek(0x20, SeekOrigin.Current);
Password = reader.ReadFixedLengthAscii(0x40);
reader.BaseStream.Seek(0x04, SeekOrigin.Current);
Unk10 = reader.ReadUInt32();
Unk11 = reader.ReadFixedLengthAscii(0x08);
}
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data, position, size);
ReadFromStream(reader);
//var info = string.Format("[<--] 接收到的数据 (hex): {0}字节", data.Length);
//Logger.WriteHex(info, data);
//Logger.Write("用户名 {0} 密码 {1} - {2}", Username, Password, BCrypt.Net.BCrypt.HashPassword(Password));
// What am I doing here even
using (var db = new ServerEf())
{
var users = from u in db.Accounts
where u.Username.ToLower().Equals(Username.ToLower())
select u;
var error = "";
Account user;
if (!users.Any())
{
// Check if there is an empty field
if (string.IsNullOrWhiteSpace(Username))
{
error = "賬戶名為空.";
user = null;
}
if (string.IsNullOrWhiteSpace(Password))
{
error = "密碼為空 #1.";
user = null;
}
// Check for special characters
else if (!Regex.IsMatch(Username, "^[a-zA-Z0-9 ]*$", RegexOptions.IgnoreCase))
{
error = "用戶名不能包含特殊字符\n請只使用字母和數字.";
user = null;
}
else // We're all good!
{
// 直接插入新账户至数据库
user = new Account
{
Username = Username.ToLower(),
Password = BCrypt.Net.BCrypt.HashPassword(Password),
Nickname = Username.ToLower(),
// Since we can't display the Nickname prompt yet, just default it to the username
SettingsIni = File.ReadAllText(ServerApp.ServerSettingsKey),
TextLang = (int)TextLang,
VoiceLang = (int)VoiceLang,
TextLang2 = (int)TextLang2,
LangLang = (int)LangLang,
LanguageCode = LanguageCode,
};
db.Accounts.Add(user);
db.SaveChanges();
context.SendPacket(new NicknameRequestPacket()); // Request Nickname
}
}
else
{
user = users.First();
var existingUser = db.Accounts.FirstOrDefault(u => u.AccountId == user.AccountId);
if (existingUser != null)
{
// 更新值
existingUser.TextLang = (int)TextLang;
existingUser.VoiceLang = (int)VoiceLang;
existingUser.TextLang2 = (int)TextLang2;
existingUser.LangLang = (int)LangLang;
existingUser.LanguageCode = LanguageCode;
// 提交更改到数据库
db.SaveChanges();
SaveNetInterfacesToDatabase(user.AccountId, Username, Interfaces);
}
//TODO 方便GM测试
//if (Password != user.Password)
//{
// if (Password == "")
// {
// error = "密碼為空 #2.";
// user = null;
// }
// else
// if (!BCrypt.Net.BCrypt.Verify(Password, user.Password))
// {
// error = "密碼錯誤.";
// user = null;
// }
//}
}
context.SendPacket(new LoginDataPacket("夢幻之星2", error, (user == null) ? (uint)0 : (uint)user.AccountId));
//Mystery packet
//var mystery = new PacketWriter();
//mystery.Write((uint)100);
//context.SendPacket(0x11, 0x49, 0, mystery.ToArray());
// SegaIDLogin response packet
if (user == null)
{
return;
}
context._account = user;
}
if (ServerApp.Config.motd != "")
{
context.SendPacket(new SystemMessagePacket(ServerApp.Config.motd, SystemMessagePacket.MessageType.AdminMessageInstant));
}
}
#endregion
}
}

View File

@ -1,24 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Protocol.Packets;
using System.IO;
using System.Linq;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x11, 0x02)]
public class CharacterListRequest : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
//context.SendPacket(File.ReadAllBytes("packets/0000_07_45_49_217936300.bin"));
context.SendPacket(new CharacterListPacket(context._account.AccountId));
}
#endregion
}
}

View File

@ -1,74 +0,0 @@
using PSO2SERVER.Database;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Party;
using System;
using System.Linq;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x11, 0x04)]
public class CharacterSelected : PacketHandler
{
public struct CharacterSelectedPacket
{
public uint CharId;
public uint Unk1;
public uint Unk2;
}
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var reader = new PacketReader(data, position, size);
var pkt = reader.ReadStruct<CharacterSelectedPacket>();
var charId = pkt.CharId;
//Logger.Write("id {0}", charId);
if (context.Character == null)
{
using (var db = new ServerEf())
{
try
{
var character = db.Characters.FirstOrDefault(c => c.CharacterID == charId && c.AccountID == context._account.AccountId);
if (character == null)
{
Logger.WriteError("数据库中未找到 {0} 角色ID {1} ({2})"
, context._account.Username
, charId
, context._account.AccountId
);
context.Socket.Close();
return;
}
context.Character = character;
}
catch (Exception ex)
{
Logger.WriteError("查询角色时发生异常: {0}", ex.Message);
context.Socket.Close();
}
}
}
// 将客户端加入空余的队伍中
PartyManager.Instance.CreateNewParty(context);
// 告诉客户端切换到加载界面
context.SendPacket(new LoadingScreenTransitionPacket());
// TODO Set area, Set character, possibly more. See PolarisLegacy for more.
}
#endregion
}
}

View File

@ -1,125 +0,0 @@
using System.IO;
using System.Data.Entity;
using PSO2SERVER.Models;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Database;
using System.Linq;
using static PSO2SERVER.Models.CharacterStruct;
using System.Security.Cryptography.Pkcs;
using System.Runtime.InteropServices;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x11, 0x05)]
public class CharacterCreate : PacketHandler
{
#region implemented abstract members of PacketHandler
public uint CharacterID { get; set; }
public uint AccountID { get; set; }
public uint Unk1 { get; set; }
public uint VoiceType { get; set; }
public ushort Unk2 { get; set; }
public short VoicePitch { get; set; }
public string Name { get; set; }
public LooksParam Looks { get; set; }
public uint Unk3 { get; set; }
public JobParam Jobs { get; set; }
public byte[] unk4 { get; set; } = new byte[148]; // 148 字节的字节数组
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
if (context._account == null)
return;
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
Logger.WriteHex(info, data);
var reader = new PacketReader(data, position, size);
CharacterID = reader.ReadUInt32();
AccountID = reader.ReadUInt32();
Unk1 = reader.ReadUInt32();//Unk1
VoiceType = reader.ReadUInt32(); // VoiceType
Unk2 = reader.ReadUInt16(); // 5 unknown bytes
VoicePitch = reader.ReadInt16(); // VoiceData
Name = reader.ReadFixedLengthUtf16(16);
Looks = reader.ReadStruct<LooksParam>();
Unk3 = reader.ReadUInt32();
Jobs = reader.ReadStruct<JobParam>();
unk4 = reader.ReadBytes(unk4.Length);
// 创建一个 Consumable 类型的物品
PSO2ItemConsumable consumableItem = new PSO2ItemConsumable
{
amount = 10,
};
PSO2Items[] items = new PSO2Items[10];
for (var i = 0; i < 10; i++)
{
items[i] = new PSO2Items
{
uuid = (ulong)i,
id = new ItemId
{
ItemType = 3,
Id = 1,
Subid = 0,
},
data = new Items { Consumable = consumableItem }
};
}
var newCharacter = new Character
{
CharacterID = 0,
AccountID = context._account.AccountId,
Unk1 = (int)Unk1,
VoiceType = (int)VoiceType,
Unk2 = (short)Unk2,
VoicePitch = VoicePitch,
Name = Name,
Looks = Looks,
Unk3 = (int)Unk3,
Jobs = Jobs,
Unk4 = unk4,
EquipedItems = items,
Account = context._account,
Fulldata = data
};
// Add to database
using (var db = new ServerEf())
{
// Check if any characters exist for this player
var existingCharacters = db.Characters.Where(c => c.AccountID == context._account.AccountId).ToList();
if (existingCharacters.Count > 0)
{
// Increment ID if characters already exist
newCharacter.CharacterID = existingCharacters.Max(c => c.CharacterID) + 1;
}
else
{
// Start with ID 1 if no characters exist
newCharacter.CharacterID = 1;
}
Logger.Write("New character data: {0}, {1}, {2}, {3}, {4}", newCharacter.CharacterID, newCharacter.Name, newCharacter.VoiceType, newCharacter.VoicePitch, newCharacter.AccountID);
db.Characters.Add(newCharacter);
db.Entry(newCharacter.Account).State = EntityState.Modified;
db.SaveChanges();
}
// Assign character to player
context.Character = newCharacter;
// Set Accounts ID
context.SendPacket(new CharacterCreateResponsePacket(CharacterCreateResponsePacket.CharacterCreationStatus.Success, (uint)context._account.AccountId));
}
#endregion
}
}

View File

@ -1,48 +0,0 @@
using System;
using System.IO;
using PSO2SERVER.Protocol.Packets;
using PSO2SERVER.Database;
using static PSO2SERVER.Protocol.Packets.CharacterDeletionPacket;
namespace PSO2SERVER.Protocol.Handlers
{
[PacketHandlerAttr(0x11, 0x06)]
public class DeleteCharacter : PacketHandler
{
#region implemented abstract members of PacketHandler
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
{
var reader = new PacketReader(data);
var CharacterID = reader.ReadInt32();
Logger.Write("[CHR] {0} 正在删除ID {1} 的角色", context._account.Username, CharacterID);
// Delete Character
using (var db = new ServerEf())
{
foreach (var character in db.Characters)
if (character.CharacterID == CharacterID)
{
db.Characters.Remove(character);
db.ChangeTracker.DetectChanges();
break;
}
// Detect the deletion and save the Database
if (db.ChangeTracker.HasChanges())
{
db.SaveChanges();
}
}
context.SendPacket(new CharacterDeletionPacket(DeletionStatus.Success));
//context.SendPacket(new CharacterDeletionPacket(DeletionStatus.UndeletableItems));
}
#endregion
}
}

Some files were not shown because too many files have changed in this diff Show More