1 | /*
|
---|
2 | Adobe Systems Incorporated(r) Source Code License Agreement
|
---|
3 | Copyright(c) 2005 Adobe Systems Incorporated. All rights reserved.
|
---|
4 |
|
---|
5 | Please read this Source Code License Agreement carefully before using
|
---|
6 | the source code.
|
---|
7 |
|
---|
8 | Adobe Systems Incorporated grants to you a perpetual, worldwide, non-exclusive,
|
---|
9 | no-charge, royalty-free, irrevocable copyright license, to reproduce,
|
---|
10 | prepare derivative works of, publicly display, publicly perform, and
|
---|
11 | distribute this source code and such derivative works in source or
|
---|
12 | object code form without any attribution requirements.
|
---|
13 |
|
---|
14 | The name "Adobe Systems Incorporated" must not be used to endorse or promote products
|
---|
15 | derived from the source code without prior written permission.
|
---|
16 |
|
---|
17 | You agree to indemnify, hold harmless and defend Adobe Systems Incorporated from and
|
---|
18 | against any loss, damage, claims or lawsuits, including attorney's
|
---|
19 | fees that arise or result from your use or distribution of the source
|
---|
20 | code.
|
---|
21 |
|
---|
22 | THIS SOURCE CODE IS PROVIDED "AS IS" AND "WITH ALL FAULTS", WITHOUT
|
---|
23 | ANY TECHNICAL SUPPORT OR ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING,
|
---|
24 | BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
---|
25 | FOR A PARTICULAR PURPOSE ARE DISCLAIMED. ALSO, THERE IS NO WARRANTY OF
|
---|
26 | NON-INFRINGEMENT, TITLE OR QUIET ENJOYMENT. IN NO EVENT SHALL MACROMEDIA
|
---|
27 | OR ITS SUPPLIERS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
---|
28 | EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
---|
29 | PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
---|
30 | OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
---|
31 | WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
---|
32 | OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOURCE CODE, EVEN IF
|
---|
33 | ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
---|
34 | */
|
---|
35 |
|
---|
36 | package com.adobe.net
|
---|
37 | {
|
---|
38 | import flash.utils.ByteArray;
|
---|
39 |
|
---|
40 | /**
|
---|
41 | * This class implements functions and utilities for working with URI's
|
---|
42 | * (Universal Resource Identifiers). For technical description of the
|
---|
43 | * URI syntax, please see RFC 3986 at http://www.ietf.org/rfc/rfc3986.txt
|
---|
44 | * or do a web search for "rfc 3986".
|
---|
45 | *
|
---|
46 | * <p>The most important aspect of URI's to understand is that URI's
|
---|
47 | * and URL's are not strings. URI's are complex data structures that
|
---|
48 | * encapsulate many pieces of information. The string version of a
|
---|
49 | * URI is the serialized representation of that data structure. This
|
---|
50 | * string serialization is used to provide a human readable
|
---|
51 | * representation and a means to transport the data over the network
|
---|
52 | * where it can then be parsed back into its' component parts.</p>
|
---|
53 | *
|
---|
54 | * <p>URI's fall into one of three categories:
|
---|
55 | * <ul>
|
---|
56 | * <li><scheme>:<scheme-specific-part>#<fragment> (non-hierarchical)</li>
|
---|
57 | * <li><scheme>:<authority><path>?<query>#<fragment> (hierarchical)</li>
|
---|
58 | * <li><path>?<query>#<fragment> (relative hierarchical)</li>
|
---|
59 | * </ul></p>
|
---|
60 | *
|
---|
61 | * <p>The query and fragment parts are optional.</p>
|
---|
62 | *
|
---|
63 | * <p>This class supports both non-hierarchical and hierarchical URI's</p>
|
---|
64 | *
|
---|
65 | * <p>This class is intended to be used "as-is" for the vast majority
|
---|
66 | * of common URI's. However, if your application requires a custom
|
---|
67 | * URI syntax (e.g. custom query syntax or special handling of
|
---|
68 | * non-hierarchical URI's), this class can be fully subclassed. If you
|
---|
69 | * intended to subclass URI, please see the source code for complete
|
---|
70 | * documation on protected members and protected fuctions.</p>
|
---|
71 | *
|
---|
72 | * @langversion ActionScript 3.0
|
---|
73 | * @playerversion Flash 9.0
|
---|
74 | */
|
---|
75 | public class URI
|
---|
76 | {
|
---|
77 | // Here we define which characters must be escaped for each
|
---|
78 | // URI part. The characters that must be escaped for each
|
---|
79 | // part differ depending on what would cause ambiguous parsing.
|
---|
80 | // RFC 3986 sec. 2.4 states that characters should only be
|
---|
81 | // encoded when they would conflict with subcomponent delimiters.
|
---|
82 | // We don't want to over-do the escaping. We only want to escape
|
---|
83 | // the minimum needed to prevent parsing problems.
|
---|
84 |
|
---|
85 | // space and % must be escaped in all cases. '%' is the delimiter
|
---|
86 | // for escaped characters.
|
---|
87 | public static const URImustEscape:String = " %";
|
---|
88 |
|
---|
89 | // Baseline of what characters must be escaped
|
---|
90 | public static const URIbaselineEscape:String = URImustEscape + ":?#/@";
|
---|
91 |
|
---|
92 | // Characters that must be escaped in the part part.
|
---|
93 | public static const URIpathEscape:String = URImustEscape + "?#";
|
---|
94 |
|
---|
95 | // Characters that must be escaped in the query part, if setting
|
---|
96 | // the query as a whole string. If the query is set by
|
---|
97 | // name/value, URIqueryPartEscape is used instead.
|
---|
98 | public static const URIqueryEscape:String = URImustEscape + "#";
|
---|
99 |
|
---|
100 | // This is what each name/value pair must escape "&=" as well
|
---|
101 | // so they don't conflict with the "param=value¶m2=value2"
|
---|
102 | // syntax.
|
---|
103 | public static const URIqueryPartEscape:String = URImustEscape + "#&=";
|
---|
104 |
|
---|
105 | // Non-hierarchical URI's can have query and fragment parts, but
|
---|
106 | // we also want to prevent '/' otherwise it might end up looking
|
---|
107 | // like a hierarchical URI to the parser.
|
---|
108 | public static const URInonHierEscape:String = URImustEscape + "?#/";
|
---|
109 |
|
---|
110 | // Baseline uninitialized setting for the URI scheme.
|
---|
111 | public static const UNKNOWN_SCHEME:String = "unknown";
|
---|
112 |
|
---|
113 | // The following bitmaps are used for performance enhanced
|
---|
114 | // character escaping.
|
---|
115 |
|
---|
116 | // Baseline characters that need to be escaped. Many parts use
|
---|
117 | // this.
|
---|
118 | protected static const URIbaselineExcludedBitmap:URIEncodingBitmap =
|
---|
119 | new URIEncodingBitmap(URIbaselineEscape);
|
---|
120 |
|
---|
121 | // Scheme escaping bitmap
|
---|
122 | protected static const URIschemeExcludedBitmap:URIEncodingBitmap =
|
---|
123 | URIbaselineExcludedBitmap;
|
---|
124 |
|
---|
125 | // User/pass escaping bitmap
|
---|
126 | protected static const URIuserpassExcludedBitmap:URIEncodingBitmap =
|
---|
127 | URIbaselineExcludedBitmap;
|
---|
128 |
|
---|
129 | // Authority escaping bitmap
|
---|
130 | protected static const URIauthorityExcludedBitmap:URIEncodingBitmap =
|
---|
131 | URIbaselineExcludedBitmap;
|
---|
132 |
|
---|
133 | // Port escaping bitmap
|
---|
134 | protected static const URIportExludedBitmap:URIEncodingBitmap =
|
---|
135 | URIbaselineExcludedBitmap;
|
---|
136 |
|
---|
137 | // Path escaping bitmap
|
---|
138 | protected static const URIpathExcludedBitmap:URIEncodingBitmap =
|
---|
139 | new URIEncodingBitmap(URIpathEscape);
|
---|
140 |
|
---|
141 | // Query (whole) escaping bitmap
|
---|
142 | protected static const URIqueryExcludedBitmap:URIEncodingBitmap =
|
---|
143 | new URIEncodingBitmap(URIqueryEscape);
|
---|
144 |
|
---|
145 | // Query (individual parts) escaping bitmap
|
---|
146 | protected static const URIqueryPartExcludedBitmap:URIEncodingBitmap =
|
---|
147 | new URIEncodingBitmap(URIqueryPartEscape);
|
---|
148 |
|
---|
149 | // Fragments are the last part in the URI. They only need to
|
---|
150 | // escape space, '#', and '%'. Turns out that is what query
|
---|
151 | // uses too.
|
---|
152 | protected static const URIfragmentExcludedBitmap:URIEncodingBitmap =
|
---|
153 | URIqueryExcludedBitmap;
|
---|
154 |
|
---|
155 | // Characters that need to be escaped in the non-hierarchical part
|
---|
156 | protected static const URInonHierexcludedBitmap:URIEncodingBitmap =
|
---|
157 | new URIEncodingBitmap(URInonHierEscape);
|
---|
158 |
|
---|
159 | // Values used by getRelation()
|
---|
160 | public static const NOT_RELATED:int = 0;
|
---|
161 | public static const CHILD:int = 1;
|
---|
162 | public static const EQUAL:int = 2;
|
---|
163 | public static const PARENT:int = 3;
|
---|
164 |
|
---|
165 | //-------------------------------------------------------------------
|
---|
166 | // protected class members
|
---|
167 | //-------------------------------------------------------------------
|
---|
168 | protected var _valid:Boolean = false;
|
---|
169 | protected var _relative:Boolean = false;
|
---|
170 | protected var _scheme:String = "";
|
---|
171 | protected var _authority:String = "";
|
---|
172 | protected var _username:String = "";
|
---|
173 | protected var _password:String = "";
|
---|
174 | protected var _port:String = "";
|
---|
175 | protected var _path:String = "";
|
---|
176 | protected var _query:String = "";
|
---|
177 | protected var _fragment:String = "";
|
---|
178 | protected var _nonHierarchical:String = "";
|
---|
179 | protected static var _resolver:IURIResolver = null;
|
---|
180 |
|
---|
181 |
|
---|
182 | /**
|
---|
183 | * URI Constructor. If no string is given, this will initialize
|
---|
184 | * this URI object to a blank URI.
|
---|
185 | */
|
---|
186 | public function URI(uri:String = null) : void
|
---|
187 | {
|
---|
188 | if (uri == null)
|
---|
189 | initialize();
|
---|
190 | else
|
---|
191 | constructURI(uri);
|
---|
192 | }
|
---|
193 |
|
---|
194 |
|
---|
195 | /**
|
---|
196 | * @private
|
---|
197 | * Method that loads the URI from the given string.
|
---|
198 | */
|
---|
199 | protected function constructURI(uri:String) : Boolean
|
---|
200 | {
|
---|
201 | if (!parseURI(uri))
|
---|
202 | _valid = false;
|
---|
203 |
|
---|
204 | return isValid();
|
---|
205 | }
|
---|
206 |
|
---|
207 |
|
---|
208 | /**
|
---|
209 | * @private Private initializiation.
|
---|
210 | */
|
---|
211 | protected function initialize() : void
|
---|
212 | {
|
---|
213 | _valid = false;
|
---|
214 | _relative = false;
|
---|
215 |
|
---|
216 | _scheme = UNKNOWN_SCHEME;
|
---|
217 | _authority = "";
|
---|
218 | _username = "";
|
---|
219 | _password = "";
|
---|
220 | _port = "";
|
---|
221 | _path = "";
|
---|
222 | _query = "";
|
---|
223 | _fragment = "";
|
---|
224 |
|
---|
225 | _nonHierarchical = "";
|
---|
226 | }
|
---|
227 |
|
---|
228 | /**
|
---|
229 | * @private Accessor to explicitly set/get the hierarchical
|
---|
230 | * state of the URI.
|
---|
231 | */
|
---|
232 | protected function set hierState(state:Boolean) : void
|
---|
233 | {
|
---|
234 | if (state)
|
---|
235 | {
|
---|
236 | // Clear the non-hierarchical data
|
---|
237 | _nonHierarchical = "";
|
---|
238 |
|
---|
239 | // Also set the state vars while we are at it
|
---|
240 | if (_scheme == "" || _scheme == UNKNOWN_SCHEME)
|
---|
241 | _relative = true;
|
---|
242 | else
|
---|
243 | _relative = false;
|
---|
244 |
|
---|
245 | if (_authority.length == 0 && _path.length == 0)
|
---|
246 | _valid = false;
|
---|
247 | else
|
---|
248 | _valid = true;
|
---|
249 | }
|
---|
250 | else
|
---|
251 | {
|
---|
252 | // Clear the hierarchical data
|
---|
253 | _authority = "";
|
---|
254 | _username = "";
|
---|
255 | _password = "";
|
---|
256 | _port = "";
|
---|
257 | _path = "";
|
---|
258 |
|
---|
259 | _relative = false;
|
---|
260 |
|
---|
261 | if (_scheme == "" || _scheme == UNKNOWN_SCHEME)
|
---|
262 | _valid = false;
|
---|
263 | else
|
---|
264 | _valid = true;
|
---|
265 | }
|
---|
266 | }
|
---|
267 | protected function get hierState() : Boolean
|
---|
268 | {
|
---|
269 | return (_nonHierarchical.length == 0);
|
---|
270 | }
|
---|
271 |
|
---|
272 |
|
---|
273 | /**
|
---|
274 | * @private Functions that performs some basic consistency validation.
|
---|
275 | */
|
---|
276 | protected function validateURI() : Boolean
|
---|
277 | {
|
---|
278 | // Check the scheme
|
---|
279 | if (isAbsolute())
|
---|
280 | {
|
---|
281 | if (_scheme.length <= 1 || _scheme == UNKNOWN_SCHEME)
|
---|
282 | {
|
---|
283 | // we probably parsed a C:\ type path or no scheme
|
---|
284 | return false;
|
---|
285 | }
|
---|
286 | else if (verifyAlpha(_scheme) == false)
|
---|
287 | return false; // Scheme contains bad characters
|
---|
288 | }
|
---|
289 |
|
---|
290 | if (hierState)
|
---|
291 | {
|
---|
292 | if (_path.search('\\') != -1)
|
---|
293 | return false; // local path
|
---|
294 | else if (isRelative() == false && _scheme == UNKNOWN_SCHEME)
|
---|
295 | return false; // It's an absolute URI, but it has a bad scheme
|
---|
296 | }
|
---|
297 | else
|
---|
298 | {
|
---|
299 | if (_nonHierarchical.search('\\') != -1)
|
---|
300 | return false; // some kind of local path
|
---|
301 | }
|
---|
302 |
|
---|
303 | // Looks like it's ok.
|
---|
304 | return true;
|
---|
305 | }
|
---|
306 |
|
---|
307 |
|
---|
308 | /**
|
---|
309 | * @private
|
---|
310 | *
|
---|
311 | * Given a URI in string format, parse that sucker into its basic
|
---|
312 | * components and assign them to this object. A URI is of the form:
|
---|
313 | * <scheme>:<authority><path>?<query>#<fragment>
|
---|
314 | *
|
---|
315 | * For simplicity, we parse the URI in the following order:
|
---|
316 | *
|
---|
317 | * 1. Fragment (anchors)
|
---|
318 | * 2. Query (CGI stuff)
|
---|
319 | * 3. Scheme ("http")
|
---|
320 | * 4. Authority (host name)
|
---|
321 | * 5. Username/Password (if any)
|
---|
322 | * 6. Port (server port if any)
|
---|
323 | * 7. Path (/homepages/mypage.html)
|
---|
324 | *
|
---|
325 | * The reason for this order is to minimize any parsing ambiguities.
|
---|
326 | * Fragments and queries can contain almost anything (they are parts
|
---|
327 | * that can contain custom data with their own syntax). Parsing
|
---|
328 | * them out first removes a large chance of parsing errors. This
|
---|
329 | * method expects well formed URI's, but performing the parse in
|
---|
330 | * this order makes us a little more tolerant of user error.
|
---|
331 | *
|
---|
332 | * REGEXP
|
---|
333 | * Why doesn't this use regular expressions to parse the URI? We
|
---|
334 | * have found that in a real world scenario, URI's are not always
|
---|
335 | * well formed. Sometimes characters that should have been escaped
|
---|
336 | * are not, and those situations would break a regexp pattern. This
|
---|
337 | * function attempts to be smart about what it is parsing based on
|
---|
338 | * location of characters relative to eachother. This function has
|
---|
339 | * been proven through real-world use to parse the vast majority
|
---|
340 | * of URI's correctly.
|
---|
341 | *
|
---|
342 | * NOTE
|
---|
343 | * It is assumed that the string in URI form is escaped. This function
|
---|
344 | * does not escape anything. If you constructed the URI string by
|
---|
345 | * hand, and used this to parse in the URI and still need it escaped,
|
---|
346 | * call forceEscape() on your URI object.
|
---|
347 | *
|
---|
348 | * Parsing Assumptions
|
---|
349 | * This routine assumes that the URI being passed is well formed.
|
---|
350 | * Passing things like local paths, malformed URI's, and the such
|
---|
351 | * will result in parsing errors. This function can handle
|
---|
352 | * - absolute hierarchical (e.g. "http://something.com/index.html),
|
---|
353 | * - relative hierarchical (e.g. "../images/flower.gif"), or
|
---|
354 | * - non-hierarchical URIs (e.g. "mailto:jsmith@fungoo.com").
|
---|
355 | *
|
---|
356 | * Anything else will probably result in a parsing error, or a bogus
|
---|
357 | * URI object.
|
---|
358 | *
|
---|
359 | * Note that non-hierarchical URIs *MUST* have a scheme, otherwise
|
---|
360 | * they will be mistaken for relative URI's.
|
---|
361 | *
|
---|
362 | * If you are not sure what is being passed to you (like manually
|
---|
363 | * entered text from UI), you can construct a blank URI object and
|
---|
364 | * call unknownToURI() passing in the unknown string.
|
---|
365 | *
|
---|
366 | * @return true if successful, false if there was some kind of
|
---|
367 | * parsing error
|
---|
368 | */
|
---|
369 | protected function parseURI(uri:String) : Boolean
|
---|
370 | {
|
---|
371 | var baseURI:String = uri;
|
---|
372 | var index:int, index2:int;
|
---|
373 |
|
---|
374 | // Make sure this object is clean before we start. If it was used
|
---|
375 | // before and we are now parsing a new URI, we don't want any stale
|
---|
376 | // info lying around.
|
---|
377 | initialize();
|
---|
378 |
|
---|
379 | // Remove any fragments (anchors) from the URI
|
---|
380 | index = baseURI.indexOf("#");
|
---|
381 | if (index != -1)
|
---|
382 | {
|
---|
383 | // Store the fragment piece if any
|
---|
384 | if (baseURI.length > (index + 1)) // +1 is to skip the '#'
|
---|
385 | _fragment = baseURI.substr(index + 1, baseURI.length - (index + 1));
|
---|
386 |
|
---|
387 | // Trim off the fragment
|
---|
388 | baseURI = baseURI.substr(0, index);
|
---|
389 | }
|
---|
390 |
|
---|
391 | // We need to strip off any CGI parameters (eg '?param=bob')
|
---|
392 | index = baseURI.indexOf("?");
|
---|
393 | if (index != -1)
|
---|
394 | {
|
---|
395 | if (baseURI.length > (index + 1))
|
---|
396 | _query = baseURI.substr(index + 1, baseURI.length - (index + 1)); // +1 is to skip the '?'
|
---|
397 |
|
---|
398 | // Trim off the query
|
---|
399 | baseURI = baseURI.substr(0, index);
|
---|
400 | }
|
---|
401 |
|
---|
402 | // Now try to find the scheme part
|
---|
403 | index = baseURI.search(':');
|
---|
404 | index2 = baseURI.search('/');
|
---|
405 |
|
---|
406 | var containsColon:Boolean = (index != -1);
|
---|
407 | var containsSlash:Boolean = (index2 != -1);
|
---|
408 |
|
---|
409 | // This value is indeterminate if "containsColon" is false.
|
---|
410 | // (if there is no colon, does the slash come before or
|
---|
411 | // after said non-existing colon?)
|
---|
412 | var colonBeforeSlash:Boolean = (!containsSlash || index < index2);
|
---|
413 |
|
---|
414 | // If it has a colon and it's before the first slash, we will treat
|
---|
415 | // it as a scheme. If a slash is before a colon, there must be a
|
---|
416 | // stray colon in a path or something. In which case, the colon is
|
---|
417 | // not the separator for the scheme. Technically, we could consider
|
---|
418 | // this an error, but since this is not an ambiguous state (we know
|
---|
419 | // 100% that this has no scheme), we will keep going.
|
---|
420 | if (containsColon && colonBeforeSlash)
|
---|
421 | {
|
---|
422 | // We found a scheme
|
---|
423 | _scheme = baseURI.substr(0, index);
|
---|
424 |
|
---|
425 | // Normalize the scheme
|
---|
426 | _scheme = _scheme.toLowerCase();
|
---|
427 |
|
---|
428 | baseURI = baseURI.substr(index + 1);
|
---|
429 |
|
---|
430 | if (baseURI.substr(0, 2) == "//")
|
---|
431 | {
|
---|
432 | // This is a hierarchical URI
|
---|
433 | _nonHierarchical = "";
|
---|
434 |
|
---|
435 | // Trim off the "//"
|
---|
436 | baseURI = baseURI.substr(2, baseURI.length - 2);
|
---|
437 | }
|
---|
438 | else
|
---|
439 | {
|
---|
440 | // This is a non-hierarchical URI like "mailto:bob@mail.com"
|
---|
441 | _nonHierarchical = baseURI;
|
---|
442 |
|
---|
443 | if ((_valid = validateURI()) == false)
|
---|
444 | initialize(); // Bad URI. Clear it.
|
---|
445 |
|
---|
446 | // No more parsing to do for this case
|
---|
447 | return isValid();
|
---|
448 | }
|
---|
449 | }
|
---|
450 | else
|
---|
451 | {
|
---|
452 | // No scheme. We will consider this a relative URI
|
---|
453 | _scheme = "";
|
---|
454 | _relative = true;
|
---|
455 | _nonHierarchical = "";
|
---|
456 | }
|
---|
457 |
|
---|
458 | // Ok, what we have left is everything after the <scheme>://
|
---|
459 |
|
---|
460 | // Now that we have stripped off any query and fragment parts, we
|
---|
461 | // need to split the authority from the path
|
---|
462 |
|
---|
463 | if (isRelative())
|
---|
464 | {
|
---|
465 | // Don't bother looking for the authority. It's a relative URI
|
---|
466 | _authority = "";
|
---|
467 | _port = "";
|
---|
468 | _path = baseURI;
|
---|
469 | }
|
---|
470 | else
|
---|
471 | {
|
---|
472 | // Check for malformed UNC style file://///server/type/path/
|
---|
473 | // By the time we get here, we have already trimmed the "file://"
|
---|
474 | // so baseURI will be ///server/type/path. If baseURI only
|
---|
475 | // has one slash, we leave it alone because that is valid (that
|
---|
476 | // is the case of "file:///path/to/file.txt" where there is no
|
---|
477 | // server - implicit "localhost").
|
---|
478 | if (baseURI.substr(0, 2) == "//")
|
---|
479 | {
|
---|
480 | // Trim all leading slashes
|
---|
481 | while(baseURI.charAt(0) == "/")
|
---|
482 | baseURI = baseURI.substr(1, baseURI.length - 1);
|
---|
483 | }
|
---|
484 |
|
---|
485 | index = baseURI.search('/');
|
---|
486 | if (index == -1)
|
---|
487 | {
|
---|
488 | // No path. We must have passed something like "http://something.com"
|
---|
489 | _authority = baseURI;
|
---|
490 | _path = "";
|
---|
491 | }
|
---|
492 | else
|
---|
493 | {
|
---|
494 | _authority = baseURI.substr(0, index);
|
---|
495 | _path = baseURI.substr(index, baseURI.length - index);
|
---|
496 | }
|
---|
497 |
|
---|
498 | // Check to see if the URI has any username or password information.
|
---|
499 | // For example: ftp://username:password@server.com
|
---|
500 | index = _authority.search('@');
|
---|
501 | if (index != -1)
|
---|
502 | {
|
---|
503 | // We have a username and possibly a password
|
---|
504 | _username = _authority.substr(0, index);
|
---|
505 |
|
---|
506 | // Remove the username/password from the authority
|
---|
507 | _authority = _authority.substr(index + 1); // Skip the '@'
|
---|
508 |
|
---|
509 | // Now check to see if the username also has a password
|
---|
510 | index = _username.search(':');
|
---|
511 | if (index != -1)
|
---|
512 | {
|
---|
513 | _password = _username.substring(index + 1, _username.length);
|
---|
514 | _username = _username.substr(0, index);
|
---|
515 | }
|
---|
516 | else
|
---|
517 | _password = "";
|
---|
518 | }
|
---|
519 | else
|
---|
520 | {
|
---|
521 | _username = "";
|
---|
522 | _password = "";
|
---|
523 | }
|
---|
524 |
|
---|
525 | // Lastly, check to see if the authorty has a port number.
|
---|
526 | // This is parsed after the username/password to avoid conflicting
|
---|
527 | // with the ':' in the 'username:password' if one exists.
|
---|
528 | index = _authority.search(':');
|
---|
529 | if (index != -1)
|
---|
530 | {
|
---|
531 | _port = _authority.substring(index + 1, _authority.length); // skip the ':'
|
---|
532 | _authority = _authority.substr(0, index);
|
---|
533 | }
|
---|
534 | else
|
---|
535 | {
|
---|
536 | _port = "";
|
---|
537 | }
|
---|
538 |
|
---|
539 | // Lastly, normalize the authority. Domain names
|
---|
540 | // are case insensitive.
|
---|
541 | _authority = _authority.toLowerCase();
|
---|
542 | }
|
---|
543 |
|
---|
544 | if ((_valid = validateURI()) == false)
|
---|
545 | initialize(); // Bad URI. Clear it
|
---|
546 |
|
---|
547 | return isValid();
|
---|
548 | }
|
---|
549 |
|
---|
550 |
|
---|
551 | /********************************************************************
|
---|
552 | * Copy function.
|
---|
553 | */
|
---|
554 | public function copyURI(uri:URI) : void
|
---|
555 | {
|
---|
556 | this._scheme = uri._scheme;
|
---|
557 | this._authority = uri._authority;
|
---|
558 | this._username = uri._username;
|
---|
559 | this._password = uri._password;
|
---|
560 | this._port = uri._port;
|
---|
561 | this._path = uri._path;
|
---|
562 | this._query = uri._query;
|
---|
563 | this._fragment = uri._fragment;
|
---|
564 | this._nonHierarchical = uri._nonHierarchical;
|
---|
565 |
|
---|
566 | this._valid = uri._valid;
|
---|
567 | this._relative = uri._relative;
|
---|
568 | }
|
---|
569 |
|
---|
570 |
|
---|
571 | /**
|
---|
572 | * @private
|
---|
573 | * Checks if the given string only contains a-z or A-Z.
|
---|
574 | */
|
---|
575 | protected function verifyAlpha(str:String) : Boolean
|
---|
576 | {
|
---|
577 | var pattern:RegExp = /[^a-z]/;
|
---|
578 | var index:int;
|
---|
579 |
|
---|
580 | str = str.toLowerCase();
|
---|
581 | index = str.search(pattern);
|
---|
582 |
|
---|
583 | if (index == -1)
|
---|
584 | return true;
|
---|
585 | else
|
---|
586 | return false;
|
---|
587 | }
|
---|
588 |
|
---|
589 | /**
|
---|
590 | * Is this a valid URI?
|
---|
591 | *
|
---|
592 | * @return true if this object represents a valid URI, false
|
---|
593 | * otherwise.
|
---|
594 | */
|
---|
595 | public function isValid() : Boolean
|
---|
596 | {
|
---|
597 | return this._valid;
|
---|
598 | }
|
---|
599 |
|
---|
600 |
|
---|
601 | /**
|
---|
602 | * Is this URI an absolute URI? An absolute URI is a complete, fully
|
---|
603 | * qualified reference to a resource. e.g. http://site.com/index.htm
|
---|
604 | * Non-hierarchical URI's are always absolute.
|
---|
605 | */
|
---|
606 | public function isAbsolute() : Boolean
|
---|
607 | {
|
---|
608 | return !this._relative;
|
---|
609 | }
|
---|
610 |
|
---|
611 |
|
---|
612 | /**
|
---|
613 | * Is this URI a relative URI? Relative URI's do not have a scheme
|
---|
614 | * and only contain a relative path with optional anchor and query
|
---|
615 | * parts. e.g. "../reports/index.htm". Non-hierarchical URI's
|
---|
616 | * will never be relative.
|
---|
617 | */
|
---|
618 | public function isRelative() : Boolean
|
---|
619 | {
|
---|
620 | return this._relative;
|
---|
621 | }
|
---|
622 |
|
---|
623 |
|
---|
624 | /**
|
---|
625 | * Does this URI point to a resource that is a directory/folder?
|
---|
626 | * The URI specification dictates that any path that ends in a slash
|
---|
627 | * is a directory. This is needed to be able to perform correct path
|
---|
628 | * logic when combining relative URI's with absolute URI's to
|
---|
629 | * obtain the correct absolute URI to a resource.
|
---|
630 | *
|
---|
631 | * @see URI.chdir
|
---|
632 | *
|
---|
633 | * @return true if this URI represents a directory resource, false
|
---|
634 | * if this URI represents a file resource.
|
---|
635 | */
|
---|
636 | public function isDirectory() : Boolean
|
---|
637 | {
|
---|
638 | if (_path.length == 0)
|
---|
639 | return false;
|
---|
640 |
|
---|
641 | return (_path.charAt(path.length - 1) == '/');
|
---|
642 | }
|
---|
643 |
|
---|
644 |
|
---|
645 | /**
|
---|
646 | * Is this URI a hierarchical URI? URI's can be
|
---|
647 | */
|
---|
648 | public function isHierarchical() : Boolean
|
---|
649 | {
|
---|
650 | return hierState;
|
---|
651 | }
|
---|
652 |
|
---|
653 |
|
---|
654 | /**
|
---|
655 | * The scheme of the URI.
|
---|
656 | */
|
---|
657 | public function get scheme() : String
|
---|
658 | {
|
---|
659 | return URI.unescapeChars(_scheme);
|
---|
660 | }
|
---|
661 | public function set scheme(schemeStr:String) : void
|
---|
662 | {
|
---|
663 | // Normalize the scheme
|
---|
664 | var normalized:String = schemeStr.toLowerCase();
|
---|
665 | _scheme = URI.fastEscapeChars(normalized, URI.URIschemeExcludedBitmap);
|
---|
666 | }
|
---|
667 |
|
---|
668 |
|
---|
669 | /**
|
---|
670 | * The authority (host) of the URI. Only valid for
|
---|
671 | * hierarchical URI's. If the URI is relative, this will
|
---|
672 | * be an empty string. When setting this value, the string
|
---|
673 | * given is assumed to be unescaped. When retrieving this
|
---|
674 | * value, the resulting string is unescaped.
|
---|
675 | */
|
---|
676 | public function get authority() : String
|
---|
677 | {
|
---|
678 | return URI.unescapeChars(_authority);
|
---|
679 | }
|
---|
680 | public function set authority(authorityStr:String) : void
|
---|
681 | {
|
---|
682 | // Normalize the authority
|
---|
683 | authorityStr = authorityStr.toLowerCase();
|
---|
684 |
|
---|
685 | _authority = URI.fastEscapeChars(authorityStr,
|
---|
686 | URI.URIauthorityExcludedBitmap);
|
---|
687 |
|
---|
688 | // Only hierarchical URI's can have an authority, make
|
---|
689 | // sure this URI is of the proper format.
|
---|
690 | this.hierState = true;
|
---|
691 | }
|
---|
692 |
|
---|
693 |
|
---|
694 | /**
|
---|
695 | * The username of the URI. Only valid for hierarchical
|
---|
696 | * URI's. If the URI is relative, this will be an empty
|
---|
697 | * string.
|
---|
698 | *
|
---|
699 | * <p>The URI specification allows for authentication
|
---|
700 | * credentials to be embedded in the URI as such:</p>
|
---|
701 | *
|
---|
702 | * <p>http://user:passwd@host/path/to/file.htm</p>
|
---|
703 | *
|
---|
704 | * <p>When setting this value, the string
|
---|
705 | * given is assumed to be unescaped. When retrieving this
|
---|
706 | * value, the resulting string is unescaped.</p>
|
---|
707 | */
|
---|
708 | public function get username() : String
|
---|
709 | {
|
---|
710 | return URI.unescapeChars(_username);
|
---|
711 | }
|
---|
712 | public function set username(usernameStr:String) : void
|
---|
713 | {
|
---|
714 | _username = URI.fastEscapeChars(usernameStr, URI.URIuserpassExcludedBitmap);
|
---|
715 |
|
---|
716 | // Only hierarchical URI's can have a username.
|
---|
717 | this.hierState = true;
|
---|
718 | }
|
---|
719 |
|
---|
720 |
|
---|
721 | /**
|
---|
722 | * The password of the URI. Similar to username.
|
---|
723 | * @see URI.username
|
---|
724 | */
|
---|
725 | public function get password() : String
|
---|
726 | {
|
---|
727 | return URI.unescapeChars(_password);
|
---|
728 | }
|
---|
729 | public function set password(passwordStr:String) : void
|
---|
730 | {
|
---|
731 | _password = URI.fastEscapeChars(passwordStr,
|
---|
732 | URI.URIuserpassExcludedBitmap);
|
---|
733 |
|
---|
734 | // Only hierarchical URI's can have a password.
|
---|
735 | this.hierState = true;
|
---|
736 | }
|
---|
737 |
|
---|
738 |
|
---|
739 | /**
|
---|
740 | * The host port number. Only valid for hierarchical URI's. If
|
---|
741 | * the URI is relative, this will be an empty string. URI's can
|
---|
742 | * contain the port number of the remote host:
|
---|
743 | *
|
---|
744 | * <p>http://site.com:8080/index.htm</p>
|
---|
745 | */
|
---|
746 | public function get port() : String
|
---|
747 | {
|
---|
748 | return URI.unescapeChars(_port);
|
---|
749 | }
|
---|
750 | public function set port(portStr:String) : void
|
---|
751 | {
|
---|
752 | _port = URI.escapeChars(portStr);
|
---|
753 |
|
---|
754 | // Only hierarchical URI's can have a port.
|
---|
755 | this.hierState = true;
|
---|
756 | }
|
---|
757 |
|
---|
758 |
|
---|
759 | /**
|
---|
760 | * The path portion of the URI. Only valid for hierarchical
|
---|
761 | * URI's. When setting this value, the string
|
---|
762 | * given is assumed to be unescaped. When retrieving this
|
---|
763 | * value, the resulting string is unescaped.
|
---|
764 | *
|
---|
765 | * <p>The path portion can be in one of two formats. 1) an absolute
|
---|
766 | * path, or 2) a relative path. An absolute path starts with a
|
---|
767 | * slash ('/'), a relative path does not.</p>
|
---|
768 | *
|
---|
769 | * <p>An absolute path may look like:</p>
|
---|
770 | * <listing>/full/path/to/my/file.htm</listing>
|
---|
771 | *
|
---|
772 | * <p>A relative path may look like:</p>
|
---|
773 | * <listing>
|
---|
774 | * path/to/my/file.htm
|
---|
775 | * ../images/logo.gif
|
---|
776 | * ../../reports/index.htm
|
---|
777 | * </listing>
|
---|
778 | *
|
---|
779 | * <p>Paths can be absolute or relative. Note that this not the same as
|
---|
780 | * an absolute or relative URI. An absolute URI can only have absolute
|
---|
781 | * paths. For example:</p>
|
---|
782 | *
|
---|
783 | * <listing>http:/site.com/path/to/file.htm</listing>
|
---|
784 | *
|
---|
785 | * <p>This absolute URI has an absolute path of "/path/to/file.htm".</p>
|
---|
786 | *
|
---|
787 | * <p>Relative URI's can have either absolute paths or relative paths.
|
---|
788 | * All of the following relative URI's are valid:</p>
|
---|
789 | *
|
---|
790 | * <listing>
|
---|
791 | * /absolute/path/to/file.htm
|
---|
792 | * path/to/file.htm
|
---|
793 | * ../path/to/file.htm
|
---|
794 | * </listing>
|
---|
795 | */
|
---|
796 | public function get path() : String
|
---|
797 | {
|
---|
798 | return URI.unescapeChars(_path);
|
---|
799 | }
|
---|
800 | public function set path(pathStr:String) : void
|
---|
801 | {
|
---|
802 | this._path = URI.fastEscapeChars(pathStr, URI.URIpathExcludedBitmap);
|
---|
803 |
|
---|
804 | if (this._scheme == UNKNOWN_SCHEME)
|
---|
805 | {
|
---|
806 | // We set the path. This is a valid URI now.
|
---|
807 | this._scheme = "";
|
---|
808 | }
|
---|
809 |
|
---|
810 | // Only hierarchical URI's can have a path.
|
---|
811 | hierState = true;
|
---|
812 | }
|
---|
813 |
|
---|
814 |
|
---|
815 | /**
|
---|
816 | * The query (CGI) portion of the URI. This part is valid for
|
---|
817 | * both hierarchical and non-hierarchical URI's.
|
---|
818 | *
|
---|
819 | * <p>This accessor should only be used if a custom query syntax
|
---|
820 | * is used. This URI class supports the common "param=value"
|
---|
821 | * style query syntax via the get/setQueryValue() and
|
---|
822 | * get/setQueryByMap() functions. Those functions should be used
|
---|
823 | * instead if the common syntax is being used.
|
---|
824 | *
|
---|
825 | * <p>The URI RFC does not specify any particular
|
---|
826 | * syntax for the query part of a URI. It is intended to allow
|
---|
827 | * any format that can be agreed upon by the two communicating hosts.
|
---|
828 | * However, most systems have standardized on the typical CGI
|
---|
829 | * format:</p>
|
---|
830 | *
|
---|
831 | * <listing>http://site.com/script.php?param1=value1¶m2=value2</listing>
|
---|
832 | *
|
---|
833 | * <p>This class has specific support for this query syntax</p>
|
---|
834 | *
|
---|
835 | * <p>This common query format is an array of name/value
|
---|
836 | * pairs with its own syntax that is different from the overall URI
|
---|
837 | * syntax. The query has its own escaping logic. For a query part
|
---|
838 | * to be properly escaped and unescaped, it must be split into its
|
---|
839 | * component parts. This accessor escapes/unescapes the entire query
|
---|
840 | * part without regard for it's component parts. This has the
|
---|
841 | * possibliity of leaving the query string in an ambiguious state in
|
---|
842 | * regards to its syntax. If the contents of the query part are
|
---|
843 | * important, it is recommended that get/setQueryValue() or
|
---|
844 | * get/setQueryByMap() are used instead.</p>
|
---|
845 | *
|
---|
846 | * If a different query syntax is being used, a subclass of URI
|
---|
847 | * can be created to handle that specific syntax.
|
---|
848 | *
|
---|
849 | * @see URI.getQueryValue, URI.getQueryByMap
|
---|
850 | */
|
---|
851 | public function get query() : String
|
---|
852 | {
|
---|
853 | return URI.unescapeChars(_query);
|
---|
854 | }
|
---|
855 | public function set query(queryStr:String) : void
|
---|
856 | {
|
---|
857 | _query = URI.fastEscapeChars(queryStr, URI.URIqueryExcludedBitmap);
|
---|
858 |
|
---|
859 | // both hierarchical and non-hierarchical URI's can
|
---|
860 | // have a query. Do not set the hierState.
|
---|
861 | }
|
---|
862 |
|
---|
863 | /**
|
---|
864 | * Accessor to the raw query data. If you are using a custom query
|
---|
865 | * syntax, this accessor can be used to get and set the query part
|
---|
866 | * directly with no escaping/unescaping. This should ONLY be used
|
---|
867 | * if your application logic is handling custom query logic and
|
---|
868 | * handling the proper escaping of the query part.
|
---|
869 | */
|
---|
870 | public function get queryRaw() : String
|
---|
871 | {
|
---|
872 | return _query;
|
---|
873 | }
|
---|
874 | public function set queryRaw(queryStr:String) : void
|
---|
875 | {
|
---|
876 | _query = queryStr;
|
---|
877 | }
|
---|
878 |
|
---|
879 |
|
---|
880 | /**
|
---|
881 | * The fragment (anchor) portion of the URI. This is valid for
|
---|
882 | * both hierarchical and non-hierarchical URI's.
|
---|
883 | */
|
---|
884 | public function get fragment() : String
|
---|
885 | {
|
---|
886 | return URI.unescapeChars(_fragment);
|
---|
887 | }
|
---|
888 | public function set fragment(fragmentStr:String) : void
|
---|
889 | {
|
---|
890 | _fragment = URI.fastEscapeChars(fragmentStr, URIfragmentExcludedBitmap);
|
---|
891 |
|
---|
892 | // both hierarchical and non-hierarchical URI's can
|
---|
893 | // have a fragment. Do not set the hierState.
|
---|
894 | }
|
---|
895 |
|
---|
896 |
|
---|
897 | /**
|
---|
898 | * The non-hierarchical part of the URI. For example, if
|
---|
899 | * this URI object represents "mailto:somebody@company.com",
|
---|
900 | * this will contain "somebody@company.com". This is valid only
|
---|
901 | * for non-hierarchical URI's.
|
---|
902 | */
|
---|
903 | public function get nonHierarchical() : String
|
---|
904 | {
|
---|
905 | return URI.unescapeChars(_nonHierarchical);
|
---|
906 | }
|
---|
907 | public function set nonHierarchical(nonHier:String) : void
|
---|
908 | {
|
---|
909 | _nonHierarchical = URI.fastEscapeChars(nonHier, URInonHierexcludedBitmap);
|
---|
910 |
|
---|
911 | // This is a non-hierarchical URI.
|
---|
912 | this.hierState = false;
|
---|
913 | }
|
---|
914 |
|
---|
915 |
|
---|
916 | /**
|
---|
917 | * Quick shorthand accessor to set the parts of this URI.
|
---|
918 | * The given parts are assumed to be in unescaped form. If
|
---|
919 | * the URI is non-hierarchical (e.g. mailto:) you will need
|
---|
920 | * to call SetScheme() and SetNonHierarchical().
|
---|
921 | */
|
---|
922 | public function setParts(schemeStr:String, authorityStr:String,
|
---|
923 | portStr:String, pathStr:String, queryStr:String,
|
---|
924 | fragmentStr:String) : void
|
---|
925 | {
|
---|
926 | this.scheme = schemeStr;
|
---|
927 | this.authority = authorityStr;
|
---|
928 | this.port = portStr;
|
---|
929 | this.path = pathStr;
|
---|
930 | this.query = queryStr;
|
---|
931 | this.fragment = fragmentStr;
|
---|
932 |
|
---|
933 | hierState = true;
|
---|
934 | }
|
---|
935 |
|
---|
936 |
|
---|
937 | /**
|
---|
938 | * URI escapes the given character string. This is similar in function
|
---|
939 | * to the global encodeURIComponent() function in ActionScript, but is
|
---|
940 | * slightly different in regards to which characters get escaped. This
|
---|
941 | * escapes the characters specified in the URIbaselineExluded set (see class
|
---|
942 | * static members). This is needed for this class to work properly.
|
---|
943 | *
|
---|
944 | * <p>If a different set of characters need to be used for the escaping,
|
---|
945 | * you may use fastEscapeChars() and specify a custom URIEncodingBitmap
|
---|
946 | * that contains the characters your application needs escaped.</p>
|
---|
947 | *
|
---|
948 | * <p>Never pass a full URI to this function. A URI can only be properly
|
---|
949 | * escaped/unescaped when split into its component parts (see RFC 3986
|
---|
950 | * section 2.4). This is due to the fact that the URI component separators
|
---|
951 | * could be characters that would normally need to be escaped.</p>
|
---|
952 | *
|
---|
953 | * @param unescaped character string to be escaped.
|
---|
954 | *
|
---|
955 | * @return escaped character string
|
---|
956 | *
|
---|
957 | * @see encodeURIComponent
|
---|
958 | * @see fastEscapeChars
|
---|
959 | */
|
---|
960 | static public function escapeChars(unescaped:String) : String
|
---|
961 | {
|
---|
962 | // This uses the excluded set by default.
|
---|
963 | return fastEscapeChars(unescaped, URI.URIbaselineExcludedBitmap);
|
---|
964 | }
|
---|
965 |
|
---|
966 |
|
---|
967 | /**
|
---|
968 | * Unescape any URI escaped characters in the given character
|
---|
969 | * string.
|
---|
970 | *
|
---|
971 | * <p>Never pass a full URI to this function. A URI can only be properly
|
---|
972 | * escaped/unescaped when split into its component parts (see RFC 3986
|
---|
973 | * section 2.4). This is due to the fact that the URI component separators
|
---|
974 | * could be characters that would normally need to be escaped.</p>
|
---|
975 | *
|
---|
976 | * @param escaped the escaped string to be unescaped.
|
---|
977 | *
|
---|
978 | * @return unescaped string.
|
---|
979 | */
|
---|
980 | static public function unescapeChars(escaped:String /*, onlyHighASCII:Boolean = false*/) : String
|
---|
981 | {
|
---|
982 | // We can just use the default AS function. It seems to
|
---|
983 | // decode everything correctly
|
---|
984 | var unescaped:String;
|
---|
985 | unescaped = decodeURIComponent(escaped);
|
---|
986 | return unescaped;
|
---|
987 | }
|
---|
988 |
|
---|
989 | /**
|
---|
990 | * Performance focused function that escapes the given character
|
---|
991 | * string using the given URIEncodingBitmap as the rule for what
|
---|
992 | * characters need to be escaped. This function is used by this
|
---|
993 | * class and can be used externally to this class to perform
|
---|
994 | * escaping on custom character sets.
|
---|
995 | *
|
---|
996 | * <p>Never pass a full URI to this function. A URI can only be properly
|
---|
997 | * escaped/unescaped when split into its component parts (see RFC 3986
|
---|
998 | * section 2.4). This is due to the fact that the URI component separators
|
---|
999 | * could be characters that would normally need to be escaped.</p>
|
---|
1000 | *
|
---|
1001 | * @param unescaped the unescaped string to be escaped
|
---|
1002 | * @param bitmap the set of characters that need to be escaped
|
---|
1003 | *
|
---|
1004 | * @return the escaped string.
|
---|
1005 | */
|
---|
1006 | static public function fastEscapeChars(unescaped:String, bitmap:URIEncodingBitmap) : String
|
---|
1007 | {
|
---|
1008 | var escaped:String = "";
|
---|
1009 | var c:String;
|
---|
1010 | var x:int, i:int;
|
---|
1011 |
|
---|
1012 | for (i = 0; i < unescaped.length; i++)
|
---|
1013 | {
|
---|
1014 | c = unescaped.charAt(i);
|
---|
1015 |
|
---|
1016 | x = bitmap.ShouldEscape(c);
|
---|
1017 | if (x)
|
---|
1018 | {
|
---|
1019 | c = x.toString(16);
|
---|
1020 | if (c.length == 1)
|
---|
1021 | c = "0" + c;
|
---|
1022 |
|
---|
1023 | c = "%" + c;
|
---|
1024 | c = c.toUpperCase();
|
---|
1025 | }
|
---|
1026 |
|
---|
1027 | escaped += c;
|
---|
1028 | }
|
---|
1029 |
|
---|
1030 | return escaped;
|
---|
1031 | }
|
---|
1032 |
|
---|
1033 |
|
---|
1034 | /**
|
---|
1035 | * Is this URI of a particular scheme type? For example,
|
---|
1036 | * passing "http" to a URI object that represents the URI
|
---|
1037 | * "http://site.com/" would return true.
|
---|
1038 | *
|
---|
1039 | * @param scheme scheme to check for
|
---|
1040 | *
|
---|
1041 | * @return true if this URI object is of the given type, false
|
---|
1042 | * otherwise.
|
---|
1043 | */
|
---|
1044 | public function isOfType(scheme:String) : Boolean
|
---|
1045 | {
|
---|
1046 | // Schemes are never case sensitive. Ignore case.
|
---|
1047 | scheme = scheme.toLowerCase();
|
---|
1048 | return (this._scheme == scheme);
|
---|
1049 | }
|
---|
1050 |
|
---|
1051 |
|
---|
1052 | /**
|
---|
1053 | * Get the value for the specified named in the query part. This
|
---|
1054 | * assumes the query part of the URI is in the common
|
---|
1055 | * "name1=value1&name2=value2" syntax. Do not call this function
|
---|
1056 | * if you are using a custom query syntax.
|
---|
1057 | *
|
---|
1058 | * @param name name of the query value to get.
|
---|
1059 | *
|
---|
1060 | * @return the value of the query name, empty string if the
|
---|
1061 | * query name does not exist.
|
---|
1062 | */
|
---|
1063 | public function getQueryValue(name:String) : String
|
---|
1064 | {
|
---|
1065 | var map:Object;
|
---|
1066 | var item:String;
|
---|
1067 | var value:String;
|
---|
1068 |
|
---|
1069 | map = getQueryByMap();
|
---|
1070 |
|
---|
1071 | for (item in map)
|
---|
1072 | {
|
---|
1073 | if (item == name)
|
---|
1074 | {
|
---|
1075 | value = map[item];
|
---|
1076 | return value;
|
---|
1077 | }
|
---|
1078 | }
|
---|
1079 |
|
---|
1080 | // Didn't find the specified key
|
---|
1081 | return new String("");
|
---|
1082 | }
|
---|
1083 |
|
---|
1084 |
|
---|
1085 | /**
|
---|
1086 | * Set the given value on the given query name. If the given name
|
---|
1087 | * does not exist, it will automatically add this name/value pair
|
---|
1088 | * to the query. If null is passed as the value, it will remove
|
---|
1089 | * the given item from the query.
|
---|
1090 | *
|
---|
1091 | * <p>This automatically escapes any characters that may conflict with
|
---|
1092 | * the query syntax so that they are "safe" within the query. The
|
---|
1093 | * strings passed are assumed to be literal unescaped name and value.</p>
|
---|
1094 | *
|
---|
1095 | * @param name name of the query value to set
|
---|
1096 | * @param value value of the query item to set. If null, this will
|
---|
1097 | * force the removal of this item from the query.
|
---|
1098 | */
|
---|
1099 | public function setQueryValue(name:String, value:String) : void
|
---|
1100 | {
|
---|
1101 | var map:Object;
|
---|
1102 |
|
---|
1103 | map = getQueryByMap();
|
---|
1104 |
|
---|
1105 | // If the key doesn't exist yet, this will create a new pair in
|
---|
1106 | // the map. If it does exist, this will overwrite the previous
|
---|
1107 | // value, which is what we want.
|
---|
1108 | map[name] = value;
|
---|
1109 |
|
---|
1110 | setQueryByMap(map);
|
---|
1111 | }
|
---|
1112 |
|
---|
1113 |
|
---|
1114 | /**
|
---|
1115 | * Get the query of the URI in an Object class that allows for easy
|
---|
1116 | * access to the query data via Object accessors. For example:
|
---|
1117 | *
|
---|
1118 | * <listing>
|
---|
1119 | * var query:Object = uri.getQueryByMap();
|
---|
1120 | * var value:String = query["param"]; // get a value
|
---|
1121 | * query["param2"] = "foo"; // set a new value
|
---|
1122 | * </listing>
|
---|
1123 | *
|
---|
1124 | * @return Object that contains the name/value pairs of the query.
|
---|
1125 | *
|
---|
1126 | * @see #setQueryByMap
|
---|
1127 | * @see #getQueryValue
|
---|
1128 | * @see #setQueryValue
|
---|
1129 | */
|
---|
1130 | public function getQueryByMap() : Object
|
---|
1131 | {
|
---|
1132 | var queryStr:String;
|
---|
1133 | var pair:String;
|
---|
1134 | var pairs:Array;
|
---|
1135 | var item:Array;
|
---|
1136 | var name:String, value:String;
|
---|
1137 | var index:int;
|
---|
1138 | var map:Object = new Object();
|
---|
1139 |
|
---|
1140 |
|
---|
1141 | // We need the raw query string, no unescaping.
|
---|
1142 | queryStr = this._query;
|
---|
1143 |
|
---|
1144 | pairs = queryStr.split('&');
|
---|
1145 | for each (pair in pairs)
|
---|
1146 | {
|
---|
1147 | if (pair.length == 0)
|
---|
1148 | continue;
|
---|
1149 |
|
---|
1150 | item = pair.split('=');
|
---|
1151 |
|
---|
1152 | if (item.length > 0)
|
---|
1153 | name = item[0];
|
---|
1154 | else
|
---|
1155 | continue; // empty array
|
---|
1156 |
|
---|
1157 | if (item.length > 1)
|
---|
1158 | value = item[1];
|
---|
1159 | else
|
---|
1160 | value = "";
|
---|
1161 |
|
---|
1162 | name = queryPartUnescape(name);
|
---|
1163 | value = queryPartUnescape(value);
|
---|
1164 |
|
---|
1165 | map[name] = value;
|
---|
1166 | }
|
---|
1167 |
|
---|
1168 | return map;
|
---|
1169 | }
|
---|
1170 |
|
---|
1171 |
|
---|
1172 | /**
|
---|
1173 | * Set the query part of this URI using the given object as the
|
---|
1174 | * content source. Any member of the object that has a value of
|
---|
1175 | * null will not be in the resulting query.
|
---|
1176 | *
|
---|
1177 | * @param map object that contains the name/value pairs as
|
---|
1178 | * members of that object.
|
---|
1179 | *
|
---|
1180 | * @see #getQueryByMap
|
---|
1181 | * @see #getQueryValue
|
---|
1182 | * @see #setQueryValue
|
---|
1183 | */
|
---|
1184 | public function setQueryByMap(map:Object) : void
|
---|
1185 | {
|
---|
1186 | var item:String;
|
---|
1187 | var name:String, value:String;
|
---|
1188 | var queryStr:String = "";
|
---|
1189 | var tmpPair:String;
|
---|
1190 | var foo:String;
|
---|
1191 |
|
---|
1192 | for (item in map)
|
---|
1193 | {
|
---|
1194 | name = item;
|
---|
1195 | value = map[item];
|
---|
1196 |
|
---|
1197 | if (value == null)
|
---|
1198 | value = "";
|
---|
1199 |
|
---|
1200 | // Need to escape the name/value pair so that they
|
---|
1201 | // don't conflict with the query syntax (specifically
|
---|
1202 | // '=', '&', and <whitespace>).
|
---|
1203 | name = queryPartEscape(name);
|
---|
1204 | value = queryPartEscape(value);
|
---|
1205 |
|
---|
1206 | tmpPair = name;
|
---|
1207 |
|
---|
1208 | if (value.length > 0)
|
---|
1209 | {
|
---|
1210 | tmpPair += "=";
|
---|
1211 | tmpPair += value;
|
---|
1212 | }
|
---|
1213 |
|
---|
1214 | if (queryStr.length != 0)
|
---|
1215 | queryStr += '&'; // Add the separator
|
---|
1216 |
|
---|
1217 | queryStr += tmpPair;
|
---|
1218 | }
|
---|
1219 |
|
---|
1220 | // We don't want to escape. We already escaped the
|
---|
1221 | // individual name/value pairs. If we escaped the
|
---|
1222 | // query string again by assigning it to "query",
|
---|
1223 | // we would have double escaping.
|
---|
1224 | _query = queryStr;
|
---|
1225 | }
|
---|
1226 |
|
---|
1227 |
|
---|
1228 | /**
|
---|
1229 | * Similar to Escape(), except this also escapes characters that
|
---|
1230 | * would conflict with the name/value pair query syntax. This is
|
---|
1231 | * intended to be called on each individual "name" and "value"
|
---|
1232 | * in the query making sure that nothing in the name or value
|
---|
1233 | * strings contain characters that would conflict with the query
|
---|
1234 | * syntax (e.g. '=' and '&').
|
---|
1235 | *
|
---|
1236 | * @param unescaped unescaped string that is to be escaped.
|
---|
1237 | *
|
---|
1238 | * @return escaped string.
|
---|
1239 | *
|
---|
1240 | * @see #queryUnescape
|
---|
1241 | */
|
---|
1242 | static public function queryPartEscape(unescaped:String) : String
|
---|
1243 | {
|
---|
1244 | var escaped:String = unescaped;
|
---|
1245 | escaped = URI.fastEscapeChars(unescaped, URI.URIqueryPartExcludedBitmap);
|
---|
1246 | return escaped;
|
---|
1247 | }
|
---|
1248 |
|
---|
1249 |
|
---|
1250 | /**
|
---|
1251 | * Unescape the individual name/value string pairs.
|
---|
1252 | *
|
---|
1253 | * @param escaped escaped string to be unescaped
|
---|
1254 | *
|
---|
1255 | * @return unescaped string
|
---|
1256 | *
|
---|
1257 | * @see #queryEscape
|
---|
1258 | */
|
---|
1259 | static public function queryPartUnescape(escaped:String) : String
|
---|
1260 | {
|
---|
1261 | var unescaped:String = escaped;
|
---|
1262 | unescaped = unescapeChars(unescaped);
|
---|
1263 | return unescaped;
|
---|
1264 | }
|
---|
1265 |
|
---|
1266 | /**
|
---|
1267 | * Output this URI as a string. The resulting string is properly
|
---|
1268 | * escaped and well formed for machine processing.
|
---|
1269 | */
|
---|
1270 | public function toString() : String
|
---|
1271 | {
|
---|
1272 | if (this == null)
|
---|
1273 | return "";
|
---|
1274 | else
|
---|
1275 | return toStringInternal(false);
|
---|
1276 | }
|
---|
1277 |
|
---|
1278 | /**
|
---|
1279 | * Output the URI as a string that is easily readable by a human.
|
---|
1280 | * This outputs the URI with all escape sequences unescaped to
|
---|
1281 | * their character representation. This makes the URI easier for
|
---|
1282 | * a human to read, but the URI could be completely invalid
|
---|
1283 | * because some unescaped characters may now cause ambiguous parsing.
|
---|
1284 | * This function should only be used if you want to display a URI to
|
---|
1285 | * a user. This function should never be used outside that specific
|
---|
1286 | * case.
|
---|
1287 | *
|
---|
1288 | * @return the URI in string format with all escape sequences
|
---|
1289 | * unescaped.
|
---|
1290 | *
|
---|
1291 | * @see #toString
|
---|
1292 | */
|
---|
1293 | public function toDisplayString() : String
|
---|
1294 | {
|
---|
1295 | return toStringInternal(true);
|
---|
1296 | }
|
---|
1297 |
|
---|
1298 |
|
---|
1299 | /**
|
---|
1300 | * @private
|
---|
1301 | *
|
---|
1302 | * The guts of toString()
|
---|
1303 | */
|
---|
1304 | protected function toStringInternal(forDisplay:Boolean) : String
|
---|
1305 | {
|
---|
1306 | var uri:String = "";
|
---|
1307 | var part:String = "";
|
---|
1308 |
|
---|
1309 | if (isHierarchical() == false)
|
---|
1310 | {
|
---|
1311 | // non-hierarchical URI
|
---|
1312 |
|
---|
1313 | uri += (forDisplay ? this.scheme : _scheme);
|
---|
1314 | uri += ":";
|
---|
1315 | uri += (forDisplay ? this.nonHierarchical : _nonHierarchical);
|
---|
1316 | }
|
---|
1317 | else
|
---|
1318 | {
|
---|
1319 | // Hierarchical URI
|
---|
1320 |
|
---|
1321 | if (isRelative() == false)
|
---|
1322 | {
|
---|
1323 | // If it is not a relative URI, then we want the scheme and
|
---|
1324 | // authority parts in the string. If it is relative, we
|
---|
1325 | // do NOT want this stuff.
|
---|
1326 |
|
---|
1327 | if (_scheme.length != 0)
|
---|
1328 | {
|
---|
1329 | part = (forDisplay ? this.scheme : _scheme);
|
---|
1330 | uri += part + ":";
|
---|
1331 | }
|
---|
1332 |
|
---|
1333 | if (_authority.length != 0 || isOfType("file"))
|
---|
1334 | {
|
---|
1335 | uri += "//";
|
---|
1336 |
|
---|
1337 | // Add on any username/password associated with this
|
---|
1338 | // authority
|
---|
1339 | if (_username.length != 0)
|
---|
1340 | {
|
---|
1341 | part = (forDisplay ? this.username : _username);
|
---|
1342 | uri += part;
|
---|
1343 |
|
---|
1344 | if (_password.length != 0)
|
---|
1345 | {
|
---|
1346 | part = (forDisplay ? this.password : _password);
|
---|
1347 | uri += ":" + part;
|
---|
1348 | }
|
---|
1349 |
|
---|
1350 | uri += "@";
|
---|
1351 | }
|
---|
1352 |
|
---|
1353 | // add the authority
|
---|
1354 | part = (forDisplay ? this.authority : _authority);
|
---|
1355 | uri += part;
|
---|
1356 |
|
---|
1357 | // Tack on the port number, if any
|
---|
1358 | if (port.length != 0)
|
---|
1359 | uri += ":" + port;
|
---|
1360 | }
|
---|
1361 | }
|
---|
1362 |
|
---|
1363 | // Tack on the path
|
---|
1364 | part = (forDisplay ? this.path : _path);
|
---|
1365 | uri += part;
|
---|
1366 |
|
---|
1367 | } // end hierarchical part
|
---|
1368 |
|
---|
1369 | // Both non-hier and hierarchical have query and fragment parts
|
---|
1370 |
|
---|
1371 | // Add on the query and fragment parts
|
---|
1372 | if (_query.length != 0)
|
---|
1373 | {
|
---|
1374 | part = (forDisplay ? this.query : _query);
|
---|
1375 | uri += "?" + part;
|
---|
1376 | }
|
---|
1377 |
|
---|
1378 | if (fragment.length != 0)
|
---|
1379 | {
|
---|
1380 | part = (forDisplay ? this.fragment : _fragment);
|
---|
1381 | uri += "#" + part;
|
---|
1382 | }
|
---|
1383 |
|
---|
1384 | return uri;
|
---|
1385 | }
|
---|
1386 |
|
---|
1387 | /**
|
---|
1388 | * Forcefully ensure that this URI is properly escaped.
|
---|
1389 | *
|
---|
1390 | * <p>Sometimes URI's are constructed by hand using strings outside
|
---|
1391 | * this class. In those cases, it is unlikely the URI has been
|
---|
1392 | * properly escaped. This function forcefully escapes this URI
|
---|
1393 | * by unescaping each part and then re-escaping it. If the URI
|
---|
1394 | * did not have any escaping, the first unescape will do nothing
|
---|
1395 | * and then the re-escape will properly escape everything. If
|
---|
1396 | * the URI was already escaped, the unescape and re-escape will
|
---|
1397 | * essentally be a no-op. This provides a safe way to make sure
|
---|
1398 | * a URI is in the proper escaped form.</p>
|
---|
1399 | */
|
---|
1400 | public function forceEscape() : void
|
---|
1401 | {
|
---|
1402 | // The accessors for each of the members will unescape
|
---|
1403 | // and then re-escape as we get and assign them.
|
---|
1404 |
|
---|
1405 | // Handle the parts that are common for both hierarchical
|
---|
1406 | // and non-hierarchical URI's
|
---|
1407 | this.scheme = this.scheme;
|
---|
1408 | this.setQueryByMap(this.getQueryByMap());
|
---|
1409 | this.fragment = this.fragment;
|
---|
1410 |
|
---|
1411 | if (isHierarchical())
|
---|
1412 | {
|
---|
1413 | this.authority = this.authority;
|
---|
1414 | this.path = this.path;
|
---|
1415 | this.port = this.port;
|
---|
1416 | this.username = this.username;
|
---|
1417 | this.password = this.password;
|
---|
1418 | }
|
---|
1419 | else
|
---|
1420 | {
|
---|
1421 | this.nonHierarchical = this.nonHierarchical;
|
---|
1422 | }
|
---|
1423 | }
|
---|
1424 |
|
---|
1425 |
|
---|
1426 | /**
|
---|
1427 | * Does this URI point to a resource of the given file type?
|
---|
1428 | * Given a file extension (or just a file name, this will strip the
|
---|
1429 | * extension), check to see if this URI points to a file of that
|
---|
1430 | * type.
|
---|
1431 | *
|
---|
1432 | * @param extension string that contains a file extension with or
|
---|
1433 | * without a dot ("html" and ".html" are both valid), or a file
|
---|
1434 | * name with an extension (e.g. "index.html").
|
---|
1435 | *
|
---|
1436 | * @return true if this URI points to a resource with the same file
|
---|
1437 | * file extension as the extension provided, false otherwise.
|
---|
1438 | */
|
---|
1439 | public function isOfFileType(extension:String) : Boolean
|
---|
1440 | {
|
---|
1441 | var thisExtension:String;
|
---|
1442 | var index:int;
|
---|
1443 |
|
---|
1444 | index = extension.lastIndexOf(".");
|
---|
1445 | if (index != -1)
|
---|
1446 | {
|
---|
1447 | // Strip the extension
|
---|
1448 | extension = extension.substr(index + 1);
|
---|
1449 | }
|
---|
1450 | else
|
---|
1451 | {
|
---|
1452 | // The caller passed something without a dot in it. We
|
---|
1453 | // will assume that it is just a plain extension (e.g. "html").
|
---|
1454 | // What they passed is exactly what we want
|
---|
1455 | }
|
---|
1456 |
|
---|
1457 | thisExtension = getExtension(true);
|
---|
1458 |
|
---|
1459 | if (thisExtension == "")
|
---|
1460 | return false;
|
---|
1461 |
|
---|
1462 | // Compare the extensions ignoring case
|
---|
1463 | if (compareStr(thisExtension, extension, false) == 0)
|
---|
1464 | return true;
|
---|
1465 | else
|
---|
1466 | return false;
|
---|
1467 | }
|
---|
1468 |
|
---|
1469 |
|
---|
1470 | /**
|
---|
1471 | * Get the ".xyz" file extension from the filename in the URI.
|
---|
1472 | * For example, if we have the following URI:
|
---|
1473 | *
|
---|
1474 | * <listing>http://something.com/path/to/my/page.html?form=yes&name=bob#anchor</listing>
|
---|
1475 | *
|
---|
1476 | * <p>This will return ".html".</p>
|
---|
1477 | *
|
---|
1478 | * @param minusDot If true, this will strip the dot from the extension.
|
---|
1479 | * If true, the above example would have returned "html".
|
---|
1480 | *
|
---|
1481 | * @return the file extension
|
---|
1482 | */
|
---|
1483 | public function getExtension(minusDot:Boolean = false) : String
|
---|
1484 | {
|
---|
1485 | var filename:String = getFilename();
|
---|
1486 | var extension:String;
|
---|
1487 | var index:int;
|
---|
1488 |
|
---|
1489 | if (filename == "")
|
---|
1490 | return String("");
|
---|
1491 |
|
---|
1492 | index = filename.lastIndexOf(".");
|
---|
1493 |
|
---|
1494 | // If it doesn't have an extension, or if it is a "hidden" file,
|
---|
1495 | // it doesn't have an extension. Hidden files on unix start with
|
---|
1496 | // a dot (e.g. ".login").
|
---|
1497 | if (index == -1 || index == 0)
|
---|
1498 | return String("");
|
---|
1499 |
|
---|
1500 | extension = filename.substr(index);
|
---|
1501 |
|
---|
1502 | // If the caller does not want the dot, remove it.
|
---|
1503 | if (minusDot && extension.charAt(0) == ".")
|
---|
1504 | extension = extension.substr(1);
|
---|
1505 |
|
---|
1506 | return extension;
|
---|
1507 | }
|
---|
1508 |
|
---|
1509 | /**
|
---|
1510 | * Quick function to retrieve the file name off the end of a URI.
|
---|
1511 | *
|
---|
1512 | * <p>For example, if the URI is:</p>
|
---|
1513 | * <listing>http://something.com/some/path/to/my/file.html</listing>
|
---|
1514 | * <p>this function will return "file.html".</p>
|
---|
1515 | *
|
---|
1516 | * @param minusExtension true if the file extension should be stripped
|
---|
1517 | *
|
---|
1518 | * @return the file name. If this URI is a directory, the return
|
---|
1519 | * value will be empty string.
|
---|
1520 | */
|
---|
1521 | public function getFilename(minusExtension:Boolean = false) : String
|
---|
1522 | {
|
---|
1523 | if (isDirectory())
|
---|
1524 | return String("");
|
---|
1525 |
|
---|
1526 | var pathStr:String = this.path;
|
---|
1527 | var filename:String;
|
---|
1528 | var index:int;
|
---|
1529 |
|
---|
1530 | // Find the last path separator.
|
---|
1531 | index = pathStr.lastIndexOf("/");
|
---|
1532 |
|
---|
1533 | if (index != -1)
|
---|
1534 | filename = pathStr.substr(index + 1);
|
---|
1535 | else
|
---|
1536 | filename = pathStr;
|
---|
1537 |
|
---|
1538 | if (minusExtension)
|
---|
1539 | {
|
---|
1540 | // The caller has requested that the extension be removed
|
---|
1541 | index = filename.lastIndexOf(".");
|
---|
1542 |
|
---|
1543 | if (index != -1)
|
---|
1544 | filename = filename.substr(0, index);
|
---|
1545 | }
|
---|
1546 |
|
---|
1547 | return filename;
|
---|
1548 | }
|
---|
1549 |
|
---|
1550 |
|
---|
1551 | /**
|
---|
1552 | * @private
|
---|
1553 | * Helper function to compare strings.
|
---|
1554 | *
|
---|
1555 | * @return true if the two strings are identical, false otherwise.
|
---|
1556 | */
|
---|
1557 | static protected function compareStr(str1:String, str2:String,
|
---|
1558 | sensitive:Boolean = true) : Boolean
|
---|
1559 | {
|
---|
1560 | if (sensitive == false)
|
---|
1561 | {
|
---|
1562 | str1 = str1.toLowerCase();
|
---|
1563 | str2 = str2.toLowerCase();
|
---|
1564 | }
|
---|
1565 |
|
---|
1566 | return (str1 == str2)
|
---|
1567 | }
|
---|
1568 |
|
---|
1569 | /**
|
---|
1570 | * Based on the type of this URI (http, ftp, etc.) get
|
---|
1571 | * the default port used for that protocol. This is
|
---|
1572 | * just intended to be a helper function for the most
|
---|
1573 | * common cases.
|
---|
1574 | */
|
---|
1575 | public function getDefaultPort() : String
|
---|
1576 | {
|
---|
1577 | if (_scheme == "http")
|
---|
1578 | return String("80");
|
---|
1579 | else if (_scheme == "ftp")
|
---|
1580 | return String("21");
|
---|
1581 | else if (_scheme == "file")
|
---|
1582 | return String("");
|
---|
1583 | else if (_scheme == "sftp")
|
---|
1584 | return String("22"); // ssh standard port
|
---|
1585 | else
|
---|
1586 | {
|
---|
1587 | // Don't know the port for this URI type
|
---|
1588 | return String("");
|
---|
1589 | }
|
---|
1590 | }
|
---|
1591 |
|
---|
1592 | /**
|
---|
1593 | * @private
|
---|
1594 | *
|
---|
1595 | * This resolves the given URI if the application has a
|
---|
1596 | * resolver interface defined. This function does not
|
---|
1597 | * modify the passed in URI and returns a new URI.
|
---|
1598 | */
|
---|
1599 | static protected function resolve(uri:URI) : URI
|
---|
1600 | {
|
---|
1601 | var copy:URI = new URI();
|
---|
1602 | copy.copyURI(uri);
|
---|
1603 |
|
---|
1604 | if (_resolver != null)
|
---|
1605 | {
|
---|
1606 | // A resolver class has been registered. Call it.
|
---|
1607 | return _resolver.resolve(copy);
|
---|
1608 | }
|
---|
1609 | else
|
---|
1610 | {
|
---|
1611 | // No resolver. Nothing to do, but we don't
|
---|
1612 | // want to reuse the one passed in.
|
---|
1613 | return copy;
|
---|
1614 | }
|
---|
1615 | }
|
---|
1616 |
|
---|
1617 | /**
|
---|
1618 | * Accessor to set and get the resolver object used by all URI
|
---|
1619 | * objects to dynamically resolve URI's before comparison.
|
---|
1620 | */
|
---|
1621 | static public function set resolver(resolver:IURIResolver) : void
|
---|
1622 | {
|
---|
1623 | _resolver = resolver;
|
---|
1624 | }
|
---|
1625 | static public function get resolver() : IURIResolver
|
---|
1626 | {
|
---|
1627 | return _resolver;
|
---|
1628 | }
|
---|
1629 |
|
---|
1630 | /**
|
---|
1631 | * Given another URI, return this URI object's relation to the one given.
|
---|
1632 | * URI's can have 1 of 4 possible relationships. They can be unrelated,
|
---|
1633 | * equal, parent, or a child of the given URI.
|
---|
1634 | *
|
---|
1635 | * @param uri URI to compare this URI object to.
|
---|
1636 | * @param caseSensitive true if the URI comparison should be done
|
---|
1637 | * taking case into account, false if the comparison should be
|
---|
1638 | * performed case insensitive.
|
---|
1639 | *
|
---|
1640 | * @return URI.NOT_RELATED, URI.CHILD, URI.PARENT, or URI.EQUAL
|
---|
1641 | */
|
---|
1642 | public function getRelation(uri:URI, caseSensitive:Boolean = true) : int
|
---|
1643 | {
|
---|
1644 | // Give the app a chance to resolve these URI's before we compare them.
|
---|
1645 | var thisURI:URI = URI.resolve(this);
|
---|
1646 | var thatURI:URI = URI.resolve(uri);
|
---|
1647 |
|
---|
1648 | if (thisURI.isRelative() || thatURI.isRelative())
|
---|
1649 | {
|
---|
1650 | // You cannot compare relative URI's due to their lack of context.
|
---|
1651 | // You could have two relative URI's that look like:
|
---|
1652 | // ../../images/
|
---|
1653 | // ../../images/marketing/logo.gif
|
---|
1654 | // These may appear related, but you have no overall context
|
---|
1655 | // from which to make the comparison. The first URI could be
|
---|
1656 | // from one site and the other URI could be from another site.
|
---|
1657 | return URI.NOT_RELATED;
|
---|
1658 | }
|
---|
1659 | else if (thisURI.isHierarchical() == false || thatURI.isHierarchical() == false)
|
---|
1660 | {
|
---|
1661 | // One or both of the URI's are non-hierarchical.
|
---|
1662 | if (((thisURI.isHierarchical() == false) && (thatURI.isHierarchical() == true)) ||
|
---|
1663 | ((thisURI.isHierarchical() == true) && (thatURI.isHierarchical() == false)))
|
---|
1664 | {
|
---|
1665 | // XOR. One is hierarchical and the other is
|
---|
1666 | // non-hierarchical. They cannot be compared.
|
---|
1667 | return URI.NOT_RELATED;
|
---|
1668 | }
|
---|
1669 | else
|
---|
1670 | {
|
---|
1671 | // They are both non-hierarchical
|
---|
1672 | if (thisURI.scheme != thatURI.scheme)
|
---|
1673 | return URI.NOT_RELATED;
|
---|
1674 |
|
---|
1675 | if (thisURI.nonHierarchical != thatURI.nonHierarchical)
|
---|
1676 | return URI.NOT_RELATED;
|
---|
1677 |
|
---|
1678 | // The two non-hierarcical URI's are equal.
|
---|
1679 | return URI.EQUAL;
|
---|
1680 | }
|
---|
1681 | }
|
---|
1682 |
|
---|
1683 | // Ok, this URI and the one we are being compared to are both
|
---|
1684 | // absolute hierarchical URI's.
|
---|
1685 |
|
---|
1686 | if (thisURI.scheme != thatURI.scheme)
|
---|
1687 | return URI.NOT_RELATED;
|
---|
1688 |
|
---|
1689 | if (thisURI.authority != thatURI.authority)
|
---|
1690 | return URI.NOT_RELATED;
|
---|
1691 |
|
---|
1692 | var thisPort:String = thisURI.port;
|
---|
1693 | var thatPort:String = thatURI.port;
|
---|
1694 |
|
---|
1695 | // Different ports are considered completely different servers.
|
---|
1696 | if (thisPort == "")
|
---|
1697 | thisPort = thisURI.getDefaultPort();
|
---|
1698 | if (thatPort == "")
|
---|
1699 | thatPort = thatURI.getDefaultPort();
|
---|
1700 |
|
---|
1701 | // Check to see if the port is the default port.
|
---|
1702 | if (thisPort != thatPort)
|
---|
1703 | return URI.NOT_RELATED;
|
---|
1704 |
|
---|
1705 | if (compareStr(thisURI.path, thatURI.path, caseSensitive))
|
---|
1706 | return URI.EQUAL;
|
---|
1707 |
|
---|
1708 | // Special case check. If we are here, the scheme, authority,
|
---|
1709 | // and port match, and it is not a relative path, but the
|
---|
1710 | // paths did not match. There is a special case where we
|
---|
1711 | // could have:
|
---|
1712 | // http://something.com/
|
---|
1713 | // http://something.com
|
---|
1714 | // Technically, these are equal. So lets, check for this case.
|
---|
1715 | var thisPath:String = thisURI.path;
|
---|
1716 | var thatPath:String = thatURI.path;
|
---|
1717 |
|
---|
1718 | if ( (thisPath == "/" || thatPath == "/") &&
|
---|
1719 | (thisPath == "" || thatPath == "") )
|
---|
1720 | {
|
---|
1721 | // We hit the special case. These two are equal.
|
---|
1722 | return URI.EQUAL;
|
---|
1723 | }
|
---|
1724 |
|
---|
1725 | // Ok, the paths do not match, but one path may be a parent/child
|
---|
1726 | // of the other. For example, we may have:
|
---|
1727 | // http://something.com/path/to/homepage/
|
---|
1728 | // http://something.com/path/to/homepage/images/logo.gif
|
---|
1729 | // In this case, the first is a parent of the second (or the second
|
---|
1730 | // is a child of the first, depending on which you compare to the
|
---|
1731 | // other). To make this comparison, we must split the path into
|
---|
1732 | // its component parts (split the string on the '/' path delimiter).
|
---|
1733 | // We then compare the
|
---|
1734 | var thisParts:Array, thatParts:Array;
|
---|
1735 | var thisPart:String, thatPart:String;
|
---|
1736 | var i:int;
|
---|
1737 |
|
---|
1738 | thisParts = thisPath.split("/");
|
---|
1739 | thatParts = thatPath.split("/");
|
---|
1740 |
|
---|
1741 | if (thisParts.length > thatParts.length)
|
---|
1742 | {
|
---|
1743 | thatPart = thatParts[thatParts.length - 1];
|
---|
1744 | if (thatPart.length > 0)
|
---|
1745 | {
|
---|
1746 | // if the last part is not empty, the passed URI is
|
---|
1747 | // not a directory. There is no way the passed URI
|
---|
1748 | // can be a parent.
|
---|
1749 | return URI.NOT_RELATED;
|
---|
1750 | }
|
---|
1751 | else
|
---|
1752 | {
|
---|
1753 | // Remove the empty trailing part
|
---|
1754 | thatParts.pop();
|
---|
1755 | }
|
---|
1756 |
|
---|
1757 | // This may be a child of the one passed in
|
---|
1758 | for (i = 0; i < thatParts.length; i++)
|
---|
1759 | {
|
---|
1760 | thisPart = thisParts[i];
|
---|
1761 | thatPart = thatParts[i];
|
---|
1762 |
|
---|
1763 | if (compareStr(thisPart, thatPart, caseSensitive) == false)
|
---|
1764 | return URI.NOT_RELATED;
|
---|
1765 | }
|
---|
1766 |
|
---|
1767 | return URI.CHILD;
|
---|
1768 | }
|
---|
1769 | else if (thisParts.length < thatParts.length)
|
---|
1770 | {
|
---|
1771 | thisPart = thisParts[thisParts.length - 1];
|
---|
1772 | if (thisPart.length > 0)
|
---|
1773 | {
|
---|
1774 | // if the last part is not empty, this URI is not a
|
---|
1775 | // directory. There is no way this object can be
|
---|
1776 | // a parent.
|
---|
1777 | return URI.NOT_RELATED;
|
---|
1778 | }
|
---|
1779 | else
|
---|
1780 | {
|
---|
1781 | // Remove the empty trailing part
|
---|
1782 | thisParts.pop();
|
---|
1783 | }
|
---|
1784 |
|
---|
1785 | // This may be the parent of the one passed in
|
---|
1786 | for (i = 0; i < thisParts.length; i++)
|
---|
1787 | {
|
---|
1788 | thisPart = thisParts[i];
|
---|
1789 | thatPart = thatParts[i];
|
---|
1790 |
|
---|
1791 | if (compareStr(thisPart, thatPart, caseSensitive) == false)
|
---|
1792 | return URI.NOT_RELATED;
|
---|
1793 | }
|
---|
1794 |
|
---|
1795 | return URI.PARENT;
|
---|
1796 | }
|
---|
1797 | else
|
---|
1798 | {
|
---|
1799 | // Both URI's have the same number of path components, but
|
---|
1800 | // it failed the equivelence check above. This means that
|
---|
1801 | // the two URI's are not related.
|
---|
1802 | return URI.NOT_RELATED;
|
---|
1803 | }
|
---|
1804 |
|
---|
1805 | // If we got here, the scheme and authority are the same,
|
---|
1806 | // but the paths pointed to two different locations that
|
---|
1807 | // were in different parts of the file system tree
|
---|
1808 | return URI.NOT_RELATED;
|
---|
1809 | }
|
---|
1810 |
|
---|
1811 | /**
|
---|
1812 | * Given another URI, return the common parent between this one
|
---|
1813 | * and the provided URI.
|
---|
1814 | *
|
---|
1815 | * @param uri the other URI from which to find a common parent
|
---|
1816 | * @para caseSensitive true if this operation should be done
|
---|
1817 | * with case sensitive comparisons.
|
---|
1818 | *
|
---|
1819 | * @return the parent URI if successful, null otherwise.
|
---|
1820 | */
|
---|
1821 | public function getCommonParent(uri:URI, caseSensitive:Boolean = true) : URI
|
---|
1822 | {
|
---|
1823 | var thisURI:URI = URI.resolve(this);
|
---|
1824 | var thatURI:URI = URI.resolve(uri);
|
---|
1825 |
|
---|
1826 | if(!thisURI.isAbsolute() || !thatURI.isAbsolute() ||
|
---|
1827 | thisURI.isHierarchical() == false ||
|
---|
1828 | thatURI.isHierarchical() == false)
|
---|
1829 | {
|
---|
1830 | // Both URI's must be absolute hierarchical for this to
|
---|
1831 | // make sense.
|
---|
1832 | return null;
|
---|
1833 | }
|
---|
1834 |
|
---|
1835 | var relation:int = thisURI.getRelation(thatURI);
|
---|
1836 | if (relation == URI.NOT_RELATED)
|
---|
1837 | {
|
---|
1838 | // The given URI is not related to this one. No
|
---|
1839 | // common parent.
|
---|
1840 | return null;
|
---|
1841 | }
|
---|
1842 |
|
---|
1843 | thisURI.chdir(".");
|
---|
1844 | thatURI.chdir(".");
|
---|
1845 |
|
---|
1846 | var strBefore:String, strAfter:String;
|
---|
1847 | do
|
---|
1848 | {
|
---|
1849 | relation = thisURI.getRelation(thatURI, caseSensitive);
|
---|
1850 | if(relation == URI.EQUAL || relation == URI.PARENT)
|
---|
1851 | break;
|
---|
1852 |
|
---|
1853 | // If strBefore and strAfter end up being the same,
|
---|
1854 | // we know we are at the root of the path because
|
---|
1855 | // chdir("..") is doing nothing.
|
---|
1856 | strBefore = thisURI.toString();
|
---|
1857 | thisURI.chdir("..");
|
---|
1858 | strAfter = thisURI.toString();
|
---|
1859 | }
|
---|
1860 | while(strBefore != strAfter);
|
---|
1861 |
|
---|
1862 | return thisURI;
|
---|
1863 | }
|
---|
1864 |
|
---|
1865 |
|
---|
1866 | /**
|
---|
1867 | * This function is used to move around in a URI in a way similar
|
---|
1868 | * to the 'cd' or 'chdir' commands on Unix. These operations are
|
---|
1869 | * completely string based, using the context of the URI to
|
---|
1870 | * determine the position within the path. The heuristics used
|
---|
1871 | * to determine the action are based off Appendix C in RFC 2396.
|
---|
1872 | *
|
---|
1873 | * <p>URI paths that end in '/' are considered paths that point to
|
---|
1874 | * directories, while paths that do not end in '/' are files. For
|
---|
1875 | * example, if you execute chdir("d") on the following URI's:<br/>
|
---|
1876 | * 1. http://something.com/a/b/c/ (directory)<br/>
|
---|
1877 | * 2. http://something.com/a/b/c (not directory)<br/>
|
---|
1878 | * you will get:<br/>
|
---|
1879 | * 1. http://something.com/a/b/c/d<br/>
|
---|
1880 | * 2. http://something.com/a/b/d<br/></p>
|
---|
1881 | *
|
---|
1882 | * <p>See RFC 2396, Appendix C for more info.</p>
|
---|
1883 | *
|
---|
1884 | * @param reference the URI or path to "cd" to.
|
---|
1885 | * @param escape true if the passed reference string should be URI
|
---|
1886 | * escaped before using it.
|
---|
1887 | *
|
---|
1888 | * @return true if the chdir was successful, false otherwise.
|
---|
1889 | */
|
---|
1890 | public function chdir(reference:String, escape:Boolean = false) : Boolean
|
---|
1891 | {
|
---|
1892 | var uriReference:URI;
|
---|
1893 | var ref:String = reference;
|
---|
1894 |
|
---|
1895 | if (escape)
|
---|
1896 | ref = URI.escapeChars(reference);
|
---|
1897 |
|
---|
1898 | if (ref == "")
|
---|
1899 | {
|
---|
1900 | // NOOP
|
---|
1901 | return true;
|
---|
1902 | }
|
---|
1903 | else if (ref.substr(0, 2) == "//")
|
---|
1904 | {
|
---|
1905 | // Special case. This is an absolute URI but without the scheme.
|
---|
1906 | // Take the scheme from this URI and tack it on. This is
|
---|
1907 | // intended to make working with chdir() a little more
|
---|
1908 | // tolerant.
|
---|
1909 | var final:String = this.scheme + ":" + ref;
|
---|
1910 |
|
---|
1911 | return constructURI(final);
|
---|
1912 | }
|
---|
1913 | else if (ref.charAt(0) == "?")
|
---|
1914 | {
|
---|
1915 | // A relative URI that is just a query part is essentially
|
---|
1916 | // a "./?query". We tack on the "./" here to make the rest
|
---|
1917 | // of our logic work.
|
---|
1918 | ref = "./" + ref;
|
---|
1919 | }
|
---|
1920 |
|
---|
1921 | // Parse the reference passed in as a URI. This way we
|
---|
1922 | // get any query and fragments parsed out as well.
|
---|
1923 | uriReference = new URI(ref);
|
---|
1924 |
|
---|
1925 | if (uriReference.isAbsolute() ||
|
---|
1926 | uriReference.isHierarchical() == false)
|
---|
1927 | {
|
---|
1928 | // If the URI given is a full URI, it replaces this one.
|
---|
1929 | copyURI(uriReference);
|
---|
1930 | return true;
|
---|
1931 | }
|
---|
1932 |
|
---|
1933 |
|
---|
1934 | var thisPath:String, thatPath:String;
|
---|
1935 | var thisParts:Array, thatParts:Array;
|
---|
1936 | var thisIsDir:Boolean = false, thatIsDir:Boolean = false;
|
---|
1937 | var thisIsAbs:Boolean = false, thatIsAbs:Boolean = false;
|
---|
1938 | var lastIsDotOperation:Boolean = false;
|
---|
1939 | var curDir:String;
|
---|
1940 | var i:int;
|
---|
1941 |
|
---|
1942 | thisPath = this.path;
|
---|
1943 | thatPath = uriReference.path;
|
---|
1944 |
|
---|
1945 | if (thisPath.length > 0)
|
---|
1946 | thisParts = thisPath.split("/");
|
---|
1947 | else
|
---|
1948 | thisParts = new Array();
|
---|
1949 |
|
---|
1950 | if (thatPath.length > 0)
|
---|
1951 | thatParts = thatPath.split("/");
|
---|
1952 | else
|
---|
1953 | thatParts = new Array();
|
---|
1954 |
|
---|
1955 | if (thisParts.length > 0 && thisParts[0] == "")
|
---|
1956 | {
|
---|
1957 | thisIsAbs = true;
|
---|
1958 | thisParts.shift(); // pop the first one off the array
|
---|
1959 | }
|
---|
1960 | if (thisParts.length > 0 && thisParts[thisParts.length - 1] == "")
|
---|
1961 | {
|
---|
1962 | thisIsDir = true;
|
---|
1963 | thisParts.pop(); // pop the last one off the array
|
---|
1964 | }
|
---|
1965 |
|
---|
1966 | if (thatParts.length > 0 && thatParts[0] == "")
|
---|
1967 | {
|
---|
1968 | thatIsAbs = true;
|
---|
1969 | thatParts.shift(); // pop the first one off the array
|
---|
1970 | }
|
---|
1971 | if (thatParts.length > 0 && thatParts[thatParts.length - 1] == "")
|
---|
1972 | {
|
---|
1973 | thatIsDir = true;
|
---|
1974 | thatParts.pop(); // pop the last one off the array
|
---|
1975 | }
|
---|
1976 |
|
---|
1977 | if (thatIsAbs)
|
---|
1978 | {
|
---|
1979 | // The reference is an absolute path (starts with a slash).
|
---|
1980 | // It replaces this path wholesale.
|
---|
1981 | this.path = uriReference.path;
|
---|
1982 |
|
---|
1983 | // And it inherits the query and fragment
|
---|
1984 | this.queryRaw = uriReference.queryRaw;
|
---|
1985 | this.fragment = uriReference.fragment;
|
---|
1986 |
|
---|
1987 | return true;
|
---|
1988 | }
|
---|
1989 | else if (thatParts.length == 0 && uriReference.query == "")
|
---|
1990 | {
|
---|
1991 | // The reference must have only been a fragment. Fragments just
|
---|
1992 | // get appended to whatever the current path is. We don't want
|
---|
1993 | // to overwrite any query that may already exist, so this case
|
---|
1994 | // only takes on the new fragment.
|
---|
1995 | this.fragment = uriReference.fragment;
|
---|
1996 | return true;
|
---|
1997 | }
|
---|
1998 | else if (thisIsDir == false && thisParts.length > 0)
|
---|
1999 | {
|
---|
2000 | // This path ends in a file. It goes away no matter what.
|
---|
2001 | thisParts.pop();
|
---|
2002 | }
|
---|
2003 |
|
---|
2004 | // By default, this assumes the query and fragment of the reference
|
---|
2005 | this.queryRaw = uriReference.queryRaw;
|
---|
2006 | this.fragment = uriReference.fragment;
|
---|
2007 |
|
---|
2008 | // Append the parts of the path from the passed in reference
|
---|
2009 | // to this object's path.
|
---|
2010 | thisParts = thisParts.concat(thatParts);
|
---|
2011 |
|
---|
2012 | for(i = 0; i < thisParts.length; i++)
|
---|
2013 | {
|
---|
2014 | curDir = thisParts[i];
|
---|
2015 | lastIsDotOperation = false;
|
---|
2016 |
|
---|
2017 | if (curDir == ".")
|
---|
2018 | {
|
---|
2019 | thisParts.splice(i, 1);
|
---|
2020 | i = i - 1; // account for removing this item
|
---|
2021 | lastIsDotOperation = true;
|
---|
2022 | }
|
---|
2023 | else if (curDir == "..")
|
---|
2024 | {
|
---|
2025 | if (i >= 1)
|
---|
2026 | {
|
---|
2027 | if (thisParts[i - 1] == "..")
|
---|
2028 | {
|
---|
2029 | // If the previous is a "..", we must have skipped
|
---|
2030 | // it due to this URI being relative. We can't
|
---|
2031 | // collapse leading ".."s in a relative URI, so
|
---|
2032 | // do nothing.
|
---|
2033 | }
|
---|
2034 | else
|
---|
2035 | {
|
---|
2036 | thisParts.splice(i - 1, 2);
|
---|
2037 | i = i - 2; // move back to account for the 2 we removed
|
---|
2038 | }
|
---|
2039 | }
|
---|
2040 | else
|
---|
2041 | {
|
---|
2042 | // This is the first thing in the path.
|
---|
2043 |
|
---|
2044 | if (isRelative())
|
---|
2045 | {
|
---|
2046 | // We can't collapse leading ".."s in a relative
|
---|
2047 | // path. Do noting.
|
---|
2048 | }
|
---|
2049 | else
|
---|
2050 | {
|
---|
2051 | // This is an abnormal case. We have dot-dotted up
|
---|
2052 | // past the base of our "file system". This is a
|
---|
2053 | // case where we had a /path/like/this.htm and were
|
---|
2054 | // given a path to chdir to like this:
|
---|
2055 | // ../../../../../../mydir
|
---|
2056 | // Obviously, it has too many ".." and will take us
|
---|
2057 | // up beyond the top of the URI. However, according
|
---|
2058 | // RFC 2396 Appendix C.2, we should try to handle
|
---|
2059 | // these abnormal cases appropriately. In this case,
|
---|
2060 | // we will do what UNIX command lines do if you are
|
---|
2061 | // at the root (/) of the filesystem and execute:
|
---|
2062 | // # cd ../../../../../bin
|
---|
2063 | // Which will put you in /bin. Essentially, the extra
|
---|
2064 | // ".."'s will just get eaten.
|
---|
2065 |
|
---|
2066 | thisParts.splice(i, 1);
|
---|
2067 | i = i - 1; // account for the ".." we just removed
|
---|
2068 | }
|
---|
2069 | }
|
---|
2070 |
|
---|
2071 | lastIsDotOperation = true;
|
---|
2072 | }
|
---|
2073 | }
|
---|
2074 |
|
---|
2075 | var finalPath:String = "";
|
---|
2076 |
|
---|
2077 | // If the last thing in the path was a "." or "..", then this thing is a
|
---|
2078 | // directory. If the last thing isn't a dot-op, then we don't want to
|
---|
2079 | // blow away any information about the directory (hence the "|=" binary
|
---|
2080 | // assignment).
|
---|
2081 | thatIsDir = thatIsDir || lastIsDotOperation;
|
---|
2082 |
|
---|
2083 | // Reconstruct the path with the abs/dir info we have
|
---|
2084 | finalPath = joinPath(thisParts, thisIsAbs, thatIsDir);
|
---|
2085 |
|
---|
2086 | // Set the path (automatically escaping it)
|
---|
2087 | this.path = finalPath;
|
---|
2088 |
|
---|
2089 | return true;
|
---|
2090 | }
|
---|
2091 |
|
---|
2092 | /**
|
---|
2093 | * @private
|
---|
2094 | * Join an array of path parts back into a URI style path string.
|
---|
2095 | * This is used by the various path logic functions to recombine
|
---|
2096 | * a path. This is different than the standard Array.join()
|
---|
2097 | * function because we need to take into account the starting and
|
---|
2098 | * ending path delimiters if this is an absolute path or a
|
---|
2099 | * directory.
|
---|
2100 | *
|
---|
2101 | * @param parts the Array that contains strings of each path part.
|
---|
2102 | * @param isAbs true if the given path is absolute
|
---|
2103 | * @param isDir true if the given path is a directory
|
---|
2104 | *
|
---|
2105 | * @return the combined path string.
|
---|
2106 | */
|
---|
2107 | protected function joinPath(parts:Array, isAbs:Boolean, isDir:Boolean) : String
|
---|
2108 | {
|
---|
2109 | var pathStr:String = "";
|
---|
2110 | var i:int;
|
---|
2111 |
|
---|
2112 | for (i = 0; i < parts.length; i++)
|
---|
2113 | {
|
---|
2114 | if (pathStr.length > 0)
|
---|
2115 | pathStr += "/";
|
---|
2116 |
|
---|
2117 | pathStr += parts[i];
|
---|
2118 | }
|
---|
2119 |
|
---|
2120 | // If this path is a directory, tack on the directory delimiter,
|
---|
2121 | // but only if the path contains something. Adding this to an
|
---|
2122 | // empty path would make it "/", which is an absolute path that
|
---|
2123 | // starts at the root.
|
---|
2124 | if (isDir && pathStr.length > 0)
|
---|
2125 | pathStr += "/";
|
---|
2126 |
|
---|
2127 | if (isAbs)
|
---|
2128 | pathStr = "/" + pathStr;
|
---|
2129 |
|
---|
2130 | return pathStr;
|
---|
2131 | }
|
---|
2132 |
|
---|
2133 | /**
|
---|
2134 | * Given an absolute URI, make this relative URI absolute using
|
---|
2135 | * the given URI as a base. This URI instance must be relative
|
---|
2136 | * and the base_uri must be absolute.
|
---|
2137 | *
|
---|
2138 | * @param base_uri URI to use as the base from which to make
|
---|
2139 | * this relative URI into an absolute URI.
|
---|
2140 | *
|
---|
2141 | * @return true if successful, false otherwise.
|
---|
2142 | */
|
---|
2143 | public function makeAbsoluteURI(base_uri:URI) : Boolean
|
---|
2144 | {
|
---|
2145 | if (isAbsolute() || base_uri.isRelative())
|
---|
2146 | {
|
---|
2147 | // This URI needs to be relative, and the base needs to be
|
---|
2148 | // absolute otherwise we won't know what to do!
|
---|
2149 | return false;
|
---|
2150 | }
|
---|
2151 |
|
---|
2152 | // Make a copy of the base URI. We don't want to modify
|
---|
2153 | // the passed URI.
|
---|
2154 | var base:URI = new URI();
|
---|
2155 | base.copyURI(base_uri);
|
---|
2156 |
|
---|
2157 | // ChDir on the base URI. This will preserve any query
|
---|
2158 | // and fragment we have.
|
---|
2159 | if (base.chdir(toString()) == false)
|
---|
2160 | return false;
|
---|
2161 |
|
---|
2162 | // It worked, so copy the base into this one
|
---|
2163 | copyURI(base);
|
---|
2164 |
|
---|
2165 | return true;
|
---|
2166 | }
|
---|
2167 |
|
---|
2168 |
|
---|
2169 | /**
|
---|
2170 | * Given a URI to use as a base from which this object should be
|
---|
2171 | * relative to, convert this object into a relative URI. For example,
|
---|
2172 | * if you have:
|
---|
2173 | *
|
---|
2174 | * <listing>
|
---|
2175 | * var uri1:URI = new URI("http://something.com/path/to/some/file.html");
|
---|
2176 | * var uri2:URI = new URI("http://something.com/path/to/another/file.html");
|
---|
2177 | *
|
---|
2178 | * uri1.MakeRelativePath(uri2);</listing>
|
---|
2179 | *
|
---|
2180 | * <p>uri1 will have a final value of "../some/file.html"</p>
|
---|
2181 | *
|
---|
2182 | * <p>Note! This function is brute force. If you have two URI's
|
---|
2183 | * that are completely unrelated, this will still attempt to make
|
---|
2184 | * the relative URI. In that case, you will most likely get a
|
---|
2185 | * relative path that looks something like:</p>
|
---|
2186 | *
|
---|
2187 | * <p>../../../../../../some/path/to/my/file.html</p>
|
---|
2188 | *
|
---|
2189 | * @param base_uri the URI from which to make this URI relative
|
---|
2190 | *
|
---|
2191 | * @return true if successful, false if the base_uri and this URI
|
---|
2192 | * are not related, of if error.
|
---|
2193 | */
|
---|
2194 | public function makeRelativeURI(base_uri:URI, caseSensitive:Boolean = true) : Boolean
|
---|
2195 | {
|
---|
2196 | var base:URI = new URI();
|
---|
2197 | base.copyURI(base_uri);
|
---|
2198 |
|
---|
2199 | var thisParts:Array, thatParts:Array;
|
---|
2200 | var finalParts:Array = new Array();
|
---|
2201 | var thisPart:String, thatPart:String, finalPath:String;
|
---|
2202 | var pathStr:String = this.path;
|
---|
2203 | var queryStr:String = this.queryRaw;
|
---|
2204 | var fragmentStr:String = this.fragment;
|
---|
2205 | var i:int;
|
---|
2206 | var diff:Boolean = false;
|
---|
2207 | var isDir:Boolean = false;
|
---|
2208 |
|
---|
2209 | if (isRelative())
|
---|
2210 | {
|
---|
2211 | // We're already relative.
|
---|
2212 | return true;
|
---|
2213 | }
|
---|
2214 |
|
---|
2215 | if (base.isRelative())
|
---|
2216 | {
|
---|
2217 | // The base is relative. A relative base doesn't make sense.
|
---|
2218 | return false;
|
---|
2219 | }
|
---|
2220 |
|
---|
2221 |
|
---|
2222 | if ( (isOfType(base_uri.scheme) == false) ||
|
---|
2223 | (this.authority != base_uri.authority) )
|
---|
2224 | {
|
---|
2225 | // The schemes and/or authorities are different. We can't
|
---|
2226 | // make a relative path to something that is completely
|
---|
2227 | // unrelated.
|
---|
2228 | return false;
|
---|
2229 | }
|
---|
2230 |
|
---|
2231 | // Record the state of this URI
|
---|
2232 | isDir = isDirectory();
|
---|
2233 |
|
---|
2234 | // We are based of the directory of the given URI. We need to
|
---|
2235 | // make sure the URI is pointing to a directory. Changing
|
---|
2236 | // directory to "." will remove any file name if the base is
|
---|
2237 | // not a directory.
|
---|
2238 | base.chdir(".");
|
---|
2239 |
|
---|
2240 | thisParts = pathStr.split("/");
|
---|
2241 | thatParts = base.path.split("/");
|
---|
2242 |
|
---|
2243 | if (thisParts.length > 0 && thisParts[0] == "")
|
---|
2244 | thisParts.shift();
|
---|
2245 |
|
---|
2246 | if (thisParts.length > 0 && thisParts[thisParts.length - 1] == "")
|
---|
2247 | {
|
---|
2248 | isDir = true;
|
---|
2249 | thisParts.pop();
|
---|
2250 | }
|
---|
2251 |
|
---|
2252 | if (thatParts.length > 0 && thatParts[0] == "")
|
---|
2253 | thatParts.shift();
|
---|
2254 | if (thatParts.length > 0 && thatParts[thatParts.length - 1] == "")
|
---|
2255 | thatParts.pop();
|
---|
2256 |
|
---|
2257 |
|
---|
2258 | // Now that we have the paths split into an array of directories,
|
---|
2259 | // we can compare the two paths. We start from the left of side
|
---|
2260 | // of the path and start comparing. When we either run out of
|
---|
2261 | // directories (one path is longer than the other), or we find
|
---|
2262 | // a directory that is different, we stop. The remaining parts
|
---|
2263 | // of each path is then used to determine the relative path. For
|
---|
2264 | // example, lets say we have:
|
---|
2265 | // path we want to make relative: /a/b/c/d/e.txt
|
---|
2266 | // path to use as base for relative: /a/b/f/
|
---|
2267 | //
|
---|
2268 | // This loop will start at the left, and remove directories
|
---|
2269 | // until we get a mismatch or run off the end of one of them.
|
---|
2270 | // In this example, the result will be:
|
---|
2271 | // c/d/e.txt
|
---|
2272 | // f
|
---|
2273 | //
|
---|
2274 | // For every part left over in the base path, we prepend a ".."
|
---|
2275 | // to the relative to get the final path:
|
---|
2276 | // ../c/d/e.txt
|
---|
2277 | while(thatParts.length > 0)
|
---|
2278 | {
|
---|
2279 | if (thisParts.length == 0)
|
---|
2280 | {
|
---|
2281 | // we matched all there is to match, we are done.
|
---|
2282 | // This is the case where "this" object is a parent
|
---|
2283 | // path of the given URI. eg:
|
---|
2284 | // this.path = /a/b/ (thisParts)
|
---|
2285 | // base.path = /a/b/c/d/e/ (thatParts)
|
---|
2286 | break;
|
---|
2287 | }
|
---|
2288 |
|
---|
2289 | thisPart = thisParts[0];
|
---|
2290 | thatPart = thatParts[0];
|
---|
2291 |
|
---|
2292 | if (compareStr(thisPart, thatPart, caseSensitive))
|
---|
2293 | {
|
---|
2294 | thisParts.shift();
|
---|
2295 | thatParts.shift();
|
---|
2296 | }
|
---|
2297 | else
|
---|
2298 | break;
|
---|
2299 | }
|
---|
2300 |
|
---|
2301 | // If there are any path info left from the base URI, that means
|
---|
2302 | // **this** object is above the given URI in the file tree. For
|
---|
2303 | // each part left over in the given URI, we need to move up one
|
---|
2304 | // directory to get where we are.
|
---|
2305 | var dotdot:String = "..";
|
---|
2306 | for (i = 0; i < thatParts.length; i++)
|
---|
2307 | {
|
---|
2308 | finalParts.push(dotdot);
|
---|
2309 | }
|
---|
2310 |
|
---|
2311 | // Append the parts of this URI to any dot-dot's we have
|
---|
2312 | finalParts = finalParts.concat(thisParts);
|
---|
2313 |
|
---|
2314 | // Join the parts back into a path
|
---|
2315 | finalPath = joinPath(finalParts, false /* not absolute */, isDir);
|
---|
2316 |
|
---|
2317 | if (finalPath.length == 0)
|
---|
2318 | {
|
---|
2319 | // The two URI's are exactly the same. The proper relative
|
---|
2320 | // path is:
|
---|
2321 | finalPath = "./";
|
---|
2322 | }
|
---|
2323 |
|
---|
2324 | // Set the parts of the URI, preserving the original query and
|
---|
2325 | // fragment parts.
|
---|
2326 | setParts("", "", "", finalPath, queryStr, fragmentStr);
|
---|
2327 |
|
---|
2328 | return true;
|
---|
2329 | }
|
---|
2330 |
|
---|
2331 | /**
|
---|
2332 | * Given a string, convert it to a URI. The string could be a
|
---|
2333 | * full URI that is improperly escaped, a malformed URI (e.g.
|
---|
2334 | * missing a protocol like "www.something.com"), a relative URI,
|
---|
2335 | * or any variation there of.
|
---|
2336 | *
|
---|
2337 | * <p>The intention of this function is to take anything that a
|
---|
2338 | * user might manually enter as a URI/URL and try to determine what
|
---|
2339 | * they mean. This function differs from the URI constructor in
|
---|
2340 | * that it makes some assumptions to make it easy to import user
|
---|
2341 | * entered URI data.</p>
|
---|
2342 | *
|
---|
2343 | * <p>This function is intended to be a helper function.
|
---|
2344 | * It is not all-knowning and will probably make mistakes
|
---|
2345 | * when attempting to parse a string of unknown origin. If
|
---|
2346 | * your applicaiton is receiving input from the user, your
|
---|
2347 | * application should already have a good idea what the user
|
---|
2348 | * should be entering, and your application should be
|
---|
2349 | * pre-processing the user's input to make sure it is well formed
|
---|
2350 | * before passing it to this function.</p>
|
---|
2351 | *
|
---|
2352 | * <p>It is assumed that the string given to this function is
|
---|
2353 | * something the user may have manually entered. Given this,
|
---|
2354 | * the URI string is probably unescaped or improperly escaped.
|
---|
2355 | * This function will attempt to properly escape the URI by
|
---|
2356 | * using forceEscape(). The result is that a toString() call
|
---|
2357 | * on a URI that was created from unknownToURI() may not match
|
---|
2358 | * the input string due to the difference in escaping.</p>
|
---|
2359 | *
|
---|
2360 | * @param unknown a potental URI string that should be parsed
|
---|
2361 | * and loaded into this object.
|
---|
2362 | * @param defaultScheme if it is determined that the passed string
|
---|
2363 | * looks like a URI, but it is missing the scheme part, this
|
---|
2364 | * string will be used as the missing scheme.
|
---|
2365 | *
|
---|
2366 | * @return true if the given string was successfully parsed into
|
---|
2367 | * a valid URI object, false otherwise.
|
---|
2368 | */
|
---|
2369 | public function unknownToURI(unknown:String, defaultScheme:String = "http") : Boolean
|
---|
2370 | {
|
---|
2371 | var temp:String;
|
---|
2372 |
|
---|
2373 | if (unknown.length == 0)
|
---|
2374 | {
|
---|
2375 | this.initialize();
|
---|
2376 | return false;
|
---|
2377 | }
|
---|
2378 |
|
---|
2379 | // Some users love the backslash key. Fix it.
|
---|
2380 | unknown = unknown.replace(/\\/g, "/");
|
---|
2381 |
|
---|
2382 | // Check for any obviously missing scheme.
|
---|
2383 | if (unknown.length >= 2)
|
---|
2384 | {
|
---|
2385 | temp = unknown.substr(0, 2);
|
---|
2386 | if (temp == "//")
|
---|
2387 | unknown = defaultScheme + ":" + unknown;
|
---|
2388 | }
|
---|
2389 |
|
---|
2390 | if (unknown.length >= 3)
|
---|
2391 | {
|
---|
2392 | temp = unknown.substr(0, 3);
|
---|
2393 | if (temp == "://")
|
---|
2394 | unknown = defaultScheme + unknown;
|
---|
2395 | }
|
---|
2396 |
|
---|
2397 | // Try parsing it as a normal URI
|
---|
2398 | var uri:URI = new URI(unknown);
|
---|
2399 |
|
---|
2400 | if (uri.isHierarchical() == false)
|
---|
2401 | {
|
---|
2402 | if (uri.scheme == UNKNOWN_SCHEME)
|
---|
2403 | {
|
---|
2404 | this.initialize();
|
---|
2405 | return false;
|
---|
2406 | }
|
---|
2407 |
|
---|
2408 | // It's a non-hierarchical URI
|
---|
2409 | copyURI(uri);
|
---|
2410 | forceEscape();
|
---|
2411 | return true;
|
---|
2412 | }
|
---|
2413 | else if ((uri.scheme != UNKNOWN_SCHEME) &&
|
---|
2414 | (uri.scheme.length > 0))
|
---|
2415 | {
|
---|
2416 | if ( (uri.authority.length > 0) ||
|
---|
2417 | (uri.scheme == "file") )
|
---|
2418 | {
|
---|
2419 | // file://... URI
|
---|
2420 | copyURI(uri);
|
---|
2421 | forceEscape(); // ensure proper escaping
|
---|
2422 | return true;
|
---|
2423 | }
|
---|
2424 | else if (uri.authority.length == 0 && uri.path.length == 0)
|
---|
2425 | {
|
---|
2426 | // It's is an incomplete URI (eg "http://")
|
---|
2427 |
|
---|
2428 | setParts(uri.scheme, "", "", "", "", "");
|
---|
2429 | return false;
|
---|
2430 | }
|
---|
2431 | }
|
---|
2432 | else
|
---|
2433 | {
|
---|
2434 | // Possible relative URI. We can only detect relative URI's
|
---|
2435 | // that start with "." or "..". If it starts with something
|
---|
2436 | // else, the parsing is ambiguous.
|
---|
2437 | var path:String = uri.path;
|
---|
2438 |
|
---|
2439 | if (path == ".." || path == "." ||
|
---|
2440 | (path.length >= 3 && path.substr(0, 3) == "../") ||
|
---|
2441 | (path.length >= 2 && path.substr(0, 2) == "./") )
|
---|
2442 | {
|
---|
2443 | // This is a relative URI.
|
---|
2444 | copyURI(uri);
|
---|
2445 | forceEscape();
|
---|
2446 | return true;
|
---|
2447 | }
|
---|
2448 | }
|
---|
2449 |
|
---|
2450 | // Ok, it looks like we are just a normal URI missing the scheme. Tack
|
---|
2451 | // on the scheme.
|
---|
2452 | uri = new URI(defaultScheme + "://" + unknown);
|
---|
2453 |
|
---|
2454 | // Check to see if we are good now
|
---|
2455 | if (uri.scheme.length > 0 && uri.authority.length > 0)
|
---|
2456 | {
|
---|
2457 | // It was just missing the scheme.
|
---|
2458 | copyURI(uri);
|
---|
2459 | forceEscape(); // Make sure we are properly encoded.
|
---|
2460 | return true;
|
---|
2461 | }
|
---|
2462 |
|
---|
2463 | // don't know what this is
|
---|
2464 | this.initialize();
|
---|
2465 | return false;
|
---|
2466 | }
|
---|
2467 |
|
---|
2468 | } // end URI class
|
---|
2469 | } // end package
|
---|