diff --git a/OpenSim/Client/MXP/ClientStack/MXPClientView.cs b/OpenSim/Client/MXP/ClientStack/MXPClientView.cs new file mode 100644 index 0000000000..4a102e9278 --- /dev/null +++ b/OpenSim/Client/MXP/ClientStack/MXPClientView.cs @@ -0,0 +1,1158 @@ +using System; +using System.Collections.Generic; +using System.Net; +using System.Reflection; +using System.Text; +using log4net; +using MXP; +using MXP.Messages; +using OpenMetaverse; +using OpenMetaverse.Packets; +using OpenSim.Framework; +using OpenSim.Framework.Client; +using Packet=OpenMetaverse.Packets.Packet; + +namespace OpenSim.Client.MXP.ClientStack +{ + class MXPClientView : IClientAPI, IClientCore + { + internal static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + + private readonly Session mxpSession; + private readonly UUID mxpSessionID; + private readonly IScene mxpHostBubble; + private readonly string mxpUsername; + + private int debugLevel; + + public MXPClientView(Session mxpSession, UUID mxpSessionID, IScene mxpHostBubble, string mxpUsername) + { + this.mxpSession = mxpSession; + this.mxpUsername = mxpUsername; + this.mxpHostBubble = mxpHostBubble; + this.mxpSessionID = mxpSessionID; + } + + public Session Session + { + get { return mxpSession; } + } + + public bool ProcessMXPPacket(Message msg) + { + if (debugLevel > 0) + m_log.Warn("[MXP] Got Action/Command Packet: " + msg); + + return false; + } + + #region IClientAPI + + public Vector3 StartPos + { + get { return new Vector3(128f, 128f, 128f); } + set { } // TODO: Implement Me + } + + public UUID AgentId + { + get { return mxpSessionID; } + } + + public UUID SessionId + { + get { return mxpSessionID; } + } + + public UUID SecureSessionId + { + get { return mxpSessionID; } + } + + public UUID ActiveGroupId + { + get { return UUID.Zero; } + } + + public string ActiveGroupName + { + get { return ""; } + } + + public ulong ActiveGroupPowers + { + get { return 0; } + } + + public ulong GetGroupPowers(UUID groupID) + { + return 0; + } + + public bool IsGroupMember(UUID GroupID) + { + return false; + } + + public string FirstName + { + get { return mxpUsername; } + } + + public string LastName + { + get { return "@mxp://" + Session.RemoteEndPoint.Address; } + } + + public IScene Scene + { + get { return mxpHostBubble; } + } + + public int NextAnimationSequenceNumber + { + get { return 0; } + } + + public string Name + { + get { return FirstName; } + } + + public bool IsActive + { + get { return Session.SessionState == SessionState.Connected; } + set + { + if (!value) + Stop(); + } + } + + // Do we need this? + public bool SendLogoutPacketWhenClosing + { + set { } + } + + public uint CircuitCode + { + get { return mxpSessionID.CRC(); } + } + + public event GenericMessage OnGenericMessage; + public event ImprovedInstantMessage OnInstantMessage; + public event ChatMessage OnChatFromClient; + public event TextureRequest OnRequestTexture; + public event RezObject OnRezObject; + public event ModifyTerrain OnModifyTerrain; + public event BakeTerrain OnBakeTerrain; + public event EstateChangeInfo OnEstateChangeInfo; + public event SetAppearance OnSetAppearance; + public event AvatarNowWearing OnAvatarNowWearing; + public event RezSingleAttachmentFromInv OnRezSingleAttachmentFromInv; + public event UUIDNameRequest OnDetachAttachmentIntoInv; + public event ObjectAttach OnObjectAttach; + public event ObjectDeselect OnObjectDetach; + public event ObjectDrop OnObjectDrop; + public event StartAnim OnStartAnim; + public event StopAnim OnStopAnim; + public event LinkObjects OnLinkObjects; + public event DelinkObjects OnDelinkObjects; + public event RequestMapBlocks OnRequestMapBlocks; + public event RequestMapName OnMapNameRequest; + public event TeleportLocationRequest OnTeleportLocationRequest; + public event DisconnectUser OnDisconnectUser; + public event RequestAvatarProperties OnRequestAvatarProperties; + public event SetAlwaysRun OnSetAlwaysRun; + public event TeleportLandmarkRequest OnTeleportLandmarkRequest; + public event DeRezObject OnDeRezObject; + public event Action OnRegionHandShakeReply; + public event GenericCall2 OnRequestWearables; + public event GenericCall2 OnCompleteMovementToRegion; + public event UpdateAgent OnAgentUpdate; + public event AgentRequestSit OnAgentRequestSit; + public event AgentSit OnAgentSit; + public event AvatarPickerRequest OnAvatarPickerRequest; + public event Action OnRequestAvatarsData; + public event AddNewPrim OnAddPrim; + public event FetchInventory OnAgentDataUpdateRequest; + public event TeleportLocationRequest OnSetStartLocationRequest; + public event RequestGodlikePowers OnRequestGodlikePowers; + public event GodKickUser OnGodKickUser; + public event ObjectDuplicate OnObjectDuplicate; + public event ObjectDuplicateOnRay OnObjectDuplicateOnRay; + public event GrabObject OnGrabObject; + public event ObjectSelect OnDeGrabObject; + public event MoveObject OnGrabUpdate; + public event UpdateShape OnUpdatePrimShape; + public event ObjectExtraParams OnUpdateExtraParams; + public event ObjectSelect OnObjectSelect; + public event ObjectDeselect OnObjectDeselect; + public event GenericCall7 OnObjectDescription; + public event GenericCall7 OnObjectName; + public event GenericCall7 OnObjectClickAction; + public event GenericCall7 OnObjectMaterial; + public event RequestObjectPropertiesFamily OnRequestObjectPropertiesFamily; + public event UpdatePrimFlags OnUpdatePrimFlags; + public event UpdatePrimTexture OnUpdatePrimTexture; + public event UpdateVector OnUpdatePrimGroupPosition; + public event UpdateVector OnUpdatePrimSinglePosition; + public event UpdatePrimRotation OnUpdatePrimGroupRotation; + public event UpdatePrimSingleRotation OnUpdatePrimSingleRotation; + public event UpdatePrimGroupRotation OnUpdatePrimGroupMouseRotation; + public event UpdateVector OnUpdatePrimScale; + public event UpdateVector OnUpdatePrimGroupScale; + public event StatusChange OnChildAgentStatus; + public event GenericCall2 OnStopMovement; + public event Action OnRemoveAvatar; + public event ObjectPermissions OnObjectPermissions; + public event CreateNewInventoryItem OnCreateNewInventoryItem; + public event CreateInventoryFolder OnCreateNewInventoryFolder; + public event UpdateInventoryFolder OnUpdateInventoryFolder; + public event MoveInventoryFolder OnMoveInventoryFolder; + public event FetchInventoryDescendents OnFetchInventoryDescendents; + public event PurgeInventoryDescendents OnPurgeInventoryDescendents; + public event FetchInventory OnFetchInventory; + public event RequestTaskInventory OnRequestTaskInventory; + public event UpdateInventoryItem OnUpdateInventoryItem; + public event CopyInventoryItem OnCopyInventoryItem; + public event MoveInventoryItem OnMoveInventoryItem; + public event RemoveInventoryFolder OnRemoveInventoryFolder; + public event RemoveInventoryItem OnRemoveInventoryItem; + public event UDPAssetUploadRequest OnAssetUploadRequest; + public event XferReceive OnXferReceive; + public event RequestXfer OnRequestXfer; + public event ConfirmXfer OnConfirmXfer; + public event AbortXfer OnAbortXfer; + public event RezScript OnRezScript; + public event UpdateTaskInventory OnUpdateTaskInventory; + public event MoveTaskInventory OnMoveTaskItem; + public event RemoveTaskInventory OnRemoveTaskItem; + public event RequestAsset OnRequestAsset; + public event UUIDNameRequest OnNameFromUUIDRequest; + public event ParcelAccessListRequest OnParcelAccessListRequest; + public event ParcelAccessListUpdateRequest OnParcelAccessListUpdateRequest; + public event ParcelPropertiesRequest OnParcelPropertiesRequest; + public event ParcelDivideRequest OnParcelDivideRequest; + public event ParcelJoinRequest OnParcelJoinRequest; + public event ParcelPropertiesUpdateRequest OnParcelPropertiesUpdateRequest; + public event ParcelSelectObjects OnParcelSelectObjects; + public event ParcelObjectOwnerRequest OnParcelObjectOwnerRequest; + public event ParcelAbandonRequest OnParcelAbandonRequest; + public event ParcelGodForceOwner OnParcelGodForceOwner; + public event ParcelReclaim OnParcelReclaim; + public event ParcelReturnObjectsRequest OnParcelReturnObjectsRequest; + public event RegionInfoRequest OnRegionInfoRequest; + public event EstateCovenantRequest OnEstateCovenantRequest; + public event FriendActionDelegate OnApproveFriendRequest; + public event FriendActionDelegate OnDenyFriendRequest; + public event FriendshipTermination OnTerminateFriendship; + public event MoneyTransferRequest OnMoneyTransferRequest; + public event EconomyDataRequest OnEconomyDataRequest; + public event MoneyBalanceRequest OnMoneyBalanceRequest; + public event UpdateAvatarProperties OnUpdateAvatarProperties; + public event ParcelBuy OnParcelBuy; + public event RequestPayPrice OnRequestPayPrice; + public event ObjectSaleInfo OnObjectSaleInfo; + public event ObjectBuy OnObjectBuy; + public event BuyObjectInventory OnBuyObjectInventory; + public event RequestTerrain OnRequestTerrain; + public event RequestTerrain OnUploadTerrain; + public event ObjectIncludeInSearch OnObjectIncludeInSearch; + public event UUIDNameRequest OnTeleportHomeRequest; + public event ScriptAnswer OnScriptAnswer; + public event AgentSit OnUndo; + public event ForceReleaseControls OnForceReleaseControls; + public event GodLandStatRequest OnLandStatRequest; + public event DetailedEstateDataRequest OnDetailedEstateDataRequest; + public event SetEstateFlagsRequest OnSetEstateFlagsRequest; + public event SetEstateTerrainBaseTexture OnSetEstateTerrainBaseTexture; + public event SetEstateTerrainDetailTexture OnSetEstateTerrainDetailTexture; + public event SetEstateTerrainTextureHeights OnSetEstateTerrainTextureHeights; + public event CommitEstateTerrainTextureRequest OnCommitEstateTerrainTextureRequest; + public event SetRegionTerrainSettings OnSetRegionTerrainSettings; + public event EstateRestartSimRequest OnEstateRestartSimRequest; + public event EstateChangeCovenantRequest OnEstateChangeCovenantRequest; + public event UpdateEstateAccessDeltaRequest OnUpdateEstateAccessDeltaRequest; + public event SimulatorBlueBoxMessageRequest OnSimulatorBlueBoxMessageRequest; + public event EstateBlueBoxMessageRequest OnEstateBlueBoxMessageRequest; + public event EstateDebugRegionRequest OnEstateDebugRegionRequest; + public event EstateTeleportOneUserHomeRequest OnEstateTeleportOneUserHomeRequest; + public event EstateTeleportAllUsersHomeRequest OnEstateTeleportAllUsersHomeRequest; + public event UUIDNameRequest OnUUIDGroupNameRequest; + public event RegionHandleRequest OnRegionHandleRequest; + public event ParcelInfoRequest OnParcelInfoRequest; + public event RequestObjectPropertiesFamily OnObjectGroupRequest; + public event ScriptReset OnScriptReset; + public event GetScriptRunning OnGetScriptRunning; + public event SetScriptRunning OnSetScriptRunning; + public event UpdateVector OnAutoPilotGo; + public event TerrainUnacked OnUnackedTerrain; + public event ActivateGesture OnActivateGesture; + public event DeactivateGesture OnDeactivateGesture; + public event ObjectOwner OnObjectOwner; + public event DirPlacesQuery OnDirPlacesQuery; + public event DirFindQuery OnDirFindQuery; + public event DirLandQuery OnDirLandQuery; + public event DirPopularQuery OnDirPopularQuery; + public event DirClassifiedQuery OnDirClassifiedQuery; + public event EventInfoRequest OnEventInfoRequest; + public event ParcelSetOtherCleanTime OnParcelSetOtherCleanTime; + public event MapItemRequest OnMapItemRequest; + public event OfferCallingCard OnOfferCallingCard; + public event AcceptCallingCard OnAcceptCallingCard; + public event DeclineCallingCard OnDeclineCallingCard; + public event SoundTrigger OnSoundTrigger; + public event StartLure OnStartLure; + public event TeleportLureRequest OnTeleportLureRequest; + public event NetworkStats OnNetworkStatsUpdate; + public event ClassifiedInfoRequest OnClassifiedInfoRequest; + public event ClassifiedInfoUpdate OnClassifiedInfoUpdate; + public event ClassifiedDelete OnClassifiedDelete; + public event ClassifiedDelete OnClassifiedGodDelete; + public event EventNotificationAddRequest OnEventNotificationAddRequest; + public event EventNotificationRemoveRequest OnEventNotificationRemoveRequest; + public event EventGodDelete OnEventGodDelete; + public event ParcelDwellRequest OnParcelDwellRequest; + public event UserInfoRequest OnUserInfoRequest; + public event UpdateUserInfo OnUpdateUserInfo; + + public void SetDebugPacketLevel(int newDebug) + { + debugLevel = newDebug; + } + + public void InPacket(object NewPack) + { + //throw new System.NotImplementedException(); + } + + public void ProcessInPacket(Packet NewPack) + { + //throw new System.NotImplementedException(); + } + + public void Close(bool ShutdownCircuit) + { + m_log.Info("[MXP ClientStack] Close Called with SC=" + ShutdownCircuit); + + // Tell the client to go + SendLogoutPacket(); + + // Let MXPPacketServer clean it up + if (Session.SessionState != SessionState.Disconnected) + { + Session.SetStateDisconnected(); + } + + // Handle OpenSim cleanup + if (ShutdownCircuit) + { + if (OnConnectionClosed != null) + OnConnectionClosed(this); + } + else + { + Scene.RemoveClient(AgentId); + } + } + + public void Kick(string message) + { + Close(false); + } + + public void Start() + { + // We dont do this + } + + public void Stop() + { + // Nor this + } + + public void SendWearables(AvatarWearable[] wearables, int serial) + { + // Need to translate to MXP somehow + } + + public void SendAppearance(UUID agentID, byte[] visualParams, byte[] textureEntry) + { + // Need to translate to MXP somehow + } + + public void SendStartPingCheck(byte seq) + { + // Need to translate to MXP somehow + } + + public void SendKillObject(ulong regionHandle, uint localID) + { + DisappearanceEventMessage de = new DisappearanceEventMessage(); + de.ObjectIndex = localID; + + Session.Send(de); + } + + public void SendAnimations(UUID[] animID, int[] seqs, UUID sourceAgentId, UUID[] objectIDs) + { + // Need to translate to MXP somehow + } + + public void SendRegionHandshake(RegionInfo regionInfo, RegionHandshakeArgs args) + { + // Need to translate to MXP somehow + } + + public void SendChatMessage(string message, byte type, Vector3 fromPos, string fromName, UUID fromAgentID, byte source, byte audible) + { + ActionEventMessage chatActionEvent = new ActionEventMessage(); + chatActionEvent.ActionFragment.ActionName = "Chat"; + chatActionEvent.ActionFragment.SourceObjectId = fromAgentID.Guid; + chatActionEvent.ActionFragment.ObservationRadius = 180.0f; + chatActionEvent.ActionFragment.ActionPayloadDialect = "TEXT"; + chatActionEvent.SetPayloadData(Encoding.UTF8.GetBytes(message)); + chatActionEvent.ActionFragment.ActionPayloadLength = (uint)chatActionEvent.GetPayloadData().Length; + + Session.Send(chatActionEvent); + } + + public void SendInstantMessage(UUID fromAgent, string message, UUID toAgent, string fromName, byte dialog, uint timeStamp) + { + // Need to translate to MXP somehow + } + + public void SendInstantMessage(UUID fromAgent, string message, UUID toAgent, string fromName, byte dialog, uint timeStamp, UUID transactionID, bool fromGroup, byte[] binaryBucket) + { + // Need to translate to MXP somehow + } + + public void SendGenericMessage(string method, List message) + { + // Need to translate to MXP somehow + } + + public void SendLayerData(float[] map) + { + // Need to translate to MXP somehow + } + + public void SendLayerData(int px, int py, float[] map) + { + // Need to translate to MXP somehow + } + + public void SendWindData(Vector2[] windSpeeds) + { + // Need to translate to MXP somehow + } + + public void MoveAgentIntoRegion(RegionInfo regInfo, Vector3 pos, Vector3 look) + { + //throw new System.NotImplementedException(); + } + + public void InformClientOfNeighbour(ulong neighbourHandle, IPEndPoint neighbourExternalEndPoint) + { + //throw new System.NotImplementedException(); + } + + public AgentCircuitData RequestClientInfo() + { + AgentCircuitData clientinfo = new AgentCircuitData(); + clientinfo.AgentID = AgentId; + clientinfo.Appearance = new AvatarAppearance(); + clientinfo.BaseFolder = UUID.Zero; + clientinfo.CapsPath = ""; + clientinfo.child = false; + clientinfo.ChildrenCapSeeds = new Dictionary(); + clientinfo.circuitcode = CircuitCode; + clientinfo.firstname = FirstName; + clientinfo.InventoryFolder = UUID.Zero; + clientinfo.lastname = LastName; + clientinfo.SecureSessionID = SecureSessionId; + clientinfo.SessionID = SessionId; + clientinfo.startpos = StartPos; + + return clientinfo; + } + + public void CrossRegion(ulong newRegionHandle, Vector3 pos, Vector3 lookAt, IPEndPoint newRegionExternalEndPoint, string capsURL) + { + // TODO: We'll want to get this one working. + // Need to translate to MXP somehow + } + + public void SendMapBlock(List mapBlocks, uint flag) + { + // Need to translate to MXP somehow + } + + public void SendLocalTeleport(Vector3 position, Vector3 lookAt, uint flags) + { + //throw new System.NotImplementedException(); + } + + public void SendRegionTeleport(ulong regionHandle, byte simAccess, IPEndPoint regionExternalEndPoint, uint locationID, uint flags, string capsURL) + { + // Need to translate to MXP somehow + } + + public void SendTeleportFailed(string reason) + { + // Need to translate to MXP somehow + } + + public void SendTeleportLocationStart() + { + // Need to translate to MXP somehow + } + + public void SendMoneyBalance(UUID transaction, bool success, byte[] description, int balance) + { + // Need to translate to MXP somehow + } + + public void SendPayPrice(UUID objectID, int[] payPrice) + { + // Need to translate to MXP somehow + } + + public void SendAvatarData(ulong regionHandle, string firstName, string lastName, string grouptitle, UUID avatarID, uint avatarLocalID, Vector3 Pos, byte[] textureEntry, uint parentID, Quaternion rotation) + { + // TODO: This needs handling - to display other avatars + } + + public void SendAvatarTerseUpdate(ulong regionHandle, ushort timeDilation, uint localID, Vector3 position, Vector3 velocity, Quaternion rotation) + { + // TODO: This probably needs handling - update other avatar positions + } + + public void SendCoarseLocationUpdate(List CoarseLocations) + { + // Minimap function, not used. + } + + public void AttachObject(uint localID, Quaternion rotation, byte attachPoint, UUID ownerID) + { + // Need to translate to MXP somehow + } + + public void SetChildAgentThrottle(byte[] throttle) + { + // Need to translate to MXP somehow + } + + public void SendPrimitiveToClient(ulong regionHandle, ushort timeDilation, uint localID, PrimitiveBaseShape primShape, Vector3 pos, Vector3 vel, Vector3 acc, Quaternion rotation, Vector3 rvel, uint flags, UUID objectID, UUID ownerID, string text, byte[] color, uint parentID, byte[] particleSystem, byte clickAction, byte material, byte[] textureanim, bool attachment, uint AttachPoint, UUID AssetId, UUID SoundId, double SoundVolume, byte SoundFlags, double SoundRadius) + { + MXPSendPrimitive(localID, ownerID, acc, rvel, primShape, pos, objectID, vel, rotation); + } + + private void MXPSendPrimitive(uint localID, UUID ownerID, Vector3 acc, Vector3 rvel, PrimitiveBaseShape primShape, Vector3 pos, UUID objectID, Vector3 vel, Quaternion rotation) + { + PerceptionEventMessage pe = new PerceptionEventMessage(); + + pe.ObjectFragment.ObjectIndex = localID; + pe.ObjectFragment.ObjectName = "Object"; + pe.ObjectFragment.OwnerId = ownerID.Guid; + pe.ObjectFragment.TypeId = Guid.Empty; + + pe.ObjectFragment.Acceleration = new[] { acc.X, acc.Y, acc.Z }; + pe.ObjectFragment.AngularAcceleration = new float[4]; + pe.ObjectFragment.AngularVelocity = new[] { rvel.X, rvel.Y, rvel.Z, 0.0f }; + pe.ObjectFragment.BoundingSphereRadius = primShape.Scale.Length()/2.0f; + pe.ObjectFragment.Location = new[] { pos.X, pos.Y, pos.Z }; + pe.ObjectFragment.Mass = 1.0f; + pe.ObjectFragment.ObjectId = objectID.Guid; + pe.ObjectFragment.Orientation = new[] {rotation.X, rotation.Y, rotation.Z, rotation.W}; + pe.ObjectFragment.ParentObjectId = Guid.Empty; + pe.ObjectFragment.Velocity = new[] { vel.X, vel.Y, vel.Z }; + + pe.ObjectFragment.StatePayloadDialect = ""; + pe.ObjectFragment.StatePayloadLength = 0; + pe.ObjectFragment.SetStatePayloadData(new byte[0]); + + Session.Send(pe); + } + + public void SendPrimitiveToClient(ulong regionHandle, ushort timeDilation, uint localID, PrimitiveBaseShape primShape, Vector3 pos, Vector3 vel, Vector3 acc, Quaternion rotation, Vector3 rvel, uint flags, UUID objectID, UUID ownerID, string text, byte[] color, uint parentID, byte[] particleSystem, byte clickAction, byte material) + { + MXPSendPrimitive(localID, ownerID, acc, rvel, primShape, pos, objectID, vel, rotation); + } + + public void SendPrimTerseUpdate(ulong regionHandle, ushort timeDilation, uint localID, Vector3 position, Quaternion rotation, Vector3 velocity, Vector3 rotationalvelocity, byte state, UUID AssetId, UUID owner, int attachPoint) + { + MovementEventMessage me = new MovementEventMessage(); + me.ObjectIndex = localID; + me.Location = new[] {position.X, position.Y, position.Z}; + me.Orientation = new[] {rotation.X, rotation.Y, rotation.Z, rotation.W}; + + Session.Send(me); + } + + public void SendInventoryFolderDetails(UUID ownerID, UUID folderID, List items, List folders, bool fetchFolders, bool fetchItems) + { + // Need to translate to MXP somehow + } + + public void SendInventoryItemDetails(UUID ownerID, InventoryItemBase item) + { + // Need to translate to MXP somehow + } + + public void SendInventoryItemCreateUpdate(InventoryItemBase Item, uint callbackId) + { + // Need to translate to MXP somehow + } + + public void SendRemoveInventoryItem(UUID itemID) + { + // Need to translate to MXP somehow + } + + public void SendTakeControls(int controls, bool passToAgent, bool TakeControls) + { + // Need to translate to MXP somehow + } + + public void SendTaskInventory(UUID taskID, short serial, byte[] fileName) + { + // Need to translate to MXP somehow + } + + public void SendBulkUpdateInventory(InventoryNodeBase node) + { + // Need to translate to MXP somehow + } + + public void SendXferPacket(ulong xferID, uint packet, byte[] data) + { + // SL Specific, Ignore. (Remove from IClient) + } + + public void SendEconomyData(float EnergyEfficiency, int ObjectCapacity, int ObjectCount, int PriceEnergyUnit, int PriceGroupCreate, int PriceObjectClaim, float PriceObjectRent, float PriceObjectScaleFactor, int PriceParcelClaim, float PriceParcelClaimFactor, int PriceParcelRent, int PricePublicObjectDecay, int PricePublicObjectDelete, int PriceRentLight, int PriceUpload, int TeleportMinPrice, float TeleportPriceExponent) + { + // SL Specific, Ignore. (Remove from IClient) + } + + public void SendAvatarPickerReply(AvatarPickerReplyAgentDataArgs AgentData, List Data) + { + // Need to translate to MXP somehow + } + + public void SendAgentDataUpdate(UUID agentid, UUID activegroupid, string firstname, string lastname, ulong grouppowers, string groupname, string grouptitle) + { + // Need to translate to MXP somehow + // TODO: This may need doing - involves displaying the users avatar name + } + + public void SendPreLoadSound(UUID objectID, UUID ownerID, UUID soundID) + { + // Need to translate to MXP somehow + } + + public void SendPlayAttachedSound(UUID soundID, UUID objectID, UUID ownerID, float gain, byte flags) + { + // Need to translate to MXP somehow + } + + public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) + { + // Need to translate to MXP somehow + } + + public void SendAttachedSoundGainChange(UUID objectID, float gain) + { + // Need to translate to MXP somehow + } + + public void SendNameReply(UUID profileId, string firstname, string lastname) + { + // SL Specific + } + + public void SendAlertMessage(string message) + { + SendChatMessage(message, 0, Vector3.Zero, "System", UUID.Zero, 0, 0); + } + + public void SendAgentAlertMessage(string message, bool modal) + { + SendChatMessage(message, 0, Vector3.Zero, "System" + (modal ? " Notice" : ""), UUID.Zero, 0, 0); + } + + public void SendLoadURL(string objectname, UUID objectID, UUID ownerID, bool groupOwned, string message, string url) + { + // TODO: Probably can do this better + SendChatMessage("Please visit: " + url, 0, Vector3.Zero, objectname, UUID.Zero, 0, 0); + } + + public void SendDialog(string objectname, UUID objectID, UUID ownerID, string msg, UUID textureID, int ch, string[] buttonlabels) + { + // TODO: Probably can do this better + SendChatMessage("Dialog: " + msg, 0, Vector3.Zero, objectname, UUID.Zero, 0, 0); + } + + public bool AddMoney(int debit) + { + SendChatMessage("You were paid: " + debit, 0, Vector3.Zero, "System", UUID.Zero, 0, 0); + return true; + } + + public void SendSunPos(Vector3 sunPos, Vector3 sunVel, ulong CurrentTime, uint SecondsPerSunCycle, uint SecondsPerYear, float OrbitalPosition) + { + // Need to translate to MXP somehow + // Send a light object? + } + + public void SendViewerEffect(ViewerEffectPacket.EffectBlock[] effectBlocks) + { + // Need to translate to MXP somehow + } + + public void SendViewerTime(int phase) + { + // Need to translate to MXP somehow + } + + public UUID GetDefaultAnimation(string name) + { + return UUID.Zero; + } + + public void SendAvatarProperties(UUID avatarID, string aboutText, string bornOn, byte[] charterMember, string flAbout, uint flags, UUID flImageID, UUID imageID, string profileURL, UUID partnerID) + { + // Need to translate to MXP somehow + } + + public void SendScriptQuestion(UUID taskID, string taskName, string ownerName, UUID itemID, int question) + { + // Need to translate to MXP somehow + } + + public void SendHealth(float health) + { + // Need to translate to MXP somehow + } + + public void SendEstateManagersList(UUID invoice, UUID[] EstateManagers, uint estateID) + { + // Need to translate to MXP somehow + } + + public void SendBannedUserList(UUID invoice, EstateBan[] banlist, uint estateID) + { + // Need to translate to MXP somehow + } + + public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) + { + // Need to translate to MXP somehow + } + + public void SendEstateCovenantInformation(UUID covenant) + { + // Need to translate to MXP somehow + } + + public void SendDetailedEstateData(UUID invoice, string estateName, uint estateID, uint parentEstate, uint estateFlags, uint sunPosition, UUID covenant, string abuseEmail, UUID estateOwner) + { + // Need to translate to MXP somehow + } + + public void SendLandProperties(int sequence_id, bool snap_selection, int request_result, LandData landData, float simObjectBonusFactor, int parcelObjectCapacity, int simObjectCapacity, uint regionFlags) + { + // Need to translate to MXP somehow + } + + public void SendLandAccessListData(List avatars, uint accessFlag, int localLandID) + { + // Need to translate to MXP somehow + } + + public void SendForceClientSelectObjects(List objectIDs) + { + // Need to translate to MXP somehow + } + + public void SendLandObjectOwners(Dictionary ownersAndCount) + { + // Need to translate to MXP somehow + } + + public void SendLandParcelOverlay(byte[] data, int sequence_id) + { + // Need to translate to MXP somehow + } + + public void SendParcelMediaCommand(uint flags, ParcelMediaCommandEnum command, float time) + { + // Need to translate to MXP somehow + } + + public void SendParcelMediaUpdate(string mediaUrl, UUID mediaTextureID, byte autoScale, string mediaType, string mediaDesc, int mediaWidth, int mediaHeight, byte mediaLoop) + { + // Need to translate to MXP somehow + } + + public void SendAssetUploadCompleteMessage(sbyte AssetType, bool Success, UUID AssetFullID) + { + // Need to translate to MXP somehow + } + + public void SendConfirmXfer(ulong xferID, uint PacketID) + { + // Need to translate to MXP somehow + } + + public void SendXferRequest(ulong XferID, short AssetType, UUID vFileID, byte FilePath, byte[] FileName) + { + // Need to translate to MXP somehow + } + + public void SendInitiateDownload(string simFileName, string clientFileName) + { + // Need to translate to MXP somehow + } + + public void SendImageFirstPart(ushort numParts, UUID ImageUUID, uint ImageSize, byte[] ImageData, byte imageCodec) + { + // Need to translate to MXP somehow + } + + public void SendImageNextPart(ushort partNumber, UUID imageUuid, byte[] imageData) + { + // Need to translate to MXP somehow + } + + public void SendImageNotFound(UUID imageid) + { + // Need to translate to MXP somehow + } + + public void SendShutdownConnectionNotice() + { + // Need to translate to MXP somehow + } + + public void SendSimStats(SimStats stats) + { + // Need to translate to MXP somehow + } + + public void SendObjectPropertiesFamilyData(uint RequestFlags, UUID ObjectUUID, UUID OwnerID, UUID GroupID, uint BaseMask, uint OwnerMask, uint GroupMask, uint EveryoneMask, uint NextOwnerMask, int OwnershipCost, byte SaleType, int SalePrice, uint Category, UUID LastOwnerID, string ObjectName, string Description) + { + //throw new System.NotImplementedException(); + } + + public void SendObjectPropertiesReply(UUID ItemID, ulong CreationDate, UUID CreatorUUID, UUID FolderUUID, UUID FromTaskUUID, UUID GroupUUID, short InventorySerial, UUID LastOwnerUUID, UUID ObjectUUID, UUID OwnerUUID, string TouchTitle, byte[] TextureID, string SitTitle, string ItemName, string ItemDescription, uint OwnerMask, uint NextOwnerMask, uint GroupMask, uint EveryoneMask, uint BaseMask, byte saleType, int salePrice) + { + //throw new System.NotImplementedException(); + } + + public void SendAgentOffline(UUID[] agentIDs) + { + // Need to translate to MXP somehow (Friends List) + } + + public void SendAgentOnline(UUID[] agentIDs) + { + // Need to translate to MXP somehow (Friends List) + } + + public void SendSitResponse(UUID TargetID, Vector3 OffsetPos, Quaternion SitOrientation, bool autopilot, Vector3 CameraAtOffset, Vector3 CameraEyeOffset, bool ForceMouseLook) + { + // Need to translate to MXP somehow + } + + public void SendAdminResponse(UUID Token, uint AdminLevel) + { + // Need to translate to MXP somehow + } + + public void SendGroupMembership(GroupMembershipData[] GroupMembership) + { + // Need to translate to MXP somehow + } + + public void SendGroupNameReply(UUID groupLLUID, string GroupName) + { + // Need to translate to MXP somehow + } + + public void SendJoinGroupReply(UUID groupID, bool success) + { + // Need to translate to MXP somehow + } + + public void SendEjectGroupMemberReply(UUID agentID, UUID groupID, bool success) + { + // Need to translate to MXP somehow + } + + public void SendLeaveGroupReply(UUID groupID, bool success) + { + // Need to translate to MXP somehow + } + + public void SendLandStatReply(uint reportType, uint requestFlags, uint resultCount, LandStatReportItem[] lsrpia) + { + // Need to translate to MXP somehow + } + + public void SendScriptRunningReply(UUID objectID, UUID itemID, bool running) + { + // Need to translate to MXP somehow + } + + public void SendAsset(AssetRequestToClient req) + { + // Need to translate to MXP somehow + } + + public void SendTexture(AssetBase TextureAsset) + { + // Need to translate to MXP somehow + } + + public byte[] GetThrottlesPacked(float multiplier) + { + // LL Specific, get out of IClientAPI + + const int singlefloat = 4; + float tResend = multiplier; + float tLand = multiplier; + float tWind = multiplier; + float tCloud = multiplier; + float tTask = multiplier; + float tTexture = multiplier; + float tAsset = multiplier; + + byte[] throttles = new byte[singlefloat * 7]; + int i = 0; + Buffer.BlockCopy(BitConverter.GetBytes(tResend), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tLand), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tWind), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tCloud), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tTask), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tTexture), 0, throttles, singlefloat * i, singlefloat); + i++; + Buffer.BlockCopy(BitConverter.GetBytes(tAsset), 0, throttles, singlefloat * i, singlefloat); + + return throttles; + } + + public event ViewerEffectEventHandler OnViewerEffect; + public event Action OnLogout; + public event Action OnConnectionClosed; + + + public void SendBlueBoxMessage(UUID FromAvatarID, string FromAvatarName, string Message) + { + SendChatMessage(Message, 0, Vector3.Zero, FromAvatarName, UUID.Zero, 0, 0); + } + + public void SendLogoutPacket() + { + LeaveRequestMessage lrm = new LeaveRequestMessage(); + Session.Send(lrm); + } + + public ClientInfo GetClientInfo() + { + return null; + //throw new System.NotImplementedException(); + } + + public void SetClientInfo(ClientInfo info) + { + //throw new System.NotImplementedException(); + } + + public void SetClientOption(string option, string value) + { + // Need to translate to MXP somehow + } + + public string GetClientOption(string option) + { + // Need to translate to MXP somehow + return ""; + } + + public void Terminate() + { + Close(false); + } + + public void SendSetFollowCamProperties(UUID objectID, SortedDictionary parameters) + { + // Need to translate to MXP somehow + } + + public void SendClearFollowCamProperties(UUID objectID) + { + // Need to translate to MXP somehow + } + + public void SendRegionHandle(UUID regoinID, ulong handle) + { + // Need to translate to MXP somehow + } + + public void SendParcelInfo(RegionInfo info, LandData land, UUID parcelID, uint x, uint y) + { + // Need to translate to MXP somehow + } + + public void SendScriptTeleportRequest(string objName, string simName, Vector3 pos, Vector3 lookAt) + { + // Need to translate to MXP somehow + } + + public void SendDirPlacesReply(UUID queryID, DirPlacesReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirPeopleReply(UUID queryID, DirPeopleReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirEventsReply(UUID queryID, DirEventsReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirGroupsReply(UUID queryID, DirGroupsReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirClassifiedReply(UUID queryID, DirClassifiedReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirLandReply(UUID queryID, DirLandReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendDirPopularReply(UUID queryID, DirPopularReplyData[] data) + { + // Need to translate to MXP somehow + } + + public void SendEventInfoReply(EventData info) + { + // Need to translate to MXP somehow + } + + public void SendMapItemReply(mapItemReply[] replies, uint mapitemtype, uint flags) + { + // Need to translate to MXP somehow + } + + public void SendAvatarGroupsReply(UUID avatarID, GroupMembershipData[] data) + { + // Need to translate to MXP somehow + } + + public void SendOfferCallingCard(UUID srcID, UUID transactionID) + { + // Need to translate to MXP somehow + } + + public void SendAcceptCallingCard(UUID transactionID) + { + // Need to translate to MXP somehow + } + + public void SendDeclineCallingCard(UUID transactionID) + { + // Need to translate to MXP somehow + } + + public void SendTerminateFriend(UUID exFriendID) + { + // Need to translate to MXP somehow + } + + public void SendAvatarClassifiedReply(UUID targetID, UUID[] classifiedID, string[] name) + { + // Need to translate to MXP somehow + } + + public void SendClassifiedInfoReply(UUID classifiedID, UUID creatorID, uint creationDate, uint expirationDate, uint category, string name, string description, UUID parcelID, uint parentEstate, UUID snapshotID, string simName, Vector3 globalPos, string parcelName, byte classifiedFlags, int price) + { + // Need to translate to MXP somehow + } + + public void SendAgentDropGroup(UUID groupID) + { + // Need to translate to MXP somehow + } + + public void SendAvatarNotesReply(UUID targetID, string text) + { + // Need to translate to MXP somehow + } + + public void SendAvatarPicksReply(UUID targetID, Dictionary picks) + { + // Need to translate to MXP somehow + } + + public void SendAvatarClassifiedReply(UUID targetID, Dictionary classifieds) + { + // Need to translate to MXP somehow + } + + public void SendParcelDwellReply(int localID, UUID parcelID, float dwell) + { + // Need to translate to MXP somehow + } + + public void SendUserInfoReply(bool imViaEmail, bool visible, string email) + { + // Need to translate to MXP somehow + } + + public void KillEndDone() + { + Stop(); + } + + public bool AddGenericPacketHandler(string MethodName, GenericMessage handler) + { + // Need to translate to MXP somehow + return true; + } + + #endregion + + #region IClientCore + + public bool TryGet(out T iface) + { + iface = default(T); + return false; + } + + public T Get() + { + return default(T); + } + + #endregion + } +} diff --git a/OpenSim/Client/MXP/MXPModule.cs b/OpenSim/Client/MXP/MXPModule.cs new file mode 100644 index 0000000000..345e4fb2b1 --- /dev/null +++ b/OpenSim/Client/MXP/MXPModule.cs @@ -0,0 +1,87 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Timers; +using MXP; +using Nini.Config; +using OpenMetaverse; +using OpenSim.Client.MXP.PacketHandler; +using OpenSim.Region.Framework.Interfaces; +using OpenSim.Region.Framework.Scenes; + +namespace OpenSim.Client.MXP +{ + public class MXPModule : IRegionModule + { + private int mxp_Port = 1253; + private double mxp_BubbleRadius = 181.01933598375616624661615669884; // Radius of a sphere big enough to encapsulate a 256x256 square + + private readonly Timer ticker = new Timer(100); + + private int ticks; + private bool shutdown = false; + + private IConfigSource config; + + private readonly Dictionary m_scenes = new Dictionary(); + + private MXPPacketServer server; + + + public void Initialise(Scene scene, IConfigSource source) + { + m_scenes.Add(scene.RegionInfo.RegionID, scene); + config = source; + } + + public void PostInitialise() + { + if (config.Configs["MXP"] != null) + { + IConfig con = config.Configs["MXP"]; + + if(!con.GetBoolean("Enabled",false)) + return; + + mxp_Port = con.GetInt("Port", mxp_Port); + + + server = new MXPPacketServer("http://null", mxp_Port, m_scenes); + + ticker.AutoReset = false; + ticker.Elapsed += ticker_Elapsed; + + ticker.Start(); + } + } + + void ticker_Elapsed(object sender, ElapsedEventArgs e) + { + server.Process(); + + if (!shutdown) + ticker.Start(); + + if(++ticks % 100 == 0) + { + server.PrintDebugInformation(); + } + } + + public void Close() + { + shutdown = true; + ticker.Stop(); + } + + public string Name + { + get { return "MXP ClientStack Module"; } + } + + public bool IsSharedModule + { + get { return true; } + } + } +} diff --git a/OpenSim/Client/MXP/MXPUtil.cs b/OpenSim/Client/MXP/MXPUtil.cs new file mode 100644 index 0000000000..7ca4533c49 --- /dev/null +++ b/OpenSim/Client/MXP/MXPUtil.cs @@ -0,0 +1,15 @@ +using System; +using System.Collections.Generic; +using System.Text; +using OpenMetaverse; + +namespace OpenSim.Client.MXP +{ + static class MXPUtil + { + public static string GenerateMXPURL(string server, int port, UUID bubbleID, Vector3 location) + { + return string.Format("mxp://{0}:{1}/{2}/{3}", server, port, bubbleID.Guid, location); + } + } +} diff --git a/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs b/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs new file mode 100644 index 0000000000..b35ab9cfdd --- /dev/null +++ b/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs @@ -0,0 +1,367 @@ +/* This file borrows heavily from MXPServer.cs - the reference MXPServer + * See http://www.bubblecloud.org for a copy of the original file and + * implementation details. */ +using System; +using System.Collections.Generic; +using System.Reflection; +using log4net; +using MXP; +using MXP.Messages; +using OpenMetaverse; +using OpenSim.Client.MXP.ClientStack; +using OpenSim.Region.Framework.Scenes; + +namespace OpenSim.Client.MXP.PacketHandler +{ + class MXPPacketServer + { + internal static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + + private readonly List Clients = new List(); + private readonly Dictionary Scenes; + + #region Fields + + private readonly Transmitter transmitter; + + private readonly IList sessions = new List(); + private readonly IList sessionsToRemove = new List(); + + private readonly String cloudUrl; + private readonly String programName; + private readonly byte programMajorVersion; + private readonly byte programMinorVersion; + + #endregion + + #region Constructors + + public MXPPacketServer(string cloudUrl, int port, Dictionary scenes) + { + this.cloudUrl = cloudUrl; + + Scenes = scenes; + + programMinorVersion = 63; + programMajorVersion = 0; + programName = "OpenSimulator"; + + transmitter = new Transmitter(port); + } + + #endregion + + #region Properties + + /// + /// Number of sessions pending. (Process() accepts pending sessions). + /// + public int PendingSessionCount + { + get + { + return transmitter.PendingSessionCount; + } + } + /// + /// Number of connected sessions. + /// + public int SessionCount + { + get + { + return sessions.Count; + } + } + /// + /// Property reflecting whether client transmitter threads are alive. + /// + public bool IsTransmitterAlive + { + get + { + return transmitter != null && transmitter.IsAlive; + } + } + /// + /// Number of packets sent. + /// + public ulong PacketsSent + { + get + { + return transmitter != null ? transmitter.PacketsSent : 0; + } + } + /// + /// Number of packets received. + /// + public ulong PacketsReceived + { + get + { + return transmitter != null ? transmitter.PacketsReceived : 0; + } + } + /// + /// Bytes client has received so far. + /// + public ulong BytesReceived + { + get + { + return transmitter != null ? transmitter.BytesReceived : 0; + } + } + /// + /// Bytes client has sent so far. + /// + public ulong BytesSent + { + get + { + return transmitter != null ? transmitter.BytesSent : 0; + } + } + /// + /// Number of bytes received (bytes per second) during past second. + /// + public double ReceiveRate + { + get + { + return transmitter != null ? transmitter.ReceiveRate : 0; + } + } + /// + /// Number of bytes sent (bytes per second) during past second. + /// + public double SendRate + { + get + { + return transmitter != null ? transmitter.SendRate : 0; + } + } + + #endregion + + #region Session Management + + public void Disconnect(Session session) + { + if (session.IsConnected) + { + Message message = MessageFactory.Current.ReserveMessage(typeof(LeaveRequestMessage)); + session.Send(message); + MessageFactory.Current.ReleaseMessage(message); + } + else + { + throw new Exception("Not connected."); + } + } + + #endregion + + + #region Processing + + public void PrintDebugInformation() + { + m_log.Info("[MXP ClientStack] Statistics report"); + m_log.Info("Pending Sessions: " + PendingSessionCount); + m_log.Info("Sessions: " + SessionCount + " (Clients: " + Clients.Count + " )"); + m_log.Info("Transmitter Alive?: " + IsTransmitterAlive); + m_log.Info("Packets Sent/Recieved: " + PacketsSent + " / " + PacketsReceived); + m_log.Info("Bytes Sent/Recieved: " + BytesSent + " / " + BytesReceived); + m_log.Info("Send/Recieve Rate (bps): " + SendRate + " / " + ReceiveRate); + } + + public void Process() + { + ProcessMessages(); + Clean(); + } + + public void Clean() + { + foreach (MXPClientView clientView in Clients) + { + if (clientView.Session.SessionState == SessionState.Disconnected) + { + sessionsToRemove.Add(clientView); + } + } + + foreach (MXPClientView clientView in sessionsToRemove) + { + clientView.Scene.RemoveClient(clientView.AgentId); + Clients.Remove(clientView); + sessions.Remove(clientView.Session); + } + + sessionsToRemove.Clear(); + } + + public bool AuthoriseUser(string participantName, string pass, UUID scene) + { + if (Scenes.ContainsKey(scene)) + return true; + + return false; + } + + public void ProcessMessages() + { + if (transmitter.PendingSessionCount > 0) + { + sessions.Add(transmitter.AcceptPendingSession()); + } + + foreach (MXPClientView clientView in Clients) + { + + int messagesProcessedCount = 0; + Session session = clientView.Session; + + while (session.AvailableMessages > 0) + { + + Message message = session.Receive(); + + if (message.GetType() == typeof(JoinRequestMessage)) + { + + JoinRequestMessage joinRequestMessage = (JoinRequestMessage)message; + + bool authorized = AuthoriseUser(joinRequestMessage.ParticipantName, + joinRequestMessage.ParticipantPassphrase, + new UUID(joinRequestMessage.BubbleId)); + + if (authorized) + { + Scene target = Scenes[new UUID(joinRequestMessage.BubbleId)]; + + UUID mxpSessionID = UUID.Random(); + + m_log.Info("[MXP ClientStack] Session join request success: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); + + AcceptConnection(session, joinRequestMessage, mxpSessionID); + + MXPClientView client = new MXPClientView(session, mxpSessionID, target, + joinRequestMessage.ParticipantName); + Clients.Add(client); + + target.AddNewClient(client); + } + else + { + m_log.Info("[MXP ClientStack] Session join request failure: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); + + DeclineConnection(session, joinRequestMessage); + } + + } + if (message.GetType() == typeof(LeaveRequestMessage)) + { + + LeaveResponseMessage leaveResponseMessage = (LeaveResponseMessage)MessageFactory.Current.ReserveMessage( + typeof(LeaveResponseMessage)); + + m_log.Info("[MXP ClientStack] Session leave request: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); + + leaveResponseMessage.RequestMessageId = message.MessageId; + leaveResponseMessage.FailureCode = 0; + session.Send(leaveResponseMessage); + + if (session.SessionState != SessionState.Disconnected) + { + session.SetStateDisconnected(); + } + + m_log.Info("[MXP ClientStack] Removing Client from Scene"); + clientView.Scene.RemoveClient(clientView.AgentId); + } + if (message.GetType() == typeof(LeaveResponseMessage)) + { + + LeaveResponseMessage leaveResponseMessage = (LeaveResponseMessage)message; + + m_log.Info("[MXP ClientStack] Session leave response: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); + + if (leaveResponseMessage.FailureCode == 0) + { + session.SetStateDisconnected(); + } + + m_log.Info("[MXP ClientStack] Removing Client from Scene"); + clientView.Scene.RemoveClient(clientView.AgentId); + } + else + { + clientView.ProcessMXPPacket(message); + } + + MessageFactory.Current.ReleaseMessage(message); + messagesProcessedCount++; + if (messagesProcessedCount > 1000) + { + break; + } + } + } + } + + private void AcceptConnection(Session session, JoinRequestMessage joinRequestMessage, UUID mxpSessionID) + { + JoinResponseMessage joinResponseMessage = (JoinResponseMessage)MessageFactory.Current.ReserveMessage( + typeof(JoinResponseMessage)); + + joinResponseMessage.RequestMessageId = joinRequestMessage.MessageId; + joinResponseMessage.FailureCode = 0; + + joinResponseMessage.ParticipantId = mxpSessionID.Guid; + joinResponseMessage.CloudUrl = cloudUrl; + + joinResponseMessage.BubbleName = Scenes[new UUID(joinRequestMessage.BubbleId)].RegionInfo.RegionName; + + joinResponseMessage.BubbleRealTime = 0; + joinResponseMessage.ProgramName = programName; + joinResponseMessage.ProgramMajorVersion = programMajorVersion; + joinResponseMessage.ProgramMinorVersion = programMinorVersion; + joinResponseMessage.ProtocolMajorVersion = MxpConstants.ProtocolMajorVersion; + joinResponseMessage.ProtocolMinorVersion = MxpConstants.ProtocolMinorVersion; + + session.Send(joinResponseMessage); + + session.SetStateConnected(); + } + + private void DeclineConnection(Session session, Message joinRequestMessage) + { + JoinResponseMessage joinResponseMessage = (JoinResponseMessage)MessageFactory.Current.ReserveMessage(typeof(JoinResponseMessage)); + + joinResponseMessage.RequestMessageId = joinRequestMessage.MessageId; + joinResponseMessage.FailureCode = 1; + + joinResponseMessage.CloudUrl = cloudUrl; + + joinResponseMessage.BubbleName = "Declined OpenSim Region"; // Dont reveal anything about the sim in the disconnect notice + + joinResponseMessage.BubbleRealTime = 0; + joinResponseMessage.ProgramName = programName; + joinResponseMessage.ProgramMajorVersion = programMajorVersion; + joinResponseMessage.ProgramMinorVersion = programMinorVersion; + joinResponseMessage.ProtocolMajorVersion = MxpConstants.ProtocolMajorVersion; + joinResponseMessage.ProtocolMinorVersion = MxpConstants.ProtocolMinorVersion; + + session.Send(joinResponseMessage); + + session.SetStateDisconnected(); + } + + #endregion + + } +} diff --git a/ThirdPartyLicenses/MXP.txt b/ThirdPartyLicenses/MXP.txt new file mode 100644 index 0000000000..fb4e47f5ec --- /dev/null +++ b/ThirdPartyLicenses/MXP.txt @@ -0,0 +1,60 @@ +MXP Reference Implementation +Code availible from: http://setp.googlecode.com/svn/trunk/ +http://www.bubblecloud.org + + +Apache License +Version 2.0, January 2004 +http://www.apache.org/licenses/ + +TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + +1. Definitions. + +"License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. + +"Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. + +"Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. + +"You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. + +"Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. + +"Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. + +"Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). + +"Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. + +"Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." + +"Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. + +2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. + +3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. + +4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: + + 1. You must give any other recipients of the Work or Derivative Works a copy of this License; and + + 2. You must cause any modified files to carry prominent notices stating that You changed the files; and + + 3. You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and + + 4. If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. + +You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. + +5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. + +6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. + +7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. + +8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. + +9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. + +END OF TERMS AND CONDITIONS diff --git a/bin/MXP.dll b/bin/MXP.dll new file mode 100644 index 0000000000..013da0095b Binary files /dev/null and b/bin/MXP.dll differ diff --git a/bin/MXP.pdb b/bin/MXP.pdb new file mode 100644 index 0000000000..28154e018f Binary files /dev/null and b/bin/MXP.pdb differ diff --git a/prebuild.xml b/prebuild.xml index 13d4c33666..f4fd32e377 100644 --- a/prebuild.xml +++ b/prebuild.xml @@ -1494,6 +1494,37 @@ + + + + + + + + + ../../../bin/ + + + + + ../../../bin/ + + + + ../../../bin/ + + + + + + + + + + + + +