/Src/TinyRadius.Net.Core/Attributes/VendorSpecificAttribute.cs

http://tinyradius4net.googlecode.com/ · C# · 340 lines · 213 code · 33 blank · 94 comment · 37 complexity · 21db8af2611be8f735ca81a87254ac2f MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Text;
  5. using TinyRadius.Net.Dictionaries;
  6. using TinyRadius.Net.Util;
  7. namespace TinyRadius.Net.Attributes
  8. {
  9. /// <summary>
  10. /// This class represents a "Vendor-Specific" attribute.
  11. /// </summary>
  12. public class VendorSpecificAttribute : RadiusAttribute
  13. {
  14. /// <summary>
  15. /// Radius attribute type code for Vendor-Specific
  16. /// </summary>
  17. public static readonly int VENDOR_SPECIFIC = 26;
  18. /// <summary>
  19. /// Sub attributes. Only set if isRawData == false.
  20. /// </summary>
  21. private List<RadiusAttribute> subAttributes = new List<RadiusAttribute>();
  22. /// <summary>
  23. /// Constructs an empty Vendor-Specific attribute that can be read from a
  24. /// Radius packet.
  25. /// </summary>
  26. public VendorSpecificAttribute()
  27. {
  28. }
  29. /// <summary>
  30. /// Constructs a new Vendor-Specific attribute to be sent.
  31. /// @param vendorId vendor ID of the sub-attributes
  32. /// </summary>
  33. public VendorSpecificAttribute(int vendorId)
  34. {
  35. Type = VENDOR_SPECIFIC;
  36. ChildVendorId = vendorId;
  37. }
  38. /// <summary>
  39. /// Gets or sets the vendor ID of the child attributes.
  40. /// @param childVendorId
  41. /// </summary>
  42. public int ChildVendorId { get; set; }
  43. public override IWritableDictionary Dictionary
  44. {
  45. get { return base.Dictionary; }
  46. set
  47. {
  48. base.Dictionary = value;
  49. foreach (RadiusAttribute a in subAttributes)
  50. {
  51. a.Dictionary = value;
  52. }
  53. }
  54. }
  55. /// <summary>
  56. /// Returns the list of sub-attributes.
  57. /// @return ArrayList of RadiusAttribute objects
  58. /// </summary>
  59. public List<RadiusAttribute> SubAttributes
  60. {
  61. get { return subAttributes; }
  62. }
  63. /// <summary>
  64. /// Adds a sub-attribute to this attribute.
  65. /// @param attribute sub-attribute to add
  66. /// </summary>
  67. public void AddSubAttribute(RadiusAttribute attribute)
  68. {
  69. if (attribute.VendorId != ChildVendorId)
  70. throw new ArgumentException(
  71. "sub attribut has incorrect vendor ID");
  72. subAttributes.Add(attribute);
  73. }
  74. /// <summary>
  75. /// Adds a sub-attribute with the specified name to this attribute.
  76. /// @param name name of the sub-attribute
  77. /// @param value value of the sub-attribute
  78. /// @exception ArgumentException invalid sub-attribute name or value
  79. /// </summary>
  80. public void AddSubAttribute(String name, String value)
  81. {
  82. if (string.IsNullOrEmpty(name))
  83. throw new ArgumentException("type name is empty");
  84. if (string.IsNullOrEmpty(value))
  85. throw new ArgumentException("value is empty");
  86. AttributeType type = Dictionary.GetAttributeTypeByName(name);
  87. if (type == null)
  88. throw new ArgumentException("unknown attribute type '"
  89. + name + "'");
  90. if (type.VendorId == -1)
  91. throw new ArgumentException("attribute type '" + name
  92. + "' is not a Vendor-Specific sub-attribute");
  93. if (type.VendorId != ChildVendorId)
  94. throw new ArgumentException("attribute type '" + name
  95. + "' does not belong to vendor ID " + ChildVendorId);
  96. RadiusAttribute attribute = CreateRadiusAttribute(Dictionary,
  97. ChildVendorId, type.TypeCode);
  98. attribute.Value = value;
  99. AddSubAttribute(attribute);
  100. }
  101. /// <summary>
  102. /// Removes the specified sub-attribute from this attribute.
  103. /// @param attribute RadiusAttribute to remove
  104. /// </summary>
  105. public void RemoveSubAttribute(RadiusAttribute attribute)
  106. {
  107. if (!subAttributes.Remove(attribute))
  108. throw new ArgumentException("no such attribute");
  109. }
  110. /// <summary>
  111. /// Returns all sub-attributes of this attribut which have the given type.
  112. /// @param attributeType type of sub-attributes to get
  113. /// @return list of RadiusAttribute objects, does not return null
  114. /// </summary>
  115. public List<RadiusAttribute> GetSubAttributes(int attributeType)
  116. {
  117. if (attributeType < 1 || attributeType > 255)
  118. throw new ArgumentException(
  119. "sub-attribute type out of bounds");
  120. var result = new List<RadiusAttribute>();
  121. foreach (RadiusAttribute a in subAttributes)
  122. {
  123. if (attributeType == a.Type)
  124. result.Add(a);
  125. }
  126. return result;
  127. }
  128. /// <summary>
  129. /// Returns a sub-attribute of the given type which may only occur once in
  130. /// this attribute.
  131. /// @param type sub-attribute type
  132. /// @return RadiusAttribute object or null if there is no such sub-attribute
  133. /// @throws NotImplementedException if there are multiple occurences of the
  134. /// requested sub-attribute type
  135. /// </summary>
  136. public RadiusAttribute GetSubAttribute(int type)
  137. {
  138. List<RadiusAttribute> attrs = GetSubAttributes(type);
  139. if (attrs.Count > 1)
  140. throw new NotImplementedException(
  141. "multiple sub-attributes of requested type " + type);
  142. else if (attrs.Count == 0)
  143. return null;
  144. else
  145. return attrs[0];
  146. }
  147. /// <summary>
  148. /// Returns a single sub-attribute of the given type name.
  149. /// @param type attribute type name
  150. /// @return RadiusAttribute object or null if there is no such attribute
  151. /// @throws NotImplementedException if the attribute occurs multiple times
  152. /// </summary>
  153. public RadiusAttribute GetSubAttribute(String type)
  154. {
  155. if (string.IsNullOrEmpty(type))
  156. throw new ArgumentException("type name is empty");
  157. AttributeType t = Dictionary.GetAttributeTypeByName(type);
  158. if (t == null)
  159. throw new ArgumentException("unknown attribute type name '"
  160. + type + "'");
  161. if (t.VendorId != ChildVendorId)
  162. throw new ArgumentException("vendor ID mismatch");
  163. return GetSubAttribute(t.TypeCode);
  164. }
  165. /// <summary>
  166. /// Returns the value of the Radius attribute of the given type or null if
  167. /// there is no such attribute.
  168. /// @param type attribute type name
  169. /// @return value of the attribute as a string or null if there is no such
  170. /// attribute
  171. /// @throws ArgumentException if the type name is unknown
  172. /// @throws NotImplementedException attribute occurs multiple times
  173. /// </summary>
  174. public String GetSubAttributeValue(String type)
  175. {
  176. RadiusAttribute attr = GetSubAttribute(type);
  177. if (attr == null)
  178. return null;
  179. else
  180. return attr.Value;
  181. }
  182. /// <summary>
  183. /// Renders this attribute as a byte array.
  184. /// @see TinyRadius.attribute.RadiusAttribute#writeAttribute()
  185. /// </summary>
  186. public override byte[] WriteAttribute()
  187. {
  188. // write vendor ID
  189. var bos = new MemoryStream(255);
  190. bos.WriteByte(Convert.ToByte(ChildVendorId >> 24 & 0x0ff));
  191. bos.WriteByte(Convert.ToByte(ChildVendorId >> 16 & 0x0ff));
  192. bos.WriteByte(Convert.ToByte(ChildVendorId >> 8 & 0x0ff));
  193. bos.WriteByte(Convert.ToByte(ChildVendorId & 0x0ff));
  194. // write sub-attributes
  195. try
  196. {
  197. foreach (RadiusAttribute a in subAttributes)
  198. {
  199. byte[] c = a.WriteAttribute();
  200. bos.Write(c, 0, c.Length);
  201. }
  202. }
  203. catch (IOException ioe)
  204. {
  205. // occurs never
  206. throw new NotImplementedException("error writing data", ioe);
  207. }
  208. // check data Length
  209. byte[] attrData = bos.ToArray();
  210. int len = attrData.Length;
  211. if (len > 253)
  212. throw new NotImplementedException("Vendor-Specific attribute too long: "
  213. + len);
  214. // compose attribute
  215. var attr = new byte[len + 2];
  216. attr[0] = Convert.ToByte(VENDOR_SPECIFIC); // code
  217. attr[1] = (byte) (len + 2); // Length
  218. Array.Copy(attrData, 0, attr, 2, len);
  219. return attr;
  220. }
  221. /// <summary>
  222. /// Reads a Vendor-Specific attribute and decodes the internal sub-attribute
  223. /// structure.
  224. /// @see TinyRadius.attribute.RadiusAttribute#readAttribute(byte[], int,
  225. /// int)
  226. /// </summary>
  227. public override void ReadAttribute(byte[] data, int offset, int length)
  228. {
  229. // check Length
  230. if (length < 6)
  231. throw new RadiusException("Vendor-Specific attribute too short: "
  232. + length);
  233. int vsaCode = data[offset];
  234. int vsaLen = (data[offset + 1] & 0x000000ff) - 6;
  235. if (vsaCode != VENDOR_SPECIFIC)
  236. throw new RadiusException("not a Vendor-Specific attribute");
  237. // read vendor ID and vendor data
  238. /*
  239. * int vendorId = (data[offset + 2] << 24 | data[offset + 3] << 16 |
  240. * data[offset + 4] << 8 | ((int)data[offset + 5] & 0x000000ff));
  241. */
  242. int vendorId = (UnsignedByteToInt(data[offset + 2]) << 24
  243. | UnsignedByteToInt(data[offset + 3]) << 16
  244. | UnsignedByteToInt(data[offset + 4]) << 8 | UnsignedByteToInt(data[offset + 5]));
  245. ChildVendorId = vendorId;
  246. // validate sub-attribute structure
  247. int pos = 0;
  248. int count = 0;
  249. while (pos < vsaLen)
  250. {
  251. if (pos + 1 >= vsaLen)
  252. throw new RadiusException("Vendor-Specific attribute malformed");
  253. // int vsaSubType = data[(offset + 6) + pos] & 0x0ff;
  254. int vsaSubLen = data[(offset + 6) + pos + 1] & 0x0ff;
  255. pos += vsaSubLen;
  256. count++;
  257. }
  258. if (pos != vsaLen)
  259. throw new RadiusException("Vendor-Specific attribute malformed");
  260. subAttributes = new List<RadiusAttribute>(count);
  261. pos = 0;
  262. while (pos < vsaLen)
  263. {
  264. int subtype = data[(offset + 6) + pos] & 0x0ff;
  265. int sublength = data[(offset + 6) + pos + 1] & 0x0ff;
  266. RadiusAttribute a = CreateRadiusAttribute(Dictionary,
  267. vendorId, subtype);
  268. a.ReadAttribute(data, (offset + 6) + pos, sublength);
  269. subAttributes.Add(a);
  270. pos += sublength;
  271. }
  272. }
  273. private static int UnsignedByteToInt(byte b)
  274. {
  275. return b & 0xFF;
  276. }
  277. /// <summary>
  278. /// Returns a string representation for debugging.
  279. /// @see TinyRadius.attribute.RadiusAttribute#toString()
  280. /// </summary>
  281. public override String ToString()
  282. {
  283. var sb = new StringBuilder();
  284. sb.Append("Vendor-Specific: ");
  285. int vendorId = ChildVendorId;
  286. String vendorName = Dictionary.GetVendorName(vendorId);
  287. if (vendorName != null)
  288. {
  289. sb.Append(vendorName)
  290. .Append(" (")
  291. .Append(vendorId)
  292. .Append(")");
  293. }
  294. else
  295. {
  296. sb.Append("vendor ID ");
  297. sb.Append(vendorId);
  298. }
  299. foreach (RadiusAttribute attr in SubAttributes)
  300. {
  301. sb.Append("\n");
  302. sb.Append(attr.ToString());
  303. }
  304. return sb.ToString();
  305. }
  306. }
  307. }