2 // System.Security.Permissions.SiteIdentityPermission.cs
5 // Sebastien Pouliot <sebastien@ximian.com>
7 // Copyright (C) 2003 Motus Technologies. http://www.motus.com
8 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
10 // Permission is hereby granted, free of charge, to any person obtaining
11 // a copy of this software and associated documentation files (the
12 // "Software"), to deal in the Software without restriction, including
13 // without limitation the rights to use, copy, modify, merge, publish,
14 // distribute, sublicense, and/or sell copies of the Software, and to
15 // permit persons to whom the Software is furnished to do so, subject to
16 // the following conditions:
18 // The above copyright notice and this permission notice shall be
19 // included in all copies or substantial portions of the Software.
21 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 using System.Globalization;
31 using System.Runtime.InteropServices;
33 namespace System.Security.Permissions {
37 public sealed class SiteIdentityPermission : CodeAccessPermission, IBuiltInPermission {
39 private const int version = 1;
45 public SiteIdentityPermission (PermissionState state)
47 // false == do not allow Unrestricted for Identity Permissions
48 CheckPermissionState (state, false);
51 public SiteIdentityPermission (string site)
61 throw new NullReferenceException ("No site.");
66 throw new ArgumentException ("Invalid site.");
73 public override IPermission Copy ()
76 return new SiteIdentityPermission (PermissionState.None);
78 return new SiteIdentityPermission (_site);
81 public override void FromXml (SecurityElement esd)
83 // General validation in CodeAccessPermission
84 CheckSecurityElement (esd, "esd", version, version);
85 // Note: we do not (yet) care about the return value
86 // as we only accept version 1 (min/max values)
88 string s = esd.Attribute ("Site");
93 public override IPermission Intersect (IPermission target)
95 SiteIdentityPermission sip = Cast (target);
96 if ((sip == null) || (IsEmpty ()))
99 if (Match (sip._site)) {
100 string s = ((_site.Length > sip._site.Length) ? _site : sip._site);
101 return new SiteIdentityPermission (s);
106 public override bool IsSubsetOf (IPermission target)
108 SiteIdentityPermission sip = Cast (target);
111 if ((_site == null) && (sip._site == null))
113 if ((_site == null) || (sip._site == null))
116 int wildcard = sip._site.IndexOf ('*');
117 if (wildcard == -1) {
119 return (_site == sip._site);
121 return _site.EndsWith (sip._site.Substring (wildcard + 1));
124 public override SecurityElement ToXml ()
126 SecurityElement e = Element (version);
128 e.AddAttribute ("Site", _site);
132 public override IPermission Union (IPermission target)
134 SiteIdentityPermission sip = Cast (target);
135 if ((sip == null) || sip.IsEmpty ())
140 if (Match (sip._site)) {
141 string s = ((_site.Length < sip._site.Length) ? _site : sip._site);
142 return new SiteIdentityPermission (s);
144 throw new ArgumentException (Locale.GetText (
145 "Cannot union two different sites."), "target");
148 // IBuiltInPermission
149 int IBuiltInPermission.GetTokenIndex ()
151 return (int) BuiltInToken.SiteIdentity;
156 private bool IsEmpty ()
158 return (_site == null);
161 private SiteIdentityPermission Cast (IPermission target)
166 SiteIdentityPermission sip = (target as SiteIdentityPermission);
168 ThrowInvalidPermission (target, typeof (SiteIdentityPermission));
174 private static bool[] valid = new bool [94] {
175 /* 33 */ true, false, true, true, true, true, true, true, true, true,
176 /* 43 */ false, false, true, true, false, true, true, true, true, true,
177 /* 53 */ true, true, true, true, false, false, false, false, false, false,
178 /* 63 */ false, true, true, true, true, true, true, true, true, true,
179 /* 73 */ true, true, true, true, true, true, true, true, true, true,
180 /* 83 */ true, true, true, true, true, true, true, true, false, false,
181 /* 93 */ false, true, true, false, true, true, true, true, true, true,
182 /* 103 */ true, true, true, true, true, true, true, true, true, true,
183 /* 113 */ true, true, true, true, true, true, true, true, true, true,
184 /* 123 */ true, false, true, true
187 private bool IsValid (string s)
189 if ((s == null) || (s.Length == 0))
192 for (int i = 0; i < s.Length; i++) {
193 ushort x = (ushort) s [i];
194 if ((x < 33) || (x > 126)) {
198 // special case for wildcards (*)
199 // must be alone or first and followed by a dot
200 if ((s.Length > 1) && ((s [i + 1] != '.') || (i > 0)))
203 if (!valid [x - 33]) {
208 // a lone dot isn't valid
210 return (s [0] != '.');
214 private bool Match (string target)
216 if ((_site == null) || (target == null))
219 int wcs = _site.IndexOf ('*');
220 int wct = target.IndexOf ('*');
222 if ((wcs == -1) && (wct == -1)) {
223 // no wildcard, this is an exact match
224 return (_site == target);
226 else if (wcs == -1) {
227 // only "target" has a wildcard, use it
228 return _site.EndsWith (target.Substring (wct + 1));
230 else if (wct == -1) {
231 // only "this" has a wildcard, use it
232 return target.EndsWith (_site.Substring (wcs + 1));
235 // both have wildcards, partial match with the smallest
236 string s = _site.Substring (wcs + 1);
237 target = target.Substring (wct + 1);
238 if (s.Length > target.Length)
239 return s.EndsWith (target);
241 return target.EndsWith (s);