using System;
using System.Collections.Generic;
using System.Text;
using System.Text.RegularExpressions;
using System.IO;
using Substrate.Nbt;
using Substrate.Core;
namespace Substrate
{
///
/// Represents a single region containing 32x32 chunks.
///
public class Region : IDisposable, IChunkContainer
{
private const int XDIM = 32;
private const int ZDIM = 32;
private const int XMASK = XDIM - 1;
private const int ZMASK = ZDIM - 1;
private const int XLOG = 5;
private const int ZLOG = 5;
private int _rx;
private int _rz;
private bool _disposed = false;
private RegionManager _regionMan;
private static Regex _namePattern = new Regex("r\\.(-?[0-9]+)\\.(-?[0-9]+)\\.mca$");
private WeakReference _regionFile;
private ChunkCache _cache;
///
/// Gets the global X-coordinate of the region.
///
public int X
{
get { return _rx; }
}
///
/// Gets the global Z-coordinate of the region.
///
public int Z
{
get { return _rz; }
}
///
/// Gets the length of the X-dimension of the region in chunks.
///
public int XDim
{
get { return XDIM; }
}
///
/// Gets the length of the Z-dimension of the region in chunks.
///
public int ZDim
{
get { return ZDIM; }
}
///
/// Creates an instance of a for a given set of coordinates.
///
/// The that should be managing this region.
/// A shared cache for holding chunks.
/// The global X-coordinate of the region.
/// The global Z-coordinate of the region.
/// The constructor will not actually open or parse any region files. Given just the region coordinates, the
/// region will be able to determien the correct region file to look for based on the naming pattern for regions:
/// r.x.z.mcr, given x and z are integers representing the region's coordinates.
/// Regions require a to be provided because they do not actually store any chunks or references
/// to chunks on their own. This allows regions to easily pass off requests outside of their bounds, if necessary.
public Region (RegionManager rm, ChunkCache cache, int rx, int rz)
{
_regionMan = rm;
_cache = cache;
_regionFile = new WeakReference(null);
_rx = rx;
_rz = rz;
if (!File.Exists(GetFilePath())) {
throw new FileNotFoundException();
}
}
///
/// Creates an instance of a for the given region file.
///
/// The that should be managing this region.
/// A shared cache for holding chunks.
/// The region file to derive the region from.
/// The constructor will not actually open or parse the region file. It will only read the file's name in order
/// to derive the region's coordinates, based on a strict naming pattern for regions: r.x.z.mcr, given x and z are integers
/// representing the region's coordinates.
/// Regions require a to be provided because they do not actually store any chunks or references
/// to chunks on their own. This allows regions to easily pass off requests outside of their bounds, if necessary.
public Region (RegionManager rm, ChunkCache cache, string filename)
{
_regionMan = rm;
_cache = cache;
_regionFile = new WeakReference(null);
ParseFileName(filename, out _rx, out _rz);
if (!File.Exists(Path.Combine(_regionMan.GetRegionPath(), filename))) {
throw new FileNotFoundException();
}
}
///
/// Region finalizer that ensures any resources are cleaned up
///
~Region ()
{
Dispose(false);
}
///
/// Disposes any managed and unmanaged resources held by the region.
///
public void Dispose ()
{
Dispose(true);
System.GC.SuppressFinalize(this);
}
///
/// Conditionally dispose managed or unmanaged resources.
///
/// True if the call to Dispose was explicit.
protected virtual void Dispose (bool disposing)
{
if (!_disposed) {
if (disposing) {
// Cleanup managed resources
RegionFile rf = _regionFile.Target as RegionFile;
if (rf != null) {
rf.Dispose();
rf = null;
}
}
// Cleanup unmanaged resources
}
_disposed = true;
}
///
/// Get the appropriate filename for this region.
///
/// The filename of the region with encoded coordinates.
public string GetFileName ()
{
return "r." + _rx + "." + _rz + ".mca";
}
///
/// Tests if the given filename conforms to the general naming pattern for any region.
///
/// The filename to test.
/// True if the filename is a valid region name; false if it does not conform to the pattern.
public static bool TestFileName (string filename)
{
Match match = _namePattern.Match(filename);
if (!match.Success) {
return false;
}
return true;
}
///
/// Parses the given filename to extract encoded region coordinates.
///
/// The region filename to parse.
/// This parameter will contain the X-coordinate of a region.
/// This parameter will contain the Z-coordinate of a region.
/// True if the filename could be correctly parse; false otherwise.
public static bool ParseFileName (string filename, out int x, out int z)
{
x = 0;
z = 0;
Match match = _namePattern.Match(filename);
if (!match.Success) {
return false;
}
x = Convert.ToInt32(match.Groups[1].Value);
z = Convert.ToInt32(match.Groups[2].Value);
return true;
}
///
/// Gets the full path of the region's backing file.
///
/// Gets the path of the region's file based on the 's region path and the region's on filename.
public string GetFilePath ()
{
return System.IO.Path.Combine(_regionMan.GetRegionPath(), GetFileName());
}
private RegionFile GetRegionFile ()
{
RegionFile rf = _regionFile.Target as RegionFile;
if (rf == null) {
rf = new RegionFile(GetFilePath());
_regionFile.Target = rf;
}
return rf;
}
///
/// Gets the for a chunk given local coordinates into the region.
///
/// The local X-coordinate of a chunk within the region.
/// The local Z-coordinate of a chunk within the region.
/// An for a local chunk, or null if there is no chunk at the given coordinates.
public NbtTree GetChunkTree (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.GetChunkTree(ForeignX(lcx), ForeignZ(lcz));
}
RegionFile rf = GetRegionFile();
Stream nbtstr = rf.GetChunkDataInputStream(lcx, lcz);
if (nbtstr == null) {
return null;
}
NbtTree tree = new NbtTree(nbtstr);
nbtstr.Close();
return tree;
}
// XXX: Exceptions
///
/// Saves an for a chunk back to the region's data store at the given local coordinates.
///
/// The local X-coordinate of the chunk within the region.
/// The local Z-coordinate of the chunk within the region.
/// The of a chunk to write back to the region.
/// True if the save succeeded.
/// It is up to the programmer to ensure that the global coordinates defined within the chunk's tree
/// are consistent with the local coordinates of the region being written into.
public bool SaveChunkTree (int lcx, int lcz, NbtTree tree)
{
return SaveChunkTree(lcx, lcz, tree, null);
}
///
/// Saves an for a chunk back to the region's data store at the given local coordinates and with the given timestamp.
///
/// The local X-coordinate of the chunk within the region.
/// The local Z-coordinate of the chunk within the region.
/// The of a chunk to write back to the region.
/// The timestamp to write to the underlying region file for this chunk.
/// True if the save succeeded.
/// It is up to the programmer to ensure that the global coordinates defined within the chunk's tree
/// are consistent with the local coordinates of the region being written into.
public bool SaveChunkTree (int lcx, int lcz, NbtTree tree, int timestamp)
{
return SaveChunkTree(lcx, lcz, tree, timestamp);
}
private bool SaveChunkTree (int lcx, int lcz, NbtTree tree, int? timestamp)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? false : alt.SaveChunkTree(ForeignX(lcx), ForeignZ(lcz), tree);
}
RegionFile rf = GetRegionFile();
Stream zipstr = (timestamp == null)
? rf.GetChunkDataOutputStream(lcx, lcz)
: rf.GetChunkDataOutputStream(lcx, lcz, (int)timestamp);
if (zipstr == null) {
return false;
}
tree.WriteTo(zipstr);
zipstr.Close();
return true;
}
///
/// Gets an output stream for replacing chunk data at the given coordinates within the region.
///
/// The local X-coordinate of the chunk to replace within the region.
/// The local Z-coordinate of the chunk to replace within the region.
/// An output stream that can be written to on demand.
/// There is no guarantee that any data will be saved until the stream is closed.
public Stream GetChunkOutStream (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.GetChunkOutStream(ForeignX(lcx), ForeignZ(lcz));
}
RegionFile rf = GetRegionFile();
return rf.GetChunkDataOutputStream(lcx, lcz);
}
///
/// Returns the count of valid chunks stored in this region.
///
/// The count of currently stored chunks.
public int ChunkCount ()
{
RegionFile rf = GetRegionFile();
int count = 0;
for (int x = 0; x < XDIM; x++) {
for (int z = 0; z < ZDIM; z++) {
if (rf.HasChunk(x, z)) {
count++;
}
}
}
return count;
}
// XXX: Consider revising foreign lookup support
///
/// Gets a for a chunk at the given local coordinates relative to this region.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// A at the given local coordinates, or null if no chunk exists.
/// The local coordinates do not strictly need to be within the bounds of the region. If coordinates are detected
/// as being out of bounds, the lookup will be delegated to the correct region and the lookup will be performed there
/// instead. This allows any to perform a similar task to , but with a
/// region-local frame of reference instead of a global frame of reference.
public ChunkRef GetChunkRef (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.GetChunkRef(ForeignX(lcx), ForeignZ(lcz));
}
int cx = lcx + _rx * XDIM;
int cz = lcz + _rz * ZDIM;
ChunkKey k = new ChunkKey(cx, cz);
ChunkRef c = _cache.Fetch(k);
if (c != null) {
return c;
}
c = ChunkRef.Create(this, lcx, lcz);
if (c != null) {
_cache.Insert(c);
}
return c;
}
///
/// Creates a new chunk at the given local coordinates relative to this region and returns a new for it.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// A for the newly created chunk.
/// If the local coordinates are out of bounds for this region, the action will be forwarded to the correct region
/// transparently.
public ChunkRef CreateChunk (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.CreateChunk(ForeignX(lcx), ForeignZ(lcz));
}
DeleteChunk(lcx, lcz);
int cx = lcx + _rx * XDIM;
int cz = lcz + _rz * ZDIM;
Chunk c = Chunk.Create(cx, cz);
c.Save(GetChunkOutStream(lcx, lcz));
ChunkRef cr = ChunkRef.Create(this, lcx, lcz);
_cache.Insert(cr);
return cr;
}
#region IChunkCollection Members
// XXX: This also feels dirty.
///
/// Gets the global X-coordinate of a chunk given an internal coordinate handed out by a container.
///
/// An internal X-coordinate given to a by any instance of a container.
/// The global X-coordinate of the corresponding chunk.
public int ChunkGlobalX (int cx)
{
return _rx * XDIM + cx;
}
///
/// Gets the global Z-coordinate of a chunk given an internal coordinate handed out by a container.
///
/// An internal Z-coordinate given to a by any instance of a container.
/// The global Z-coordinate of the corresponding chunk.
public int ChunkGlobalZ (int cz)
{
return _rz * ZDIM + cz;
}
///
/// Gets the region-local X-coordinate of a chunk given an internal coordinate handed out by a container.
///
/// An internal X-coordinate given to a by any instance of a container.
/// The region-local X-coordinate of the corresponding chunk.
public int ChunkLocalX (int cx)
{
return cx;
}
///
/// Gets the region-local Z-coordinate of a chunk given an internal coordinate handed out by a container.
///
/// An internal Z-coordinate given to a by any instance of a container.
/// The region-local Z-coordinate of the corresponding chunk.
public int ChunkLocalZ (int cz)
{
return cz;
}
///
/// Returns a given local coordinates relative to this region.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// A object for the given coordinates, or null if the chunk does not exist.
/// If the local coordinates are out of bounds for this region, the action will be forwarded to the correct region
/// transparently. The returned object may either come from cache, or be regenerated from disk.
public Chunk GetChunk (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.GetChunk(ForeignX(lcx), ForeignZ(lcz));
}
if (!ChunkExists(lcx, lcz)) {
return null;
}
return Chunk.CreateVerified(GetChunkTree(lcx, lcz));
}
///
/// Checks if a chunk exists at the given local coordinates relative to this region.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// True if there is a chunk at the given coordinates; false otherwise.
/// If the local coordinates are out of bounds for this region, the action will be forwarded to the correct region
/// transparently.
public bool ChunkExists (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? false : alt.ChunkExists(ForeignX(lcx), ForeignZ(lcz));
}
RegionFile rf = GetRegionFile();
return rf.HasChunk(lcx, lcz);
}
///
/// Deletes a chunk from the underlying data store at the given local coordinates relative to this region.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// True if there is a chunk was deleted; false otherwise.
/// If the local coordinates are out of bounds for this region, the action will be forwarded to the correct region
/// transparently.
public bool DeleteChunk (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? false : alt.DeleteChunk(ForeignX(lcx), ForeignZ(lcz));
}
RegionFile rf = GetRegionFile();
if (!rf.HasChunk(lcx, lcz)) {
return false;
}
rf.DeleteChunk(lcx, lcz);
ChunkKey k = new ChunkKey(ChunkGlobalX(lcx), ChunkGlobalZ(lcz));
_cache.Remove(k);
if (ChunkCount() == 0) {
_regionMan.DeleteRegion(X, Z);
_regionFile.Target = null;
}
return true;
}
///
/// Saves an existing to the region at the given local coordinates.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// A to save to the given location.
/// A represneting the at its new location.
/// If the local coordinates are out of bounds for this region, the action will be forwarded to the correct region
/// transparently. The 's internal global coordinates will be updated to reflect the new location.
public ChunkRef SetChunk (int lcx, int lcz, Chunk chunk)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? null : alt.CreateChunk(ForeignX(lcx), ForeignZ(lcz));
}
DeleteChunk(lcx, lcz);
int cx = lcx + _rx * XDIM;
int cz = lcz + _rz * ZDIM;
chunk.SetLocation(cx, cz);
chunk.Save(GetChunkOutStream(lcx, lcz));
ChunkRef cr = ChunkRef.Create(this, lcx, lcz);
_cache.Insert(cr);
return cr;
}
///
/// Saves all chunks within this region that have been marked as dirty.
///
/// The number of chunks that were saved.
public int Save ()
{
_cache.SyncDirty();
int saved = 0;
IEnumerator en = _cache.GetDirtyEnumerator();
while (en.MoveNext()) {
ChunkRef chunk = en.Current;
if (!ChunkExists(chunk.LocalX, chunk.LocalZ)) {
throw new MissingChunkException();
}
if (chunk.Save(GetChunkOutStream(chunk.LocalX, chunk.LocalZ))) {
saved++;
}
}
_cache.ClearDirty();
return saved;
}
// XXX: Allows a chunk not part of this region to be saved to it
///
public bool SaveChunk (Chunk chunk)
{
//Console.WriteLine("Region[{0}, {1}].Save({2}, {3})", _rx, _rz, ForeignX(chunk.X),ForeignZ(chunk.Z));
return chunk.Save(GetChunkOutStream(ForeignX(chunk.X), ForeignZ(chunk.Z)));
}
///
/// Checks if this container supports delegating an action on out-of-bounds coordinates to another container.
///
public bool CanDelegateCoordinates
{
get { return true; }
}
///
/// Gets the timestamp of a chunk from the underlying region file.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// The timestamp of the chunk slot in the region.
/// The value returned may differ from any timestamp stored in the chunk data itself.
public int GetChunkTimestamp (int lcx, int lcz)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
return (alt == null) ? 0 : alt.GetChunkTimestamp(ForeignX(lcx), ForeignZ(lcz));
}
RegionFile rf = GetRegionFile();
return rf.GetTimestamp(lcx, lcz);
}
///
/// Sets the timestamp of a chunk in the underlying region file.
///
/// The local X-coordinate of a chunk relative to this region.
/// The local Z-coordinate of a chunk relative to this region.
/// The new timestamp value.
/// This function will only update the timestamp of the chunk slot in the underlying region file. It will not update
/// any timestamp information in the chunk data itself.
public void SetChunkTimestamp (int lcx, int lcz, int timestamp)
{
if (!LocalBoundsCheck(lcx, lcz)) {
Region alt = GetForeignRegion(lcx, lcz);
if (alt != null)
alt.SetChunkTimestamp(ForeignX(lcx), ForeignZ(lcz), timestamp);
}
RegionFile rf = GetRegionFile();
rf.SetTimestamp(lcx, lcz, timestamp);
}
#endregion
private bool LocalBoundsCheck (int lcx, int lcz)
{
return (lcx >= 0 && lcx < XDIM && lcz >= 0 && lcz < ZDIM);
}
private Region GetForeignRegion (int lcx, int lcz)
{
return _regionMan.GetRegion(_rx + (lcx >> XLOG), _rz + (lcz >> ZLOG));
}
private int ForeignX (int lcx)
{
return (lcx + XDIM * 10000) & XMASK;
}
private int ForeignZ (int lcz)
{
return (lcz + ZDIM * 10000) & ZMASK;
}
}
}