1 | # -*- coding: utf-8 -*-
|
---|
2 | # $Id: webutils.py 56295 2015-06-09 14:29:55Z vboxsync $
|
---|
3 |
|
---|
4 | """
|
---|
5 | Common Web Utility Functions.
|
---|
6 | """
|
---|
7 |
|
---|
8 | __copyright__ = \
|
---|
9 | """
|
---|
10 | Copyright (C) 2012-2015 Oracle Corporation
|
---|
11 |
|
---|
12 | This file is part of VirtualBox Open Source Edition (OSE), as
|
---|
13 | available from http://www.virtualbox.org. This file is free software;
|
---|
14 | you can redistribute it and/or modify it under the terms of the GNU
|
---|
15 | General Public License (GPL) as published by the Free Software
|
---|
16 | Foundation, in version 2 as it comes in the "COPYING" file of the
|
---|
17 | VirtualBox OSE distribution. VirtualBox OSE is distributed in the
|
---|
18 | hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
|
---|
19 |
|
---|
20 | The contents of this file may alternatively be used under the terms
|
---|
21 | of the Common Development and Distribution License Version 1.0
|
---|
22 | (CDDL) only, as it comes in the "COPYING.CDDL" file of the
|
---|
23 | VirtualBox OSE distribution, in which case the provisions of the
|
---|
24 | CDDL are applicable instead of those of the GPL.
|
---|
25 |
|
---|
26 | You may elect to license modified versions of this file under the
|
---|
27 | terms and conditions of either the GPL or the CDDL or both.
|
---|
28 | """
|
---|
29 | __version__ = "$Revision: 56295 $"
|
---|
30 |
|
---|
31 | # Standard Python imports.
|
---|
32 | import os;
|
---|
33 | import shutil;
|
---|
34 | import sys;
|
---|
35 | import unittest;
|
---|
36 |
|
---|
37 | # Validation Kit imports.
|
---|
38 | from common import utils;
|
---|
39 |
|
---|
40 | # Python 3 hacks:
|
---|
41 | if sys.version_info[0] < 3:
|
---|
42 | from urllib import urlopen as urllib_urlopen;
|
---|
43 | from urllib2 import quote as urllib_quote;
|
---|
44 | from urllib import urlencode as urllib_urlencode;
|
---|
45 | else:
|
---|
46 | from urllib.parse import quote as urllib_quote; # pylint: disable=F0401,E0611
|
---|
47 | from urllib.parse import urlencode as urllib_urlencode; # pylint: disable=F0401,E0611
|
---|
48 | from urllib.request import urlopen as urllib_urlopen; # pylint: disable=F0401,E0611
|
---|
49 |
|
---|
50 |
|
---|
51 | def escapeElem(sText):
|
---|
52 | """
|
---|
53 | Escapes special character to HTML-safe sequences.
|
---|
54 | """
|
---|
55 | sText = sText.replace('&', '&')
|
---|
56 | sText = sText.replace('<', '<')
|
---|
57 | return sText.replace('>', '>')
|
---|
58 |
|
---|
59 | def escapeAttr(sText):
|
---|
60 | """
|
---|
61 | Escapes special character to HTML-safe sequences.
|
---|
62 | """
|
---|
63 | sText = sText.replace('&', '&')
|
---|
64 | sText = sText.replace('<', '<')
|
---|
65 | sText = sText.replace('>', '>')
|
---|
66 | return sText.replace('"', '"')
|
---|
67 |
|
---|
68 | def escapeElemToStr(oObject):
|
---|
69 | """
|
---|
70 | Stringifies the object and hands it to escapeElem.
|
---|
71 | """
|
---|
72 | if utils.isString(oObject):
|
---|
73 | return escapeElem(oObject);
|
---|
74 | return escapeElem(str(oObject));
|
---|
75 |
|
---|
76 | def escapeAttrToStr(oObject):
|
---|
77 | """
|
---|
78 | Stringifies the object and hands it to escapeAttr. May return unicode string.
|
---|
79 | """
|
---|
80 | return escapeAttr(oObject);
|
---|
81 |
|
---|
82 | def escapeAttrJavaScriptStringDQ(sText):
|
---|
83 | """ Escapes a javascript string that is to be emitted between double quotes. """
|
---|
84 | if '"' not in sText:
|
---|
85 | chMin = min(sText);
|
---|
86 | if ord(chMin) >= 0x20:
|
---|
87 | return sText;
|
---|
88 |
|
---|
89 | sRet = '';
|
---|
90 | for ch in sText:
|
---|
91 | if ch == '"':
|
---|
92 | sRet += '\\"';
|
---|
93 | elif ord(ch) >= 0x20:
|
---|
94 | sRet += ch;
|
---|
95 | elif ch == '\n':
|
---|
96 | sRet += '\\n';
|
---|
97 | elif ch == '\r':
|
---|
98 | sRet += '\\r';
|
---|
99 | elif ch == '\t':
|
---|
100 | sRet += '\\t';
|
---|
101 | else:
|
---|
102 | sRet += '\\x%02x' % (ch,);
|
---|
103 | return sRet;
|
---|
104 |
|
---|
105 | def quoteUrl(sText):
|
---|
106 | """
|
---|
107 | See urllib.quote().
|
---|
108 | """
|
---|
109 | return urllib_quote(sText);
|
---|
110 |
|
---|
111 | def encodeUrlParams(dParams):
|
---|
112 | """
|
---|
113 | See urllib.urlencode().
|
---|
114 | """
|
---|
115 | return urllib_urlencode(dParams, doseq=True)
|
---|
116 |
|
---|
117 | def hasSchema(sUrl):
|
---|
118 | """
|
---|
119 | Checks if the URL has a schema (e.g. http://) or is file/server relative.
|
---|
120 | Returns True if schema is present, False if not.
|
---|
121 | """
|
---|
122 | iColon = sUrl.find(':');
|
---|
123 | if iColon > 0:
|
---|
124 | sSchema = sUrl[0:iColon];
|
---|
125 | if len(sSchema) >= 2 and len(sSchema) < 16 and sSchema.islower() and sSchema.isalpha():
|
---|
126 | return True;
|
---|
127 | return False;
|
---|
128 |
|
---|
129 | def getFilename(sUrl):
|
---|
130 | """
|
---|
131 | Extracts the filename from the URL.
|
---|
132 | """
|
---|
133 | ## @TODO This isn't entirely correct. Use the urlparser instead!
|
---|
134 | sFilename = os.path.basename(sUrl.replace('/', os.path.sep));
|
---|
135 | return sFilename;
|
---|
136 |
|
---|
137 |
|
---|
138 | def downloadFile(sUrlFile, sDstFile, sLocalPrefix, fnLog, fnError = None, fNoProxies=True):
|
---|
139 | """
|
---|
140 | Downloads the given file if an URL is given, otherwise assume it's
|
---|
141 | something on the build share and copy it from there.
|
---|
142 |
|
---|
143 | Raises no exceptions, returns log + success indicator instead.
|
---|
144 |
|
---|
145 | Note! This method may use proxies configured on the system and the
|
---|
146 | http_proxy, ftp_proxy, no_proxy environment variables.
|
---|
147 |
|
---|
148 | @todo Fixed build burn here. Please set default value for fNoProxies
|
---|
149 | to appropriate one.
|
---|
150 | """
|
---|
151 | if fnError is None:
|
---|
152 | fnError = fnLog;
|
---|
153 |
|
---|
154 | if sUrlFile.startswith('http://') \
|
---|
155 | or sUrlFile.startswith('https://') \
|
---|
156 | or sUrlFile.startswith('ftp://'):
|
---|
157 | # Download the file.
|
---|
158 | fnLog('Downloading "%s" to "%s"...' % (sUrlFile, sDstFile));
|
---|
159 | try:
|
---|
160 | ## @todo We get 404.html content instead of exceptions here, which is confusing and should be addressed.
|
---|
161 | if fNoProxies:
|
---|
162 | oSrc = urllib_urlopen(sUrlFile);
|
---|
163 | else:
|
---|
164 | oSrc = urllib_urlopen(sUrlFile, proxies = dict());
|
---|
165 | oDst = utils.openNoInherit(sDstFile, 'wb');
|
---|
166 | oDst.write(oSrc.read());
|
---|
167 | oDst.close();
|
---|
168 | oSrc.close();
|
---|
169 | except Exception, oXcpt:
|
---|
170 | fnError('Error downloading "%s" to "%s": %s' % (sUrlFile, sDstFile, oXcpt));
|
---|
171 | return False;
|
---|
172 | else:
|
---|
173 | # Assumes file from the build share.
|
---|
174 | sSrcPath = os.path.join(sLocalPrefix, sUrlFile);
|
---|
175 | fnLog('Copying "%s" to "%s"...' % (sSrcPath, sDstFile));
|
---|
176 | try:
|
---|
177 | shutil.copyfile(sSrcPath, sDstFile);
|
---|
178 | except Exception, oXcpt:
|
---|
179 | fnError('Error copying "%s" to "%s": %s' % (sSrcPath, sDstFile, oXcpt));
|
---|
180 | return False;
|
---|
181 |
|
---|
182 | return True;
|
---|
183 |
|
---|
184 |
|
---|
185 |
|
---|
186 | #
|
---|
187 | # Unit testing.
|
---|
188 | #
|
---|
189 |
|
---|
190 | # pylint: disable=C0111
|
---|
191 | class CommonUtilsTestCase(unittest.TestCase):
|
---|
192 | def testHasSchema(self):
|
---|
193 | self.assertTrue(hasSchema('http://www.oracle.com/'));
|
---|
194 | self.assertTrue(hasSchema('https://virtualbox.com/'));
|
---|
195 | self.assertFalse(hasSchema('://virtualbox.com/'));
|
---|
196 | self.assertFalse(hasSchema('/usr/bin'));
|
---|
197 | self.assertFalse(hasSchema('usr/bin'));
|
---|
198 | self.assertFalse(hasSchema('bin'));
|
---|
199 | self.assertFalse(hasSchema('C:\\WINNT'));
|
---|
200 |
|
---|
201 | if __name__ == '__main__':
|
---|
202 | unittest.main();
|
---|
203 | # not reached.
|
---|
204 |
|
---|