uri_parser 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. data/.gitignore +6 -0
  2. data/.rvmrc +1 -0
  3. data/Gemfile +6 -0
  4. data/Rakefile +13 -0
  5. data/ext/uri_parser/basictypes.h +89 -0
  6. data/ext/uri_parser/extconf.h +6 -0
  7. data/ext/uri_parser/extconf.rb +50 -0
  8. data/ext/uri_parser/logging.h +5 -0
  9. data/ext/uri_parser/scoped_ptr.h +322 -0
  10. data/ext/uri_parser/string16.cc +95 -0
  11. data/ext/uri_parser/string16.h +194 -0
  12. data/ext/uri_parser/uri_parser.cc +87 -0
  13. data/ext/uri_parser/url_canon.h +872 -0
  14. data/ext/uri_parser/url_canon_etc.cc +392 -0
  15. data/ext/uri_parser/url_canon_fileurl.cc +215 -0
  16. data/ext/uri_parser/url_canon_host.cc +401 -0
  17. data/ext/uri_parser/url_canon_icu.cc +207 -0
  18. data/ext/uri_parser/url_canon_icu.h +63 -0
  19. data/ext/uri_parser/url_canon_internal.cc +427 -0
  20. data/ext/uri_parser/url_canon_internal.h +453 -0
  21. data/ext/uri_parser/url_canon_internal_file.h +157 -0
  22. data/ext/uri_parser/url_canon_ip.cc +737 -0
  23. data/ext/uri_parser/url_canon_ip.h +101 -0
  24. data/ext/uri_parser/url_canon_mailtourl.cc +137 -0
  25. data/ext/uri_parser/url_canon_path.cc +380 -0
  26. data/ext/uri_parser/url_canon_pathurl.cc +128 -0
  27. data/ext/uri_parser/url_canon_query.cc +189 -0
  28. data/ext/uri_parser/url_canon_relative.cc +572 -0
  29. data/ext/uri_parser/url_canon_stdstring.h +134 -0
  30. data/ext/uri_parser/url_canon_stdurl.cc +211 -0
  31. data/ext/uri_parser/url_common.h +48 -0
  32. data/ext/uri_parser/url_file.h +108 -0
  33. data/ext/uri_parser/url_parse.cc +760 -0
  34. data/ext/uri_parser/url_parse.h +336 -0
  35. data/ext/uri_parser/url_parse_file.cc +243 -0
  36. data/ext/uri_parser/url_parse_internal.h +112 -0
  37. data/ext/uri_parser/url_util.cc +553 -0
  38. data/ext/uri_parser/url_util.h +222 -0
  39. data/lib/uri_parser.rb +28 -0
  40. data/lib/uri_parser/version.rb +3 -0
  41. data/spec/spec_helper.rb +16 -0
  42. data/spec/uri_parser_spec.rb +54 -0
  43. data/uri_parser.gemspec +26 -0
  44. metadata +117 -0
@@ -0,0 +1,112 @@
1
+ // Copyright 2007, Google Inc.
2
+ // All rights reserved.
3
+ //
4
+ // Redistribution and use in source and binary forms, with or without
5
+ // modification, are permitted provided that the following conditions are
6
+ // met:
7
+ //
8
+ // * Redistributions of source code must retain the above copyright
9
+ // notice, this list of conditions and the following disclaimer.
10
+ // * Redistributions in binary form must reproduce the above
11
+ // copyright notice, this list of conditions and the following disclaimer
12
+ // in the documentation and/or other materials provided with the
13
+ // distribution.
14
+ // * Neither the name of Google Inc. nor the names of its
15
+ // contributors may be used to endorse or promote products derived from
16
+ // this software without specific prior written permission.
17
+ //
18
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
+
30
+ // Contains common inline helper functions used by the URL parsing routines.
31
+
32
+ #ifndef GOOGLEURL_SRC_URL_PARSE_INTERNAL_H__
33
+ #define GOOGLEURL_SRC_URL_PARSE_INTERNAL_H__
34
+
35
+ #include "url_parse.h"
36
+
37
+ namespace url_parse {
38
+
39
+ // We treat slashes and backslashes the same for IE compatability.
40
+ inline bool IsURLSlash(char16 ch) {
41
+ return ch == '/' || ch == '\\';
42
+ }
43
+
44
+ // Returns true if we should trim this character from the URL because it is a
45
+ // space or a control character.
46
+ inline bool ShouldTrimFromURL(char16 ch) {
47
+ return ch <= ' ';
48
+ }
49
+
50
+ // Given an already-initialized begin index and length, this shrinks the range
51
+ // to eliminate "should-be-trimmed" characters. Note that the length does *not*
52
+ // indicate the length of untrimmed data from |*begin|, but rather the position
53
+ // in the input string (so the string starts at character |*begin| in the spec,
54
+ // and goes until |*len|).
55
+ template<typename CHAR>
56
+ inline void TrimURL(const CHAR* spec, int* begin, int* len) {
57
+ // Strip leading whitespace and control characters.
58
+ while (*begin < *len && ShouldTrimFromURL(spec[*begin]))
59
+ (*begin)++;
60
+
61
+ // Strip trailing whitespace and control characters. We need the >i test for
62
+ // when the input string is all blanks; we don't want to back past the input.
63
+ while (*len > *begin && ShouldTrimFromURL(spec[*len - 1]))
64
+ (*len)--;
65
+ }
66
+
67
+ // Counts the number of consecutive slashes starting at the given offset
68
+ // in the given string of the given length.
69
+ template<typename CHAR>
70
+ inline int CountConsecutiveSlashes(const CHAR *str,
71
+ int begin_offset, int str_len) {
72
+ int count = 0;
73
+ while (begin_offset + count < str_len &&
74
+ IsURLSlash(str[begin_offset + count]))
75
+ ++count;
76
+ return count;
77
+ }
78
+
79
+ // Internal functions in url_parse.cc that parse the path, that is, everything
80
+ // following the authority section. The input is the range of everything
81
+ // following the authority section, and the output is the identified ranges.
82
+ //
83
+ // This is designed for the file URL parser or other consumers who may do
84
+ // special stuff at the beginning, but want regular path parsing, it just
85
+ // maps to the internal parsing function for paths.
86
+ void ParsePathInternal(const char* spec,
87
+ const Component& path,
88
+ Component* filepath,
89
+ Component* query,
90
+ Component* ref);
91
+ void ParsePathInternal(const char16* spec,
92
+ const Component& path,
93
+ Component* filepath,
94
+ Component* query,
95
+ Component* ref);
96
+
97
+
98
+ // Given a spec and a pointer to the character after the colon following the
99
+ // scheme, this parses it and fills in the structure, Every item in the parsed
100
+ // structure is filled EXCEPT for the scheme, which is untouched.
101
+ void ParseAfterScheme(const char* spec,
102
+ int spec_len,
103
+ int after_scheme,
104
+ Parsed* parsed);
105
+ void ParseAfterScheme(const char16* spec,
106
+ int spec_len,
107
+ int after_scheme,
108
+ Parsed* parsed);
109
+
110
+ } // namespace url_parse
111
+
112
+ #endif // GOOGLEURL_SRC_URL_PARSE_INTERNAL_H__
@@ -0,0 +1,553 @@
1
+ // Copyright 2007, Google Inc.
2
+ // All rights reserved.
3
+ //
4
+ // Redistribution and use in source and binary forms, with or without
5
+ // modification, are permitted provided that the following conditions are
6
+ // met:
7
+ //
8
+ // * Redistributions of source code must retain the above copyright
9
+ // notice, this list of conditions and the following disclaimer.
10
+ // * Redistributions in binary form must reproduce the above
11
+ // copyright notice, this list of conditions and the following disclaimer
12
+ // in the documentation and/or other materials provided with the
13
+ // distribution.
14
+ // * Neither the name of Google Inc. nor the names of its
15
+ // contributors may be used to endorse or promote products derived from
16
+ // this software without specific prior written permission.
17
+ //
18
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
+
30
+ #include <string.h>
31
+ #include <vector>
32
+
33
+ #include "url_util.h"
34
+
35
+ #include "logging.h"
36
+ #include "url_canon_internal.h"
37
+ #include "url_file.h"
38
+
39
+ namespace url_util {
40
+
41
+ namespace {
42
+
43
+ // ASCII-specific tolower. The standard library's tolower is locale sensitive,
44
+ // so we don't want to use it here.
45
+ template <class Char> inline Char ToLowerASCII(Char c) {
46
+ return (c >= 'A' && c <= 'Z') ? (c + ('a' - 'A')) : c;
47
+ }
48
+
49
+ // Backend for LowerCaseEqualsASCII.
50
+ template<typename Iter>
51
+ inline bool DoLowerCaseEqualsASCII(Iter a_begin, Iter a_end, const char* b) {
52
+ for (Iter it = a_begin; it != a_end; ++it, ++b) {
53
+ if (!*b || ToLowerASCII(*it) != *b)
54
+ return false;
55
+ }
56
+ return *b == 0;
57
+ }
58
+
59
+ const char kFileScheme[] = "file"; // Used in a number of places.
60
+ const char kMailtoScheme[] = "mailto";
61
+
62
+ const int kNumStandardURLSchemes = 7;
63
+ const char* kStandardURLSchemes[kNumStandardURLSchemes] = {
64
+ "http",
65
+ "https",
66
+ kFileScheme, // Yes, file urls can have a hostname!
67
+ "ftp",
68
+ "gopher",
69
+ "ws", // WebSocket.
70
+ "wss", // WebSocket secure.
71
+ };
72
+
73
+ // List of the currently installed standard schemes. This list is lazily
74
+ // initialized by InitStandardSchemes and is leaked on shutdown to prevent
75
+ // any destructors from being called that will slow us down or cause problems.
76
+ std::vector<const char*>* standard_schemes = NULL;
77
+
78
+ // See the LockStandardSchemes declaration in the header.
79
+ bool standard_schemes_locked = false;
80
+
81
+ // Ensures that the standard_schemes list is initialized, does nothing if it
82
+ // already has values.
83
+ void InitStandardSchemes() {
84
+ if (standard_schemes)
85
+ return;
86
+ standard_schemes = new std::vector<const char*>;
87
+ for (int i = 0; i < kNumStandardURLSchemes; i++)
88
+ standard_schemes->push_back(kStandardURLSchemes[i]);
89
+ }
90
+
91
+ // Given a string and a range inside the string, compares it to the given
92
+ // lower-case |compare_to| buffer.
93
+ template<typename CHAR>
94
+ inline bool CompareSchemeComponent(const CHAR* spec,
95
+ const url_parse::Component& component,
96
+ const char* compare_to) {
97
+ if (!component.is_nonempty())
98
+ return compare_to[0] == 0; // When component is empty, match empty scheme.
99
+ return LowerCaseEqualsASCII(&spec[component.begin],
100
+ &spec[component.end()],
101
+ compare_to);
102
+ }
103
+
104
+ // Returns true if the given scheme identified by |scheme| within |spec| is one
105
+ // of the registered "standard" schemes.
106
+ template<typename CHAR>
107
+ bool DoIsStandard(const CHAR* spec, const url_parse::Component& scheme) {
108
+ if (!scheme.is_nonempty())
109
+ return false; // Empty or invalid schemes are non-standard.
110
+
111
+ InitStandardSchemes();
112
+ for (size_t i = 0; i < standard_schemes->size(); i++) {
113
+ if (LowerCaseEqualsASCII(&spec[scheme.begin], &spec[scheme.end()],
114
+ standard_schemes->at(i)))
115
+ return true;
116
+ }
117
+ return false;
118
+ }
119
+
120
+ template<typename CHAR>
121
+ bool DoFindAndCompareScheme(const CHAR* str,
122
+ int str_len,
123
+ const char* compare,
124
+ url_parse::Component* found_scheme) {
125
+ // Before extracting scheme, canonicalize the URL to remove any whitespace.
126
+ // This matches the canonicalization done in DoCanonicalize function.
127
+ url_canon::RawCanonOutputT<CHAR> whitespace_buffer;
128
+ int spec_len;
129
+ const CHAR* spec = RemoveURLWhitespace(str, str_len,
130
+ &whitespace_buffer, &spec_len);
131
+
132
+ url_parse::Component our_scheme;
133
+ if (!url_parse::ExtractScheme(spec, spec_len, &our_scheme)) {
134
+ // No scheme.
135
+ if (found_scheme)
136
+ *found_scheme = url_parse::Component();
137
+ return false;
138
+ }
139
+ if (found_scheme)
140
+ *found_scheme = our_scheme;
141
+ return CompareSchemeComponent(spec, our_scheme, compare);
142
+ }
143
+
144
+ template<typename CHAR>
145
+ bool DoCanonicalize(const CHAR* in_spec, int in_spec_len,
146
+ url_canon::CharsetConverter* charset_converter,
147
+ url_canon::CanonOutput* output,
148
+ url_parse::Parsed* output_parsed) {
149
+ // Remove any whitespace from the middle of the relative URL, possibly
150
+ // copying to the new buffer.
151
+ url_canon::RawCanonOutputT<CHAR> whitespace_buffer;
152
+ int spec_len;
153
+ const CHAR* spec = RemoveURLWhitespace(in_spec, in_spec_len,
154
+ &whitespace_buffer, &spec_len);
155
+
156
+ url_parse::Parsed parsed_input;
157
+ #ifdef WIN32
158
+ // For Windows, we allow things that look like absolute Windows paths to be
159
+ // fixed up magically to file URLs. This is done for IE compatability. For
160
+ // example, this will change "c:/foo" into a file URL rather than treating
161
+ // it as a URL with the protocol "c". It also works for UNC ("\\foo\bar.txt").
162
+ // There is similar logic in url_canon_relative.cc for
163
+ //
164
+ // For Max & Unix, we don't do this (the equivalent would be "/foo/bar" which
165
+ // has no meaning as an absolute path name. This is because browsers on Mac
166
+ // & Unix don't generally do this, so there is no compatibility reason for
167
+ // doing so.
168
+ if (url_parse::DoesBeginUNCPath(spec, 0, spec_len, false) ||
169
+ url_parse::DoesBeginWindowsDriveSpec(spec, 0, spec_len)) {
170
+ url_parse::ParseFileURL(spec, spec_len, &parsed_input);
171
+ return url_canon::CanonicalizeFileURL(spec, spec_len, parsed_input,
172
+ charset_converter,
173
+ output, output_parsed);
174
+ }
175
+ #endif
176
+
177
+ url_parse::Component scheme;
178
+ if (!url_parse::ExtractScheme(spec, spec_len, &scheme))
179
+ return false;
180
+
181
+ // This is the parsed version of the input URL, we have to canonicalize it
182
+ // before storing it in our object.
183
+ bool success;
184
+ if (CompareSchemeComponent(spec, scheme, kFileScheme)) {
185
+ // File URLs are special.
186
+ url_parse::ParseFileURL(spec, spec_len, &parsed_input);
187
+ success = url_canon::CanonicalizeFileURL(spec, spec_len, parsed_input,
188
+ charset_converter,
189
+ output, output_parsed);
190
+
191
+ } else if (DoIsStandard(spec, scheme)) {
192
+ // All "normal" URLs.
193
+ url_parse::ParseStandardURL(spec, spec_len, &parsed_input);
194
+ success = url_canon::CanonicalizeStandardURL(spec, spec_len, parsed_input,
195
+ charset_converter,
196
+ output, output_parsed);
197
+
198
+ } else if (CompareSchemeComponent(spec, scheme, kMailtoScheme)) {
199
+ // Mailto are treated like a standard url with only a scheme, path, query
200
+ url_parse::ParseMailtoURL(spec, spec_len, &parsed_input);
201
+ success = url_canon::CanonicalizeMailtoURL(spec, spec_len, parsed_input,
202
+ output, output_parsed);
203
+
204
+ } else {
205
+ // "Weird" URLs like data: and javascript:
206
+ url_parse::ParsePathURL(spec, spec_len, &parsed_input);
207
+ success = url_canon::CanonicalizePathURL(spec, spec_len, parsed_input,
208
+ output, output_parsed);
209
+ }
210
+ return success;
211
+ }
212
+
213
+ template<typename CHAR>
214
+ bool DoResolveRelative(const char* base_spec,
215
+ int base_spec_len,
216
+ const url_parse::Parsed& base_parsed,
217
+ const CHAR* in_relative,
218
+ int in_relative_length,
219
+ url_canon::CharsetConverter* charset_converter,
220
+ url_canon::CanonOutput* output,
221
+ url_parse::Parsed* output_parsed) {
222
+ // Remove any whitespace from the middle of the relative URL, possibly
223
+ // copying to the new buffer.
224
+ url_canon::RawCanonOutputT<CHAR> whitespace_buffer;
225
+ int relative_length;
226
+ const CHAR* relative = RemoveURLWhitespace(in_relative, in_relative_length,
227
+ &whitespace_buffer,
228
+ &relative_length);
229
+
230
+ // See if our base URL should be treated as "standard".
231
+ bool standard_base_scheme =
232
+ base_parsed.scheme.is_nonempty() &&
233
+ DoIsStandard(base_spec, base_parsed.scheme);
234
+
235
+ bool is_relative;
236
+ url_parse::Component relative_component;
237
+ if (!url_canon::IsRelativeURL(base_spec, base_parsed,
238
+ relative, relative_length,
239
+ standard_base_scheme,
240
+ &is_relative,
241
+ &relative_component)) {
242
+ // Error resolving.
243
+ return false;
244
+ }
245
+
246
+ if (is_relative) {
247
+ // Relative, resolve and canonicalize.
248
+ bool file_base_scheme = base_parsed.scheme.is_nonempty() &&
249
+ CompareSchemeComponent(base_spec, base_parsed.scheme, kFileScheme);
250
+ return url_canon::ResolveRelativeURL(base_spec, base_parsed,
251
+ file_base_scheme, relative,
252
+ relative_component, charset_converter,
253
+ output, output_parsed);
254
+ }
255
+
256
+ // Not relative, canonicalize the input.
257
+ return DoCanonicalize(relative, relative_length, charset_converter,
258
+ output, output_parsed);
259
+ }
260
+
261
+ template<typename CHAR>
262
+ bool DoReplaceComponents(const char* spec,
263
+ int spec_len,
264
+ const url_parse::Parsed& parsed,
265
+ const url_canon::Replacements<CHAR>& replacements,
266
+ url_canon::CharsetConverter* charset_converter,
267
+ url_canon::CanonOutput* output,
268
+ url_parse::Parsed* out_parsed) {
269
+ // If the scheme is overridden, just do a simple string substitution and
270
+ // reparse the whole thing. There are lots of edge cases that we really don't
271
+ // want to deal with. Like what happens if I replace "http://e:8080/foo"
272
+ // with a file. Does it become "file:///E:/8080/foo" where the port number
273
+ // becomes part of the path? Parsing that string as a file URL says "yes"
274
+ // but almost no sane rule for dealing with the components individually would
275
+ // come up with that.
276
+ //
277
+ // Why allow these crazy cases at all? Programatically, there is almost no
278
+ // case for replacing the scheme. The most common case for hitting this is
279
+ // in JS when building up a URL using the location object. In this case, the
280
+ // JS code expects the string substitution behavior:
281
+ // http://www.w3.org/TR/2008/WD-html5-20080610/structured.html#common3
282
+ if (replacements.IsSchemeOverridden()) {
283
+ // Canonicalize the new scheme so it is 8-bit and can be concatenated with
284
+ // the existing spec.
285
+ url_canon::RawCanonOutput<128> scheme_replaced;
286
+ url_parse::Component scheme_replaced_parsed;
287
+ url_canon::CanonicalizeScheme(
288
+ replacements.sources().scheme,
289
+ replacements.components().scheme,
290
+ &scheme_replaced, &scheme_replaced_parsed);
291
+
292
+ // We can assume that the input is canonicalized, which means it always has
293
+ // a colon after the scheme (or where the scheme would be).
294
+ int spec_after_colon = parsed.scheme.is_valid() ? parsed.scheme.end() + 1
295
+ : 1;
296
+ if (spec_len - spec_after_colon > 0) {
297
+ scheme_replaced.Append(&spec[spec_after_colon],
298
+ spec_len - spec_after_colon);
299
+ }
300
+
301
+ // We now need to completely re-parse the resulting string since its meaning
302
+ // may have changed with the different scheme.
303
+ url_canon::RawCanonOutput<128> recanonicalized;
304
+ url_parse::Parsed recanonicalized_parsed;
305
+ DoCanonicalize(scheme_replaced.data(), scheme_replaced.length(),
306
+ charset_converter,
307
+ &recanonicalized, &recanonicalized_parsed);
308
+
309
+ // Recurse using the version with the scheme already replaced. This will now
310
+ // use the replacement rules for the new scheme.
311
+ //
312
+ // Warning: this code assumes that ReplaceComponents will re-check all
313
+ // components for validity. This is because we can't fail if DoCanonicalize
314
+ // failed above since theoretically the thing making it fail could be
315
+ // getting replaced here. If ReplaceComponents didn't re-check everything,
316
+ // we wouldn't know if something *not* getting replaced is a problem.
317
+ // If the scheme-specific replacers are made more intelligent so they don't
318
+ // re-check everything, we should instead recanonicalize the whole thing
319
+ // after this call to check validity (this assumes replacing the scheme is
320
+ // much much less common than other types of replacements, like clearing the
321
+ // ref).
322
+ url_canon::Replacements<CHAR> replacements_no_scheme = replacements;
323
+ replacements_no_scheme.SetScheme(NULL, url_parse::Component());
324
+ return DoReplaceComponents(recanonicalized.data(), recanonicalized.length(),
325
+ recanonicalized_parsed, replacements_no_scheme,
326
+ charset_converter, output, out_parsed);
327
+ }
328
+
329
+ // If we get here, then we know the scheme doesn't need to be replaced, so can
330
+ // just key off the scheme in the spec to know how to do the replacements.
331
+ if (CompareSchemeComponent(spec, parsed.scheme, kFileScheme)) {
332
+ return url_canon::ReplaceFileURL(spec, parsed, replacements,
333
+ charset_converter, output, out_parsed);
334
+ }
335
+ if (DoIsStandard(spec, parsed.scheme)) {
336
+ return url_canon::ReplaceStandardURL(spec, parsed, replacements,
337
+ charset_converter, output, out_parsed);
338
+ }
339
+ if (CompareSchemeComponent(spec, parsed.scheme, kMailtoScheme)) {
340
+ return url_canon::ReplaceMailtoURL(spec, parsed, replacements,
341
+ output, out_parsed);
342
+ }
343
+
344
+ // Default is a path URL.
345
+ return url_canon::ReplacePathURL(spec, parsed, replacements,
346
+ output, out_parsed);
347
+ }
348
+
349
+ } // namespace
350
+
351
+ void Initialize() {
352
+ InitStandardSchemes();
353
+ }
354
+
355
+ void Shutdown() {
356
+ if (standard_schemes) {
357
+ delete standard_schemes;
358
+ standard_schemes = NULL;
359
+ }
360
+ }
361
+
362
+ void AddStandardScheme(const char* new_scheme) {
363
+ // If this assert triggers, it means you've called AddStandardScheme after
364
+ // LockStandardSchemes have been called (see the header file for
365
+ // LockStandardSchemes for more).
366
+ //
367
+ // This normally means you're trying to set up a new standard scheme too late
368
+ // in your application's init process. Locate where your app does this
369
+ // initialization and calls LockStandardScheme, and add your new standard
370
+ // scheme there.
371
+ DCHECK(!standard_schemes_locked) <<
372
+ "Trying to add a standard scheme after the list has been locked.";
373
+
374
+ size_t scheme_len = strlen(new_scheme);
375
+ if (scheme_len == 0)
376
+ return;
377
+
378
+ // Dulicate the scheme into a new buffer and add it to the list of standard
379
+ // schemes. This pointer will be leaked on shutdown.
380
+ char* dup_scheme = new char[scheme_len + 1];
381
+ memcpy(dup_scheme, new_scheme, scheme_len + 1);
382
+
383
+ InitStandardSchemes();
384
+ standard_schemes->push_back(dup_scheme);
385
+ }
386
+
387
+ void LockStandardSchemes() {
388
+ standard_schemes_locked = true;
389
+ }
390
+
391
+ bool IsStandard(const char* spec, const url_parse::Component& scheme) {
392
+ return DoIsStandard(spec, scheme);
393
+ }
394
+
395
+ bool IsStandard(const char16* spec, const url_parse::Component& scheme) {
396
+ return DoIsStandard(spec, scheme);
397
+ }
398
+
399
+ bool FindAndCompareScheme(const char* str,
400
+ int str_len,
401
+ const char* compare,
402
+ url_parse::Component* found_scheme) {
403
+ return DoFindAndCompareScheme(str, str_len, compare, found_scheme);
404
+ }
405
+
406
+ bool FindAndCompareScheme(const char16* str,
407
+ int str_len,
408
+ const char* compare,
409
+ url_parse::Component* found_scheme) {
410
+ return DoFindAndCompareScheme(str, str_len, compare, found_scheme);
411
+ }
412
+
413
+ bool Canonicalize(const char* spec,
414
+ int spec_len,
415
+ url_canon::CharsetConverter* charset_converter,
416
+ url_canon::CanonOutput* output,
417
+ url_parse::Parsed* output_parsed) {
418
+ return DoCanonicalize(spec, spec_len, charset_converter,
419
+ output, output_parsed);
420
+ }
421
+
422
+ bool Canonicalize(const char16* spec,
423
+ int spec_len,
424
+ url_canon::CharsetConverter* charset_converter,
425
+ url_canon::CanonOutput* output,
426
+ url_parse::Parsed* output_parsed) {
427
+ return DoCanonicalize(spec, spec_len, charset_converter,
428
+ output, output_parsed);
429
+ }
430
+
431
+ bool ResolveRelative(const char* base_spec,
432
+ int base_spec_len,
433
+ const url_parse::Parsed& base_parsed,
434
+ const char* relative,
435
+ int relative_length,
436
+ url_canon::CharsetConverter* charset_converter,
437
+ url_canon::CanonOutput* output,
438
+ url_parse::Parsed* output_parsed) {
439
+ return DoResolveRelative(base_spec, base_spec_len, base_parsed,
440
+ relative, relative_length,
441
+ charset_converter, output, output_parsed);
442
+ }
443
+
444
+ bool ResolveRelative(const char* base_spec,
445
+ int base_spec_len,
446
+ const url_parse::Parsed& base_parsed,
447
+ const char16* relative,
448
+ int relative_length,
449
+ url_canon::CharsetConverter* charset_converter,
450
+ url_canon::CanonOutput* output,
451
+ url_parse::Parsed* output_parsed) {
452
+ return DoResolveRelative(base_spec, base_spec_len, base_parsed,
453
+ relative, relative_length,
454
+ charset_converter, output, output_parsed);
455
+ }
456
+
457
+ bool ReplaceComponents(const char* spec,
458
+ int spec_len,
459
+ const url_parse::Parsed& parsed,
460
+ const url_canon::Replacements<char>& replacements,
461
+ url_canon::CharsetConverter* charset_converter,
462
+ url_canon::CanonOutput* output,
463
+ url_parse::Parsed* out_parsed) {
464
+ return DoReplaceComponents(spec, spec_len, parsed, replacements,
465
+ charset_converter, output, out_parsed);
466
+ }
467
+
468
+ bool ReplaceComponents(const char* spec,
469
+ int spec_len,
470
+ const url_parse::Parsed& parsed,
471
+ const url_canon::Replacements<char16>& replacements,
472
+ url_canon::CharsetConverter* charset_converter,
473
+ url_canon::CanonOutput* output,
474
+ url_parse::Parsed* out_parsed) {
475
+ return DoReplaceComponents(spec, spec_len, parsed, replacements,
476
+ charset_converter, output, out_parsed);
477
+ }
478
+
479
+ // Front-ends for LowerCaseEqualsASCII.
480
+ bool LowerCaseEqualsASCII(const char* a_begin,
481
+ const char* a_end,
482
+ const char* b) {
483
+ return DoLowerCaseEqualsASCII(a_begin, a_end, b);
484
+ }
485
+
486
+ bool LowerCaseEqualsASCII(const char* a_begin,
487
+ const char* a_end,
488
+ const char* b_begin,
489
+ const char* b_end) {
490
+ while (a_begin != a_end && b_begin != b_end &&
491
+ ToLowerASCII(*a_begin) == *b_begin) {
492
+ a_begin++;
493
+ b_begin++;
494
+ }
495
+ return a_begin == a_end && b_begin == b_end;
496
+ }
497
+
498
+ bool LowerCaseEqualsASCII(const char16* a_begin,
499
+ const char16* a_end,
500
+ const char* b) {
501
+ return DoLowerCaseEqualsASCII(a_begin, a_end, b);
502
+ }
503
+
504
+ void DecodeURLEscapeSequences(const char* input, int length,
505
+ url_canon::CanonOutputW* output) {
506
+ url_canon::RawCanonOutputT<char> unescaped_chars;
507
+ for (int i = 0; i < length; i++) {
508
+ if (input[i] == '%') {
509
+ unsigned char ch;
510
+ if (url_canon::DecodeEscaped(input, &i, length, &ch)) {
511
+ unescaped_chars.push_back(ch);
512
+ } else {
513
+ // Invalid escape sequence, copy the percent literal.
514
+ unescaped_chars.push_back('%');
515
+ }
516
+ } else {
517
+ // Regular non-escaped 8-bit character.
518
+ unescaped_chars.push_back(input[i]);
519
+ }
520
+ }
521
+
522
+ // Convert that 8-bit to UTF-16. It's not clear IE does this at all to
523
+ // JavaScript URLs, but Firefox and Safari do.
524
+ for (int i = 0; i < unescaped_chars.length(); i++) {
525
+ unsigned char uch = static_cast<unsigned char>(unescaped_chars.at(i));
526
+ if (uch < 0x80) {
527
+ // Non-UTF-8, just append directly
528
+ output->push_back(uch);
529
+ } else {
530
+ // next_ch will point to the last character of the decoded
531
+ // character.
532
+ int next_character = i;
533
+ unsigned code_point;
534
+ if (url_canon::ReadUTFChar(unescaped_chars.data(), &next_character,
535
+ unescaped_chars.length(), &code_point)) {
536
+ // Valid UTF-8 character, convert to UTF-16.
537
+ url_canon::AppendUTF16Value(code_point, output);
538
+ i = next_character;
539
+ } else {
540
+ // If there are any sequences that are not valid UTF-8, we keep
541
+ // invalid code points and promote to UTF-16. We copy all characters
542
+ // from the current position to the end of the identified sequence.
543
+ while (i < next_character) {
544
+ output->push_back(static_cast<unsigned char>(unescaped_chars.at(i)));
545
+ i++;
546
+ }
547
+ output->push_back(static_cast<unsigned char>(unescaped_chars.at(i)));
548
+ }
549
+ }
550
+ }
551
+ }
552
+
553
+ } // namespace url_util