/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using System.Collections.Specialized; using Apache.NMS.Util; namespace Apache.NMS.Stomp.Commands { /// /// Summary description for Destination. /// public abstract class Destination : BaseDataStructure, IDestination { /// /// Topic Destination object /// public const int STOMP_TOPIC = 1; /// /// Temporary Topic Destination object /// public const int STOMP_TEMPORARY_TOPIC = 2; /// /// Queue Destination object /// public const int STOMP_QUEUE = 3; /// /// Temporary Queue Destination object /// public const int STOMP_TEMPORARY_QUEUE = 4; private const String TEMP_PREFIX = "{TD{"; private const String TEMP_POSTFIX = "}TD}"; private const String COMPOSITE_SEPARATOR = ","; private String physicalName = ""; private StringDictionary options = null; /// /// The Default Constructor /// protected Destination() { } /// /// Construct the Destination with a defined physical name; /// /// protected Destination(String name) { setPhysicalName(name); } public bool IsTopic { get { int destinationType = GetDestinationType(); return STOMP_TOPIC == destinationType || STOMP_TEMPORARY_TOPIC == destinationType; } } public bool IsQueue { get { int destinationType = GetDestinationType(); return STOMP_QUEUE == destinationType || STOMP_TEMPORARY_QUEUE == destinationType; } } public bool IsTemporary { get { int destinationType = GetDestinationType(); return STOMP_TEMPORARY_QUEUE == destinationType || STOMP_TEMPORARY_TOPIC == destinationType; } } /// /// Dictionary of name/value pairs representing option values specified /// in the URI used to create this Destination. A null value is returned /// if no options were specified. /// internal StringDictionary Options { get { return this.options; } } private void setPhysicalName(string name) { this.physicalName = name; int p = name.IndexOf('?'); if(p >= 0) { String optstring = physicalName.Substring(p + 1); this.physicalName = name.Substring(0, p); options = URISupport.ParseQuery(optstring); } } /// /// A helper method to return a descriptive string for the topic or queue /// /// /// a descriptive string for this queue or topic public static String Inspect(Destination destination) { if(destination is ITopic) { return "Topic(" + destination.ToString() + ")"; } else { return "Queue(" + destination.ToString() + ")"; } } /// /// /// /// public static Destination Transform(IDestination destination) { Destination result = null; if(destination != null) { if(destination is Destination) { result = (Destination) destination; } else { if(destination is ITemporaryQueue) { result = new TempQueue(((IQueue) destination).QueueName); } else if(destination is ITemporaryTopic) { result = new TempTopic(((ITopic) destination).TopicName); } else if(destination is IQueue) { result = new Queue(((IQueue) destination).QueueName); } else if(destination is ITopic) { result = new Topic(((ITopic) destination).TopicName); } } } return result; } /// /// Create a Destination /// /// /// /// public static Destination CreateDestination(int type, String pyhsicalName) { Destination result = null; if(pyhsicalName == null) { return null; } else if(type == STOMP_TOPIC) { result = new Topic(pyhsicalName); } else if(type == STOMP_TEMPORARY_TOPIC) { result = new TempTopic(pyhsicalName); } else if(type == STOMP_QUEUE) { result = new Queue(pyhsicalName); } else { result = new TempQueue(pyhsicalName); } return result; } /// /// Create a temporary name from the clientId /// /// /// public static String CreateTemporaryName(String clientId) { return TEMP_PREFIX + clientId + TEMP_POSTFIX; } /// /// From a temporary destination find the clientId of the Connection that created it /// /// /// the clientId or null if not a temporary destination public static String GetClientId(Destination destination) { String answer = null; if(destination != null && destination.IsTemporary) { String name = destination.PhysicalName; int start = name.IndexOf(TEMP_PREFIX); if(start >= 0) { start += TEMP_PREFIX.Length; int stop = name.LastIndexOf(TEMP_POSTFIX); if(stop > start && stop < name.Length) { answer = name.Substring(start, stop); } } } return answer; } /// /// /// object to compare /// 1 if this is less than o else 0 if they are equal or -1 if this is less than o public int CompareTo(Object o) { if(o is Destination) { return CompareTo((Destination) o); } return -1; } /// /// Lets sort by name first then lets sort topics greater than queues /// /// another destination to compare against /// 1 if this is less than o else 0 if they are equal or -1 if this is less than o public int CompareTo(Destination that) { int answer = 0; if(physicalName != that.physicalName) { if(physicalName == null) { return -1; } else if(that.physicalName == null) { return 1; } answer = physicalName.CompareTo(that.physicalName); } if(answer == 0) { if(IsTopic) { if(that.IsQueue) { return 1; } } else { if(that.IsTopic) { return -1; } } } return answer; } /// /// /// Returns the Destination type public abstract int GetDestinationType(); public String PhysicalName { get { return this.physicalName; } set { this.physicalName = value; } } /// /// Returns true if this destination represents a collection of /// destinations; allowing a set of destinations to be published to or subscribed /// from in one NMS operation. ///

/// If this destination is a composite then you can call {@link #getChildDestinations()} /// to return the list of child destinations. ///

public bool IsComposite { get { return physicalName.IndexOf(COMPOSITE_SEPARATOR) > 0; } } /*public List GetChildDestinations() { List answer = new ArrayList(); StringTokenizer iter = new StringTokenizer(physicalName, COMPOSITE_SEPARATOR); while (iter.hasMoreTokens()) { String name = iter.nextToken(); Destination child = null; if (name.StartsWith(QUEUE_PREFIX)) { child = new ActiveMQQueue(name.Substring(QUEUE_PREFIX.Length)); } else if (name.StartsWith(TOPIC_PREFIX)) { child = new ActiveMQTopic(name.Substring(TOPIC_PREFIX.Length)); } else { child = createDestination(name); } answer.add(child); } if (answer.size() == 1) { // lets put ourselves inside the collection // as we are not really a composite destination answer.set(0, this); } return answer; }*/ /// /// /// string representation of this instance public override String ToString() { switch(DestinationType) { case DestinationType.Topic: return "topic://" + PhysicalName; case DestinationType.TemporaryTopic: return "temp-topic://" + PhysicalName; case DestinationType.TemporaryQueue: return "temp-queue://" + PhysicalName; default: return "queue://" + PhysicalName; } } /// /// /// hashCode for this instance public override int GetHashCode() { int answer = 37; if(this.physicalName != null) { answer = physicalName.GetHashCode(); } if(IsTopic) { answer ^= 0xfabfab; } return answer; } /// /// if the object passed in is equivalent, return true /// /// the object to compare /// true if this instance and obj are equivalent public override bool Equals(Object obj) { bool result = this == obj; if(!result && obj != null && obj is Destination) { Destination other = (Destination) obj; result = this.GetDestinationType() == other.GetDestinationType() && this.physicalName.Equals(other.physicalName); } return result; } /// /// Factory method to create a child destination if this destination is a composite /// /// /// the created Destination public abstract Destination CreateDestination(String name); public abstract DestinationType DestinationType { get; } public override Object Clone() { // Since we are a derived class use the base's Clone() // to perform the shallow copy. Since it is shallow it // will include our derived class. Since we are derived, // this method is an override. Destination o = (Destination) base.Clone(); // Now do the deep work required // If any new variables are added then this routine will // likely need updating return o; } } }