/*
TShock, a server mod for Terraria
Copyright (C) 2011 The TShock Team
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Security.Cryptography;
using System.Text;
using Terraria;
namespace TShockAPI
{
public class Utils
{
private readonly static int firstItemPrefix = 1;
private readonly static int lastItemPrefix = 83;
// Utils is a Singleton
private static readonly Utils instance = new Utils();
private Utils() {}
public static Utils Instance { get { return instance; } }
public Random Random = new Random();
//private static List groups = new List();
///
/// Provides the real IP address from a RemoteEndPoint string that contains a port and an IP
///
/// A string IPv4 address in IP:PORT form.
/// A string IPv4 address.
public string GetRealIP(string mess)
{
return mess.Split(':')[0];
}
///
/// Used for some places where a list of players might be used.
///
/// String of players seperated by commas.
public string GetPlayers()
{
var sb = new StringBuilder();
foreach (TSPlayer player in TShock.Players)
{
if (player != null && player.Active)
{
if (sb.Length != 0)
{
sb.Append(", ");
}
sb.Append(player.Name);
}
}
return sb.ToString();
}
///
/// Used for some places where a list of players might be used.
///
/// String of players and their id seperated by commas.
public string GetPlayersWithIds()
{
var sb = new StringBuilder();
foreach (TSPlayer player in TShock.Players)
{
if (player != null && player.Active)
{
if (sb.Length != 0)
{
sb.Append(", ");
}
sb.Append(player.Name);
string id = "( " + Convert.ToString(TShock.Users.GetUserID(player.UserAccountName)) + " )";
sb.Append(id);
}
}
return sb.ToString();
}
///
/// Finds a player and gets IP as string
///
/// Player name
public string GetPlayerIP(string playername)
{
foreach (TSPlayer player in TShock.Players)
{
if (player != null && player.Active)
{
if (playername.ToLower() == player.Name.ToLower())
{
return player.IP;
}
}
}
return null;
}
///
/// It's a clamp function
///
///
/// Value to clamp
/// Maximum bounds of the clamp
/// Minimum bounds of the clamp
///
public T Clamp(T value, T max, T min)
where T : IComparable
{
T result = value;
if (value.CompareTo(max) > 0)
result = max;
if (value.CompareTo(min) < 0)
result = min;
return result;
}
///
/// Saves the map data
///
public void SaveWorld()
{
SaveManager.Instance.SaveWorld();
}
///
/// Broadcasts a message to all players
///
/// string message
public void Broadcast(string msg)
{
Broadcast(msg, Color.Green);
}
public void Broadcast(string msg, byte red, byte green, byte blue)
{
TSPlayer.All.SendMessage(msg, red, green, blue);
TSPlayer.Server.SendMessage(msg, red, green, blue);
Log.Info(string.Format("Broadcast: {0}", msg));
}
public void Broadcast(string msg, Color color)
{
Broadcast(msg, color.R, color.G, color.B);
}
///
/// Sends message to all users with 'logs' permission.
///
/// Message to send
/// Color of the message
public void SendLogs(string log, Color color)
{
Log.Info(log);
TSPlayer.Server.SendMessage(log, color);
foreach (TSPlayer player in TShock.Players)
{
if (player != null && player.Active && player.Group.HasPermission(Permissions.logs) && player.DisplayLogs &&
TShock.Config.DisableSpewLogs == false)
player.SendMessage(log, color);
}
}
///
/// The number of active players on the server.
///
/// int playerCount
public int ActivePlayers()
{
return Main.player.Where(p => null != p && p.active).Count();
}
///
/// Finds a player ID based on name
///
/// Player name
///
public List FindPlayer(string ply)
{
var found = new List();
// Avoid errors caused by null search
if (null == ply)
return found;
ply = ply.ToLower();
foreach (TSPlayer player in TShock.Players)
{
if (player == null)
continue;
string name = player.Name.ToLower();
if (name.Equals(ply))
return new List {player};
if (name.Contains(ply))
found.Add(player);
}
return found;
}
///
/// Gets a random clear tile in range
///
/// Bound X
/// Bound Y
/// Range on the X axis
/// Range on the Y axis
/// X location
/// Y location
public void GetRandomClearTileWithInRange(int startTileX, int startTileY, int tileXRange, int tileYRange,
out int tileX, out int tileY)
{
int j = 0;
do
{
if (j == 100)
{
tileX = startTileX;
tileY = startTileY;
break;
}
tileX = startTileX + Random.Next(tileXRange*-1, tileXRange);
tileY = startTileY + Random.Next(tileYRange*-1, tileYRange);
j++;
} while (TileValid(tileX, tileY) && !TileClear(tileX, tileY));
}
///
/// Determines if a tile is valid
///
/// Location X
/// Location Y
/// If the tile is valid
private bool TileValid(int tileX, int tileY)
{
return tileX >= 0 && tileX <= Main.maxTilesX && tileY >= 0 && tileY <= Main.maxTilesY;
}
///
/// Clears a tile
///
/// Location X
/// Location Y
/// The state of the tile
private bool TileClear(int tileX, int tileY)
{
return !Main.tile[tileX, tileY].active;
}
///
/// Gets a list of items by ID or name
///
/// Item ID or name
/// List of Items
public List- GetItemByIdOrName(string idOrName)
{
int type = -1;
if (int.TryParse(idOrName, out type))
{
return new List
- {GetItemById(type)};
}
return GetItemByName(idOrName);
}
///
/// Gets an item by ID
///
/// ID
/// Item
public Item GetItemById(int id)
{
Item item = new Item();
item.netDefaults(id);
return item;
}
///
/// Gets items by name
///
/// name
/// List of Items
public List
- GetItemByName(string name)
{
//Method #1 - must be exact match, allows support for different pickaxes/hammers/swords etc
for (int i = 1; i < Main.maxItemTypes; i++)
{
Item item = new Item();
item.SetDefaults(name);
if (item.name == name)
return new List
- {item};
}
//Method #2 - allows impartial matching
var found = new List
- ();
for (int i = -24; i < Main.maxItemTypes; i++)
{
try
{
Item item = new Item();
item.netDefaults(i);
if (item.name.ToLower() == name.ToLower())
return new List
- {item};
if (item.name.ToLower().StartsWith(name.ToLower()))
found.Add(item);
}
catch
{
}
}
return found;
}
///
/// Gets an NPC by ID or Name
///
///
/// List of NPCs
public List GetNPCByIdOrName(string idOrName)
{
int type = -1;
if (int.TryParse(idOrName, out type))
{
return new List {GetNPCById(type)};
}
return GetNPCByName(idOrName);
}
///
/// Gets an NPC by ID
///
/// ID
/// NPC
public NPC GetNPCById(int id)
{
NPC npc = new NPC();
npc.netDefaults(id);
return npc;
}
///
/// Gets a NPC by name
///
/// Name
/// List of matching NPCs
public List GetNPCByName(string name)
{
//Method #1 - must be exact match, allows support for different coloured slimes
for (int i = -17; i < Main.maxNPCTypes; i++)
{
NPC npc = new NPC();
npc.SetDefaults(name);
if (npc.name == name)
return new List {npc};
}
//Method #2 - allows impartial matching
var found = new List();
for (int i = 1; i < Main.maxNPCTypes; i++)
{
NPC npc = new NPC();
npc.netDefaults(i);
if (npc.name.ToLower() == name.ToLower())
return new List {npc};
if (npc.name.ToLower().StartsWith(name.ToLower()))
found.Add(npc);
}
return found;
}
///
/// Gets a buff name by id
///
/// ID
/// name
public string GetBuffName(int id)
{
return (id > 0 && id < Main.maxBuffs) ? Main.buffName[id] : "null";
}
///
/// Gets the description of a buff
///
/// ID
/// description
public string GetBuffDescription(int id)
{
return (id > 0 && id < Main.maxBuffs) ? Main.buffTip[id] : "null";
}
///
/// Gets a list of buffs by name
///
/// name
/// Matching list of buff ids
public List GetBuffByName(string name)
{
for (int i = 1; i < Main.maxBuffs; i++)
{
if (Main.buffName[i].ToLower() == name)
return new List {i};
}
var found = new List();
for (int i = 1; i < Main.maxBuffs; i++)
{
if (Main.buffName[i].ToLower().StartsWith(name.ToLower()))
found.Add(i);
}
return found;
}
///
/// Gets a prefix based on its id
///
/// ID
/// Prefix name
public string GetPrefixById(int id)
{
var item = new Item();
item.SetDefaults(0);
item.prefix = (byte) id;
item.AffixName();
return item.name.Trim();
}
///
/// Gets a list of prefixes by name
///
/// Name
/// List of prefix IDs
public List GetPrefixByName(string name)
{
Item item = new Item();
item.SetDefaults(0);
string lowerName = name.ToLower();
var found = new List();
for (int i = firstItemPrefix; i <= lastItemPrefix; i++)
{
try
{
item.prefix = (byte)i;
string trimmed = item.AffixName().Trim();
if (trimmed == name)
{
// Exact match
found.Add(i);
return found;
}
else
{
string trimmedLower = trimmed.ToLower();
if (trimmedLower == lowerName)
{
// Exact match (caseinsensitive)
found.Add(i);
return found;
}
else if (trimmedLower.StartsWith(lowerName)) // Partial match
found.Add(i);
}
}
catch
{
}
}
return found;
}
///
/// Gets a prefix by ID or name
///
/// ID or name
/// List of prefix IDs
public List GetPrefixByIdOrName(string idOrName)
{
int type = -1;
if (int.TryParse(idOrName, out type) && type >= firstItemPrefix && type <= lastItemPrefix)
{
return new List {type};
}
return GetPrefixByName(idOrName);
}
///
/// Kicks all player from the server without checking for immunetokick permission.
///
/// int player
/// string reason
public void ForceKickAll(string reason)
{
foreach (TSPlayer player in TShock.Players)
{
if (player != null && player.Active)
{
ForceKick(player, reason);
}
}
}
///
/// Stops the server after kicking all players with a reason message, and optionally saving the world
///
/// bool perform a world save before stop (default: true)
/// string reason (default: "Server shutting down!")
public void StopServer(bool save = true, string reason = "Server shutting down!")
{
ForceKickAll(reason);
if (save)
SaveManager.Instance.SaveWorld();
// Save takes a while so kick again
ForceKickAll(reason);
// Broadcast so console can see we are shutting down as well
TShock.Utils.Broadcast(reason, Color.Red);
// Disconnect after kick as that signifies server is exiting and could cause a race
Netplay.disconnect = true;
}
#if COMPAT_SIGS
[Obsolete("This method is for signature compatibility for external code only")]
public bool ForceKick(TSPlayer player, string reason)
{
return Kick(player, reason, true, false, string.Empty);
}
#endif
///
/// Kicks a player from the server without checking for immunetokick permission.
///
/// int player
/// string reason
/// bool silent (default: false)
public void ForceKick(TSPlayer player, string reason, bool silent = false)
{
Kick(player, reason, true, silent);
}
#if COMPAT_SIGS
[Obsolete("This method is for signature compatibility for external code only")]
public bool Kick(TSPlayer player, string reason, string adminUserName)
{
return Kick(player, reason, false, false, adminUserName);
}
#endif
///
/// Kicks a player from the server.
///
/// int player
/// string reason
/// bool force (default: false)
/// bool silent (default: false)
/// bool silent (default: null)
public bool Kick(TSPlayer player, string reason, bool force = false, bool silent = false, string adminUserName = null)
{
if (!player.ConnectionAlive)
return true;
if (force || !player.Group.HasPermission(Permissions.immunetokick))
{
string playerName = player.Name;
player.SilentKickInProgress = silent;
player.Disconnect(string.Format("Kicked: {0}", reason));
Log.ConsoleInfo(string.Format("Kicked {0} for : {1}", playerName, reason));
string verb = force ? "force " : "";
if (string.IsNullOrWhiteSpace(adminUserName))
Broadcast(string.Format("{0} was {1}kicked for {2}", playerName, verb, reason.ToLower()));
else
Broadcast(string.Format("{0} {1}kicked {2} for {3}", adminUserName, verb, playerName, reason.ToLower()));
return true;
}
return false;
}
#if COMPAT_SIGS
[Obsolete("This method is for signature compatibility for external code only")]
public bool Ban(TSPlayer player, string reason, string adminUserName)
{
return Ban(player, reason, false, adminUserName);
}
#endif
///
/// Bans and kicks a player from the server.
///
/// int player
/// string reason
/// bool force (default: false)
/// bool silent (default: null)
public bool Ban(TSPlayer player, string reason, bool force = false, string adminUserName = null)
{
if (!player.ConnectionAlive)
return true;
if (force || !player.Group.HasPermission(Permissions.immunetoban))
{
string ip = player.IP;
string playerName = player.Name;
TShock.Bans.AddBan(ip, playerName, reason);
player.Disconnect(string.Format("Banned: {0}", reason));
Log.ConsoleInfo(string.Format("Banned {0} for : {1}", playerName, reason));
string verb = force ? "force " : "";
if (string.IsNullOrWhiteSpace(adminUserName))
Broadcast(string.Format("{0} was {1}banned for {1}", playerName, verb, reason.ToLower()));
else
Broadcast(string.Format("{0} {1}banned {1} for {2}", adminUserName, verb, playerName, reason.ToLower()));
return true;
}
return false;
}
///
/// Shows a file to the user.
///
/// int player
/// string filename reletave to savedir
//Todo: Fix this
public void ShowFileToUser(TSPlayer player, string file)
{
string foo = "";
using (var tr = new StreamReader(Path.Combine(TShock.SavePath, file)))
{
while ((foo = tr.ReadLine()) != null)
{
foo = foo.Replace("%map%", Main.worldName);
foo = foo.Replace("%players%", GetPlayers());
//foo = SanitizeString(foo);
if (foo.Substring(0, 1) == "%" && foo.Substring(12, 1) == "%") //Look for a beginning color code.
{
string possibleColor = foo.Substring(0, 13);
foo = foo.Remove(0, 13);
float[] pC = {0, 0, 0};
possibleColor = possibleColor.Replace("%", "");
string[] pCc = possibleColor.Split(',');
if (pCc.Length == 3)
{
try
{
player.SendMessage(foo, (byte) Convert.ToInt32(pCc[0]), (byte) Convert.ToInt32(pCc[1]),
(byte) Convert.ToInt32(pCc[2]));
continue;
}
catch (Exception e)
{
Log.Error(e.ToString());
}
}
}
player.SendMessage(foo);
}
}
}
///
/// Returns a Group from the name of the group
///
/// string groupName
public Group GetGroup(string groupName)
{
//first attempt on cached groups
for (int i = 0; i < TShock.Groups.groups.Count; i++)
{
if (TShock.Groups.groups[i].Name.Equals(groupName))
{
return TShock.Groups.groups[i];
}
}
return new Group(TShock.Config.DefaultGuestGroupName);
}
///
/// Returns an IPv4 address from a DNS query
///
/// string ip
public string GetIPv4Address(string hostname)
{
try
{
//Get the ipv4 address from GetHostAddresses, if an ip is passed it will return that ip
var ip = Dns.GetHostAddresses(hostname).FirstOrDefault(i => i.AddressFamily == AddressFamily.InterNetwork);
//if the dns query was successful then return it, otherwise return an empty string
return ip != null ? ip.ToString() : "";
}
catch (SocketException)
{
}
return "";
}
public string HashAlgo = "sha512";
public readonly Dictionary> HashTypes = new Dictionary>
{
{"sha512", () => new SHA512Managed()},
{"sha256", () => new SHA256Managed()},
{"md5", () => new MD5Cng()},
{"sha512-xp", () => SHA512.Create()},
{"sha256-xp", () => SHA256.Create()},
{"md5-xp", () => MD5.Create()},
};
///
/// Returns a Sha256 string for a given string
///
/// bytes to hash
/// string sha256
public string HashPassword(byte[] bytes)
{
if (bytes == null)
throw new NullReferenceException("bytes");
Func func;
if (!HashTypes.TryGetValue(HashAlgo.ToLower(), out func))
throw new NotSupportedException("Hashing algorithm {0} is not supported".SFormat(HashAlgo.ToLower()));
using (var hash = func())
{
var ret = hash.ComputeHash(bytes);
return ret.Aggregate("", (s, b) => s + b.ToString("X2"));
}
}
///
/// Returns a Sha256 string for a given string
///
/// bytes to hash
/// string sha256
public string HashPassword(string password)
{
if (string.IsNullOrEmpty(password) || password == "non-existant password")
return "non-existant password";
return HashPassword(Encoding.UTF8.GetBytes(password));
}
///
/// Checks if the string contains any unprintable characters
///
/// String to check
/// True if the string only contains printable characters
public bool ValidString(string str)
{
foreach (var c in str)
{
if (c < 0x20 || c > 0xA9)
return false;
}
return true;
}
///
/// Checks if world has hit the max number of chests
///
/// True if the entire chest array is used
public bool MaxChests()
{
for (int i = 0; i < Main.chest.Length; i++)
{
if (Main.chest[i] == null)
return false;
}
return true;
}
///
/// Searches for a projectile by identity and owner
///
/// identity
/// owner
/// projectile ID
public int SearchProjectile(short identity, int owner)
{
for (int i = 0; i < Main.maxProjectiles; i++)
{
if (Main.projectile[i].identity == identity && Main.projectile[i].owner == owner)
return i;
}
return 1000;
}
///
/// Sanitizes input strings
///
/// string
/// sanitized string
public string SanitizeString(string str)
{
var returnstr = str.ToCharArray();
for (int i = 0; i < str.Length; i++)
{
if (!ValidString(str[i].ToString()))
returnstr[i] = ' ';
}
return new string(returnstr);
}
}
}