selenium-core-runner 0.0.3
Sign up to get free protection for your applications and to get access to all the features.
- data/Gemfile +9 -0
- data/MIT-LICENSE +20 -0
- data/README.rdoc +3 -0
- data/Rakefile +30 -0
- data/app/controllers/selenium_core_runner/suites_controller.rb +19 -0
- data/app/views/selenium_core_runner/suites/index.html.erb +177 -0
- data/app/views/selenium_core_runner/suites/show.html.erb +0 -0
- data/config/routes.rb +6 -0
- data/lib/selenium-core-runner/engine.rb +19 -0
- data/lib/selenium-core-runner.rb +3 -0
- data/public/selenium-core-runner/Blank.html +7 -0
- data/public/selenium-core-runner/InjectedRemoteRunner.html +8 -0
- data/public/selenium-core-runner/RemoteRunner.html +101 -0
- data/public/selenium-core-runner/SeleniumLog.html +109 -0
- data/public/selenium-core-runner/TestPrompt.html +145 -0
- data/public/selenium-core-runner/TestRunner-splash.html +55 -0
- data/public/selenium-core-runner/TestRunner.hta +177 -0
- data/public/selenium-core-runner/TestRunner.html +177 -0
- data/public/selenium-core-runner/icons/all.png +0 -0
- data/public/selenium-core-runner/icons/continue.png +0 -0
- data/public/selenium-core-runner/icons/continue_disabled.png +0 -0
- data/public/selenium-core-runner/icons/pause.png +0 -0
- data/public/selenium-core-runner/icons/pause_disabled.png +0 -0
- data/public/selenium-core-runner/icons/selected.png +0 -0
- data/public/selenium-core-runner/icons/step.png +0 -0
- data/public/selenium-core-runner/icons/step_disabled.png +0 -0
- data/public/selenium-core-runner/iedoc-core.xml +1789 -0
- data/public/selenium-core-runner/iedoc.xml +1830 -0
- data/public/selenium-core-runner/lib/cssQuery/cssQuery-p.js +6 -0
- data/public/selenium-core-runner/lib/cssQuery/src/cssQuery-level2.js +142 -0
- data/public/selenium-core-runner/lib/cssQuery/src/cssQuery-level3.js +150 -0
- data/public/selenium-core-runner/lib/cssQuery/src/cssQuery-standard.js +53 -0
- data/public/selenium-core-runner/lib/cssQuery/src/cssQuery.js +356 -0
- data/public/selenium-core-runner/lib/prototype.js +2006 -0
- data/public/selenium-core-runner/lib/scriptaculous/builder.js +101 -0
- data/public/selenium-core-runner/lib/scriptaculous/controls.js +815 -0
- data/public/selenium-core-runner/lib/scriptaculous/dragdrop.js +915 -0
- data/public/selenium-core-runner/lib/scriptaculous/effects.js +958 -0
- data/public/selenium-core-runner/lib/scriptaculous/scriptaculous.js +47 -0
- data/public/selenium-core-runner/lib/scriptaculous/slider.js +283 -0
- data/public/selenium-core-runner/lib/scriptaculous/unittest.js +383 -0
- data/public/selenium-core-runner/lib/snapsie.js +91 -0
- data/public/selenium-core-runner/scripts/find_matching_child.js +69 -0
- data/public/selenium-core-runner/scripts/htmlutils.js +1623 -0
- data/public/selenium-core-runner/scripts/injection.html +72 -0
- data/public/selenium-core-runner/scripts/selenium-api.js +3240 -0
- data/public/selenium-core-runner/scripts/selenium-browserbot.js +2333 -0
- data/public/selenium-core-runner/scripts/selenium-browserdetect.js +153 -0
- data/public/selenium-core-runner/scripts/selenium-commandhandlers.js +379 -0
- data/public/selenium-core-runner/scripts/selenium-executionloop.js +175 -0
- data/public/selenium-core-runner/scripts/selenium-logging.js +148 -0
- data/public/selenium-core-runner/scripts/selenium-remoterunner.js +695 -0
- data/public/selenium-core-runner/scripts/selenium-testrunner.js +1362 -0
- data/public/selenium-core-runner/scripts/selenium-version.js +5 -0
- data/public/selenium-core-runner/scripts/ui-doc.html +803 -0
- data/public/selenium-core-runner/scripts/ui-element.js +1627 -0
- data/public/selenium-core-runner/scripts/ui-map-sample.js +979 -0
- data/public/selenium-core-runner/scripts/user-extensions.js +3 -0
- data/public/selenium-core-runner/scripts/user-extensions.js.sample +75 -0
- data/public/selenium-core-runner/scripts/xmlextras.js +153 -0
- data/public/selenium-core-runner/selenium-logo.png +0 -0
- data/public/selenium-core-runner/selenium-test.css +43 -0
- data/public/selenium-core-runner/selenium.css +316 -0
- data/public/selenium-core-runner/xpath/dom.js +566 -0
- data/public/selenium-core-runner/xpath/javascript-xpath-0.1.11.js +2816 -0
- data/public/selenium-core-runner/xpath/util.js +549 -0
- data/public/selenium-core-runner/xpath/xmltoken.js +149 -0
- data/public/selenium-core-runner/xpath/xpath.js +2481 -0
- metadata +121 -0
@@ -0,0 +1,695 @@
|
|
1
|
+
/*
|
2
|
+
* Copyright 2005 ThoughtWorks, Inc
|
3
|
+
*
|
4
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
* you may not use this file except in compliance with the License.
|
6
|
+
* You may obtain a copy of the License at
|
7
|
+
*
|
8
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
*
|
10
|
+
* Unless required by applicable law or agreed to in writing, software
|
11
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
* See the License for the specific language governing permissions and
|
14
|
+
* limitations under the License.
|
15
|
+
*
|
16
|
+
*/
|
17
|
+
|
18
|
+
passColor = "#cfffcf";
|
19
|
+
failColor = "#ffcfcf";
|
20
|
+
errorColor = "#ffffff";
|
21
|
+
workingColor = "#DEE7EC";
|
22
|
+
doneColor = "#FFFFCC";
|
23
|
+
|
24
|
+
var injectedSessionId;
|
25
|
+
|
26
|
+
var postResult = "START";
|
27
|
+
var debugMode = false;
|
28
|
+
var relayToRC = null;
|
29
|
+
var proxyInjectionMode = false;
|
30
|
+
var uniqueId = 'sel_' + Math.round(100000 * Math.random());
|
31
|
+
var seleniumSequenceNumber = 0;
|
32
|
+
var cmd8 = "";
|
33
|
+
var cmd7 = "";
|
34
|
+
var cmd6 = "";
|
35
|
+
var cmd5 = "";
|
36
|
+
var cmd4 = "";
|
37
|
+
var cmd3 = "";
|
38
|
+
var cmd2 = "";
|
39
|
+
var cmd1 = "";
|
40
|
+
var lastCmd = "";
|
41
|
+
var lastCmdTime = new Date();
|
42
|
+
|
43
|
+
var RemoteRunnerOptions = classCreate();
|
44
|
+
objectExtend(RemoteRunnerOptions.prototype, URLConfiguration.prototype);
|
45
|
+
objectExtend(RemoteRunnerOptions.prototype, {
|
46
|
+
initialize: function() {
|
47
|
+
this._acquireQueryString();
|
48
|
+
},
|
49
|
+
isDebugMode: function() {
|
50
|
+
return this._isQueryParameterTrue("debugMode");
|
51
|
+
},
|
52
|
+
|
53
|
+
getContinue: function() {
|
54
|
+
return this._getQueryParameter("continue");
|
55
|
+
},
|
56
|
+
|
57
|
+
getDriverUrl: function() {
|
58
|
+
return this._getQueryParameter("driverUrl");
|
59
|
+
},
|
60
|
+
|
61
|
+
// requires per-session extension Javascript as soon as this Selenium
|
62
|
+
// instance becomes aware of the session identifier
|
63
|
+
getSessionId: function() {
|
64
|
+
var sessionId = this._getQueryParameter("sessionId");
|
65
|
+
requireExtensionJs(sessionId);
|
66
|
+
return sessionId;
|
67
|
+
},
|
68
|
+
|
69
|
+
_acquireQueryString: function () {
|
70
|
+
if (this.queryString) return;
|
71
|
+
if (browserVersion.isHTA) {
|
72
|
+
var args = this._extractArgs();
|
73
|
+
if (args.length < 2) return null;
|
74
|
+
this.queryString = args[1];
|
75
|
+
} else if (proxyInjectionMode) {
|
76
|
+
this.queryString = window.location.search.substr(1);
|
77
|
+
} else {
|
78
|
+
this.queryString = top.location.search.substr(1);
|
79
|
+
}
|
80
|
+
}
|
81
|
+
|
82
|
+
});
|
83
|
+
var runOptions;
|
84
|
+
|
85
|
+
function runSeleniumTest() {
|
86
|
+
runOptions = new RemoteRunnerOptions();
|
87
|
+
var testAppWindow;
|
88
|
+
|
89
|
+
if (runOptions.isMultiWindowMode()) {
|
90
|
+
testAppWindow = openSeparateApplicationWindow('Blank.html', true);
|
91
|
+
} else if (sel$('selenium_myiframe') != null) {
|
92
|
+
var myiframe = sel$('selenium_myiframe');
|
93
|
+
if (myiframe) {
|
94
|
+
testAppWindow = myiframe.contentWindow;
|
95
|
+
}
|
96
|
+
}
|
97
|
+
else {
|
98
|
+
proxyInjectionMode = true;
|
99
|
+
testAppWindow = window;
|
100
|
+
}
|
101
|
+
selenium = Selenium.createForWindow(testAppWindow, proxyInjectionMode);
|
102
|
+
if (runOptions.getBaseUrl()) {
|
103
|
+
selenium.browserbot.baseUrl = runOptions.getBaseUrl();
|
104
|
+
}
|
105
|
+
if (!debugMode) {
|
106
|
+
debugMode = runOptions.isDebugMode();
|
107
|
+
}
|
108
|
+
if (proxyInjectionMode) {
|
109
|
+
LOG.logHook = logToRc;
|
110
|
+
selenium.browserbot._modifyWindow(testAppWindow);
|
111
|
+
}
|
112
|
+
else if (debugMode) {
|
113
|
+
LOG.logHook = logToRc;
|
114
|
+
}
|
115
|
+
window.selenium = selenium;
|
116
|
+
|
117
|
+
commandFactory = new CommandHandlerFactory();
|
118
|
+
commandFactory.registerAll(selenium);
|
119
|
+
|
120
|
+
currentTest = new RemoteRunner(commandFactory);
|
121
|
+
|
122
|
+
var doContinue = runOptions.getContinue();
|
123
|
+
if (doContinue != null) postResult = "OK";
|
124
|
+
|
125
|
+
currentTest.start();
|
126
|
+
}
|
127
|
+
|
128
|
+
function buildDriverUrl() {
|
129
|
+
var driverUrl = runOptions.getDriverUrl();
|
130
|
+
if (driverUrl != null) {
|
131
|
+
return driverUrl;
|
132
|
+
}
|
133
|
+
var s = window.location.href
|
134
|
+
var slashPairOffset = s.indexOf("//") + "//".length
|
135
|
+
var pathSlashOffset = s.substring(slashPairOffset).indexOf("/")
|
136
|
+
return s.substring(0, slashPairOffset + pathSlashOffset) + "/selenium-server/driver/";
|
137
|
+
//return "http://localhost" + uniqueId + "/selenium-server/driver/";
|
138
|
+
}
|
139
|
+
|
140
|
+
function logToRc(logLevel, message) {
|
141
|
+
if (debugMode) {
|
142
|
+
if (logLevel == null) {
|
143
|
+
logLevel = "debug";
|
144
|
+
}
|
145
|
+
sendToRCAndForget("logLevel=" + logLevel + ":" + message.replace(/[\n\r\015]/g, " ") + "\n", "logging=true");
|
146
|
+
}
|
147
|
+
}
|
148
|
+
|
149
|
+
function serializeString(name, s) {
|
150
|
+
return name + "=unescape(\"" + escape(s) + "\");";
|
151
|
+
}
|
152
|
+
|
153
|
+
function serializeObject(name, x)
|
154
|
+
{
|
155
|
+
var s = '';
|
156
|
+
|
157
|
+
if (isArray(x))
|
158
|
+
{
|
159
|
+
s = name + "=new Array(); ";
|
160
|
+
var len = x["length"];
|
161
|
+
for (var j = 0; j < len; j++)
|
162
|
+
{
|
163
|
+
s += serializeString(name + "[" + j + "]", x[j]);
|
164
|
+
}
|
165
|
+
}
|
166
|
+
else if (typeof x == "string")
|
167
|
+
{
|
168
|
+
s = serializeString(name, x);
|
169
|
+
}
|
170
|
+
else
|
171
|
+
{
|
172
|
+
throw "unrecognized object not encoded: " + name + "(" + x + ")";
|
173
|
+
}
|
174
|
+
return s;
|
175
|
+
}
|
176
|
+
|
177
|
+
function relayBotToRC(s) {
|
178
|
+
}
|
179
|
+
|
180
|
+
// seems like no one uses this, but in fact it is called using eval from server-side PI mode code; however,
|
181
|
+
// because multiple names can map to the same popup, assigning a single name confuses matters sometimes;
|
182
|
+
// thus, I'm disabling this for now. -Nelson 10/21/06
|
183
|
+
function setSeleniumWindowName(seleniumWindowName) {
|
184
|
+
//selenium.browserbot.getCurrentWindow()['seleniumWindowName'] = seleniumWindowName;
|
185
|
+
}
|
186
|
+
|
187
|
+
RemoteRunner = classCreate();
|
188
|
+
objectExtend(RemoteRunner.prototype, new TestLoop());
|
189
|
+
objectExtend(RemoteRunner.prototype, {
|
190
|
+
initialize : function(commandFactory) {
|
191
|
+
this.commandFactory = commandFactory;
|
192
|
+
this.requiresCallBack = true;
|
193
|
+
this.commandNode = null;
|
194
|
+
this.xmlHttpForCommandsAndResults = null;
|
195
|
+
},
|
196
|
+
|
197
|
+
nextCommand : function() {
|
198
|
+
var urlParms = "";
|
199
|
+
if (postResult == "START") {
|
200
|
+
urlParms += "seleniumStart=true";
|
201
|
+
}
|
202
|
+
this.xmlHttpForCommandsAndResults = XmlHttp.create();
|
203
|
+
sendToRC(postResult, urlParms, fnBind(this._HandleHttpResponse, this), this.xmlHttpForCommandsAndResults);
|
204
|
+
},
|
205
|
+
|
206
|
+
commandStarted : function(command) {
|
207
|
+
this.commandNode = document.createElement("div");
|
208
|
+
var cmdText = command.command + '(';
|
209
|
+
if (command.target != null && command.target != "") {
|
210
|
+
cmdText += command.target;
|
211
|
+
if (command.value != null && command.value != "") {
|
212
|
+
cmdText += ', ' + command.value;
|
213
|
+
}
|
214
|
+
}
|
215
|
+
if (cmdText.length > 70) {
|
216
|
+
cmdText = cmdText.substring(0, 70) + "...\n";
|
217
|
+
} else {
|
218
|
+
cmdText += ")\n";
|
219
|
+
}
|
220
|
+
|
221
|
+
if (cmdText == lastCmd) {
|
222
|
+
var rightNow = new Date();
|
223
|
+
var msSinceStart = rightNow.getTime() - lastCmdTime.getTime();
|
224
|
+
var sinceStart = msSinceStart + "ms";
|
225
|
+
if (msSinceStart > 1000) {
|
226
|
+
sinceStart = Math.round(msSinceStart / 1000) + "s";
|
227
|
+
}
|
228
|
+
cmd1 = "Same command (" + sinceStart + "): " + lastCmd;
|
229
|
+
} else {
|
230
|
+
lastCmdTime = new Date();
|
231
|
+
cmd8 = cmd7;
|
232
|
+
cmd7 = cmd6;
|
233
|
+
cmd6 = cmd5;
|
234
|
+
cmd5 = cmd4;
|
235
|
+
cmd4 = cmd3;
|
236
|
+
cmd3 = cmd2;
|
237
|
+
cmd2 = cmd1;
|
238
|
+
cmd1 = cmdText;
|
239
|
+
}
|
240
|
+
lastCmd = cmdText;
|
241
|
+
|
242
|
+
if (! proxyInjectionMode) {
|
243
|
+
var commandList = document.commands.commandList;
|
244
|
+
commandList.value = cmd8 + cmd7 + cmd6 + cmd5 + cmd4 + cmd3 + cmd2 + cmd1;
|
245
|
+
commandList.scrollTop = commandList.scrollHeight;
|
246
|
+
}
|
247
|
+
},
|
248
|
+
|
249
|
+
commandComplete : function(result) {
|
250
|
+
|
251
|
+
if (result.failed) {
|
252
|
+
if (postResult == "CONTINUATION") {
|
253
|
+
currentTest.aborted = true;
|
254
|
+
}
|
255
|
+
postResult = result.failureMessage;
|
256
|
+
this.commandNode.title = result.failureMessage;
|
257
|
+
this.commandNode.style.backgroundColor = failColor;
|
258
|
+
} else if (result.passed) {
|
259
|
+
postResult = "OK";
|
260
|
+
this.commandNode.style.backgroundColor = passColor;
|
261
|
+
} else {
|
262
|
+
if (result.result == null) {
|
263
|
+
postResult = "OK";
|
264
|
+
} else {
|
265
|
+
var actualResult = result.result;
|
266
|
+
actualResult = selArrayToString(actualResult);
|
267
|
+
postResult = "OK," + actualResult;
|
268
|
+
}
|
269
|
+
this.commandNode.style.backgroundColor = doneColor;
|
270
|
+
}
|
271
|
+
},
|
272
|
+
|
273
|
+
commandError : function(message) {
|
274
|
+
postResult = "ERROR: " + message;
|
275
|
+
this.commandNode.style.backgroundColor = errorColor;
|
276
|
+
this.commandNode.titcle = message;
|
277
|
+
},
|
278
|
+
|
279
|
+
testComplete : function() {
|
280
|
+
window.status = "Selenium Tests Complete, for this Test"
|
281
|
+
// Continue checking for new results
|
282
|
+
this.continueTest();
|
283
|
+
postResult = "START";
|
284
|
+
},
|
285
|
+
|
286
|
+
_HandleHttpResponse : function() {
|
287
|
+
// When request is completed
|
288
|
+
if (this.xmlHttpForCommandsAndResults.readyState == 4) {
|
289
|
+
// OK
|
290
|
+
if (this.xmlHttpForCommandsAndResults.status == 200) {
|
291
|
+
if (this.xmlHttpForCommandsAndResults.responseText=="") {
|
292
|
+
LOG.error("saw blank string xmlHttpForCommandsAndResults.responseText");
|
293
|
+
return;
|
294
|
+
}
|
295
|
+
var command = this._extractCommand(this.xmlHttpForCommandsAndResults);
|
296
|
+
if (command.command == 'retryLast') {
|
297
|
+
setTimeout(fnBind(function() {
|
298
|
+
sendToRC("RETRY", "retry=true", fnBind(this._HandleHttpResponse, this), this.xmlHttpForCommandsAndResults, true);
|
299
|
+
}, this), 1000);
|
300
|
+
} else {
|
301
|
+
this.currentCommand = command;
|
302
|
+
this.continueTestAtCurrentCommand();
|
303
|
+
}
|
304
|
+
}
|
305
|
+
// Not OK
|
306
|
+
else {
|
307
|
+
var s = 'xmlHttp returned: ' + this.xmlHttpForCommandsAndResults.status + ": " + this.xmlHttpForCommandsAndResults.statusText;
|
308
|
+
LOG.error(s);
|
309
|
+
this.currentCommand = null;
|
310
|
+
setTimeout(fnBind(this.continueTestAtCurrentCommand, this), 2000);
|
311
|
+
}
|
312
|
+
|
313
|
+
}
|
314
|
+
},
|
315
|
+
|
316
|
+
_extractCommand : function(xmlHttp) {
|
317
|
+
var command, text, json;
|
318
|
+
text = command = xmlHttp.responseText;
|
319
|
+
if (/^json=/.test(text)) {
|
320
|
+
eval(text);
|
321
|
+
if (json.rest) {
|
322
|
+
eval(json.rest);
|
323
|
+
}
|
324
|
+
return json;
|
325
|
+
}
|
326
|
+
try {
|
327
|
+
var re = new RegExp("^(.*?)\n((.|[\r\n])*)");
|
328
|
+
if (re.exec(xmlHttp.responseText)) {
|
329
|
+
command = RegExp.$1;
|
330
|
+
var rest = RegExp.$2;
|
331
|
+
rest = rest.trim();
|
332
|
+
if (rest) {
|
333
|
+
eval(rest);
|
334
|
+
}
|
335
|
+
}
|
336
|
+
else {
|
337
|
+
command = xmlHttp.responseText;
|
338
|
+
}
|
339
|
+
} catch (e) {
|
340
|
+
alert('could not get responseText: ' + e.message);
|
341
|
+
}
|
342
|
+
if (command.substr(0, '|testComplete'.length) == '|testComplete') {
|
343
|
+
return null;
|
344
|
+
}
|
345
|
+
|
346
|
+
return this._createCommandFromRequest(command);
|
347
|
+
},
|
348
|
+
|
349
|
+
|
350
|
+
_delay : function(millis) {
|
351
|
+
var startMillis = new Date();
|
352
|
+
while (true) {
|
353
|
+
milli = new Date();
|
354
|
+
if (milli - startMillis > millis) {
|
355
|
+
break;
|
356
|
+
}
|
357
|
+
}
|
358
|
+
},
|
359
|
+
|
360
|
+
// Parses a URI query string into a SeleniumCommand object
|
361
|
+
_createCommandFromRequest : function(commandRequest) {
|
362
|
+
//decodeURIComponent doesn't strip plus signs
|
363
|
+
var processed = commandRequest.replace(/\+/g, "%20");
|
364
|
+
// strip trailing spaces
|
365
|
+
var processed = processed.replace(/\s+$/, "");
|
366
|
+
var vars = processed.split("&");
|
367
|
+
var cmdArgs = new Object();
|
368
|
+
for (var i = 0; i < vars.length; i++) {
|
369
|
+
var pair = vars[i].split("=");
|
370
|
+
cmdArgs[pair[0]] = pair[1];
|
371
|
+
}
|
372
|
+
var cmd = cmdArgs['cmd'];
|
373
|
+
var arg1 = cmdArgs['1'];
|
374
|
+
if (null == arg1) arg1 = "";
|
375
|
+
arg1 = decodeURIComponent(arg1);
|
376
|
+
var arg2 = cmdArgs['2'];
|
377
|
+
if (null == arg2) arg2 = "";
|
378
|
+
arg2 = decodeURIComponent(arg2);
|
379
|
+
if (cmd == null) {
|
380
|
+
throw new Error("Bad command request: " + commandRequest);
|
381
|
+
}
|
382
|
+
return new SeleniumCommand(cmd, arg1, arg2);
|
383
|
+
}
|
384
|
+
|
385
|
+
})
|
386
|
+
|
387
|
+
|
388
|
+
function sendToRC(dataToBePosted, urlParms, callback, xmlHttpObject, async) {
|
389
|
+
if (async == null) {
|
390
|
+
async = true;
|
391
|
+
}
|
392
|
+
if (xmlHttpObject == null) {
|
393
|
+
xmlHttpObject = XmlHttp.create();
|
394
|
+
}
|
395
|
+
var url = buildDriverUrl() + "?"
|
396
|
+
if (urlParms) {
|
397
|
+
url += urlParms;
|
398
|
+
}
|
399
|
+
url = addUrlParams(url);
|
400
|
+
url += "&sequenceNumber=" + seleniumSequenceNumber++;
|
401
|
+
|
402
|
+
var postedData = "postedData=" + encodeURIComponent(dataToBePosted);
|
403
|
+
|
404
|
+
//xmlHttpObject.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
|
405
|
+
xmlHttpObject.open("POST", url, async);
|
406
|
+
if (callback) xmlHttpObject.onreadystatechange = callback;
|
407
|
+
xmlHttpObject.send(postedData);
|
408
|
+
return null;
|
409
|
+
}
|
410
|
+
|
411
|
+
function addUrlParams(url) {
|
412
|
+
return url + "&localFrameAddress=" + (proxyInjectionMode ? makeAddressToAUTFrame() : "top")
|
413
|
+
+ getSeleniumWindowNameURLparameters()
|
414
|
+
+ "&uniqueId=" + uniqueId
|
415
|
+
+ buildDriverParams() + preventBrowserCaching()
|
416
|
+
}
|
417
|
+
|
418
|
+
function sendToRCAndForget(dataToBePosted, urlParams) {
|
419
|
+
var url;
|
420
|
+
if (!(browserVersion.isChrome || browserVersion.isHTA)) {
|
421
|
+
// DGF we're behind a proxy, so we can send our logging message to literally any host, to avoid 2-connection limit
|
422
|
+
var protocol = "http:";
|
423
|
+
if (window.location.protocol == "https:") {
|
424
|
+
// DGF if we're in HTTPS, use another HTTPS url to avoid security warning
|
425
|
+
protocol = "https:";
|
426
|
+
}
|
427
|
+
// we don't choose a super large random value, but rather 1 - 16, because this matches with the pre-computed
|
428
|
+
// tunnels waiting on the Selenium Server side. This gives us higher throughput than the two-connection-per-host
|
429
|
+
// limitation, but doesn't require we generate an extremely large ammount of fake SSL certs either.
|
430
|
+
url = protocol + "//" + Math.floor(Math.random()* 16 + 1) + ".selenium.doesnotexist/selenium-server/driver/?" + urlParams;
|
431
|
+
} else {
|
432
|
+
url = buildDriverUrl() + "?" + urlParams;
|
433
|
+
}
|
434
|
+
url = addUrlParams(url);
|
435
|
+
|
436
|
+
var method = "GET";
|
437
|
+
if (method == "POST") {
|
438
|
+
// DGF submit a request using an iframe; we can't see the response, but we don't need to
|
439
|
+
// TODO not using this mechanism because it screws up back-button
|
440
|
+
var loggingForm = document.createElement("form");
|
441
|
+
loggingForm.method = "POST";
|
442
|
+
loggingForm.action = url;
|
443
|
+
loggingForm.target = "seleniumLoggingFrame";
|
444
|
+
var postedDataInput = document.createElement("input");
|
445
|
+
postedDataInput.type = "hidden";
|
446
|
+
postedDataInput.name = "postedData";
|
447
|
+
postedDataInput.value = dataToBePosted;
|
448
|
+
loggingForm.appendChild(postedDataInput);
|
449
|
+
document.body.appendChild(loggingForm);
|
450
|
+
loggingForm.submit();
|
451
|
+
document.body.removeChild(loggingForm);
|
452
|
+
} else {
|
453
|
+
var postedData = "&postedData=" + encodeURIComponent(dataToBePosted);
|
454
|
+
var scriptTag = document.createElement("script");
|
455
|
+
scriptTag.src = url + postedData;
|
456
|
+
document.body.appendChild(scriptTag);
|
457
|
+
document.body.removeChild(scriptTag);
|
458
|
+
}
|
459
|
+
}
|
460
|
+
|
461
|
+
function buildDriverParams() {
|
462
|
+
var params = "";
|
463
|
+
|
464
|
+
var sessionId = runOptions.getSessionId();
|
465
|
+
if (sessionId == undefined) {
|
466
|
+
sessionId = injectedSessionId;
|
467
|
+
}
|
468
|
+
if (sessionId != undefined) {
|
469
|
+
params = params + "&sessionId=" + sessionId;
|
470
|
+
}
|
471
|
+
return params;
|
472
|
+
}
|
473
|
+
|
474
|
+
function preventBrowserCaching() {
|
475
|
+
var t = (new Date()).getTime();
|
476
|
+
return "&counterToMakeURsUniqueAndSoStopPageCachingInTheBrowser=" + t;
|
477
|
+
}
|
478
|
+
|
479
|
+
//
|
480
|
+
// Return URL parameters pertaining to the name(s?) of the current window
|
481
|
+
//
|
482
|
+
// In selenium, the main (i.e., first) window's name is a blank string.
|
483
|
+
//
|
484
|
+
// Additional pop-ups are associated with either 1.) the name given by the 2nd parameter to window.open, or 2.) the name of a
|
485
|
+
// property on the opening window which points at the window.
|
486
|
+
//
|
487
|
+
// An example of #2: if window X contains JavaScript as follows:
|
488
|
+
//
|
489
|
+
// var windowABC = window.open(...)
|
490
|
+
//
|
491
|
+
// Note that the example JavaScript above is equivalent to
|
492
|
+
//
|
493
|
+
// window["windowABC"] = window.open(...)
|
494
|
+
//
|
495
|
+
function getSeleniumWindowNameURLparameters() {
|
496
|
+
var w = (proxyInjectionMode ? selenium.browserbot.getCurrentWindow() : window).top;
|
497
|
+
var s = "&seleniumWindowName=";
|
498
|
+
if (w.opener == null) {
|
499
|
+
return s;
|
500
|
+
}
|
501
|
+
if (w["seleniumWindowName"] == null) {
|
502
|
+
if (w.name) {
|
503
|
+
w["seleniumWindowName"] = w.name;
|
504
|
+
} else {
|
505
|
+
w["seleniumWindowName"] = 'generatedSeleniumWindowName_' + Math.round(100000 * Math.random());
|
506
|
+
}
|
507
|
+
}
|
508
|
+
s += w["seleniumWindowName"];
|
509
|
+
var windowOpener = w.opener;
|
510
|
+
for (key in windowOpener) {
|
511
|
+
var val = null;
|
512
|
+
try {
|
513
|
+
val = windowOpener[key];
|
514
|
+
}
|
515
|
+
catch(e) {
|
516
|
+
}
|
517
|
+
if (val==w) {
|
518
|
+
s += "&jsWindowNameVar=" + key; // found a js variable in the opener referring to this window
|
519
|
+
}
|
520
|
+
}
|
521
|
+
return s;
|
522
|
+
}
|
523
|
+
|
524
|
+
// construct a JavaScript expression which leads to my frame (i.e., the frame containing the window
|
525
|
+
// in which this code is operating)
|
526
|
+
function makeAddressToAUTFrame(w, frameNavigationalJSexpression)
|
527
|
+
{
|
528
|
+
if (w == null)
|
529
|
+
{
|
530
|
+
w = top;
|
531
|
+
frameNavigationalJSexpression = "top";
|
532
|
+
}
|
533
|
+
|
534
|
+
if (w == selenium.browserbot.getCurrentWindow())
|
535
|
+
{
|
536
|
+
return frameNavigationalJSexpression;
|
537
|
+
}
|
538
|
+
for (var j = 0; j < w.frames.length; j++)
|
539
|
+
{
|
540
|
+
var t = makeAddressToAUTFrame(w.frames[j], frameNavigationalJSexpression + ".frames[" + j + "]");
|
541
|
+
if (t != null)
|
542
|
+
{
|
543
|
+
return t;
|
544
|
+
}
|
545
|
+
}
|
546
|
+
return null;
|
547
|
+
}
|
548
|
+
|
549
|
+
Selenium.prototype.doSetContext = function(context) {
|
550
|
+
/**
|
551
|
+
* Writes a message to the status bar and adds a note to the browser-side
|
552
|
+
* log.
|
553
|
+
*
|
554
|
+
* @param context
|
555
|
+
* the message to be sent to the browser
|
556
|
+
*/
|
557
|
+
//set the current test title
|
558
|
+
var ctx = document.getElementById("context");
|
559
|
+
if (ctx != null) {
|
560
|
+
ctx.innerHTML = context;
|
561
|
+
}
|
562
|
+
};
|
563
|
+
|
564
|
+
/**
|
565
|
+
* Adds a script tag referencing a specially-named user extensions "file". The
|
566
|
+
* resource handler for this special file (which won't actually exist) will use
|
567
|
+
* the session ID embedded in its name to retrieve per-session specified user
|
568
|
+
* extension javascript.
|
569
|
+
*
|
570
|
+
* @param sessionId
|
571
|
+
*/
|
572
|
+
function requireExtensionJs(sessionId) {
|
573
|
+
var src = 'scripts/user-extensions.js[' + sessionId + ']';
|
574
|
+
if (document.getElementById(src) == null) {
|
575
|
+
var scriptTag = document.createElement('script');
|
576
|
+
scriptTag.language = 'JavaScript';
|
577
|
+
scriptTag.type = 'text/javascript';
|
578
|
+
scriptTag.src = src;
|
579
|
+
scriptTag.id = src;
|
580
|
+
var headTag = document.getElementsByTagName('head')[0];
|
581
|
+
headTag.appendChild(scriptTag);
|
582
|
+
}
|
583
|
+
}
|
584
|
+
|
585
|
+
Selenium.prototype.doAttachFile = function(fieldLocator,fileLocator) {
|
586
|
+
/**
|
587
|
+
* Sets a file input (upload) field to the file listed in fileLocator
|
588
|
+
*
|
589
|
+
* @param fieldLocator an <a href="#locators">element locator</a>
|
590
|
+
* @param fileLocator a URL pointing to the specified file. Before the file
|
591
|
+
* can be set in the input field (fieldLocator), Selenium RC may need to transfer the file
|
592
|
+
* to the local machine before attaching the file in a web page form. This is common in selenium
|
593
|
+
* grid configurations where the RC server driving the browser is not the same
|
594
|
+
* machine that started the test.
|
595
|
+
*
|
596
|
+
* Supported Browsers: Firefox ("*chrome") only.
|
597
|
+
*
|
598
|
+
*/
|
599
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
600
|
+
};
|
601
|
+
|
602
|
+
Selenium.prototype.doCaptureScreenshot = function(filename) {
|
603
|
+
/**
|
604
|
+
* Captures a PNG screenshot to the specified file.
|
605
|
+
*
|
606
|
+
* @param filename the absolute path to the file to be written, e.g. "c:\blah\screenshot.png"
|
607
|
+
*/
|
608
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
609
|
+
};
|
610
|
+
|
611
|
+
Selenium.prototype.doCaptureScreenshotToString = function() {
|
612
|
+
/**
|
613
|
+
* Capture a PNG screenshot. It then returns the file as a base 64 encoded string.
|
614
|
+
*
|
615
|
+
* @return string The base 64 encoded string of the screen shot (PNG file)
|
616
|
+
*/
|
617
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
618
|
+
};
|
619
|
+
|
620
|
+
Selenium.prototype.doCaptureEntirePageScreenshotToString = function(kwargs) {
|
621
|
+
/**
|
622
|
+
* Downloads a screenshot of the browser current window canvas to a
|
623
|
+
* based 64 encoded PNG file. The <em>entire</em> windows canvas is captured,
|
624
|
+
* including parts rendered outside of the current view port.
|
625
|
+
*
|
626
|
+
* Currently this only works in Mozilla and when running in chrome mode.
|
627
|
+
*
|
628
|
+
* @param kwargs A kwargs string that modifies the way the screenshot is captured. Example: "background=#CCFFDD". This may be useful to set for capturing screenshots of less-than-ideal layouts, for example where absolute positioning causes the calculation of the canvas dimension to fail and a black background is exposed (possibly obscuring black text).
|
629
|
+
*
|
630
|
+
* @return string The base 64 encoded string of the page screenshot (PNG file)
|
631
|
+
*/
|
632
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
633
|
+
};
|
634
|
+
|
635
|
+
Selenium.prototype.doShutDownSeleniumServer = function(keycode) {
|
636
|
+
/**
|
637
|
+
* Kills the running Selenium Server and all browser sessions. After you run this command, you will no longer be able to send
|
638
|
+
* commands to the server; you can't remotely start the server once it has been stopped. Normally
|
639
|
+
* you should prefer to run the "stop" command, which terminates the current browser session, rather than
|
640
|
+
* shutting down the entire server.
|
641
|
+
*
|
642
|
+
*/
|
643
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
644
|
+
};
|
645
|
+
|
646
|
+
Selenium.prototype.doRetrieveLastRemoteControlLogs = function() {
|
647
|
+
/**
|
648
|
+
* Retrieve the last messages logged on a specific remote control. Useful for error reports, especially
|
649
|
+
* when running multiple remote controls in a distributed environment. The maximum number of log messages
|
650
|
+
* that can be retrieve is configured on remote control startup.
|
651
|
+
*
|
652
|
+
* @return string The last N log messages as a multi-line string.
|
653
|
+
*/
|
654
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
655
|
+
};
|
656
|
+
|
657
|
+
Selenium.prototype.doKeyDownNative = function(keycode) {
|
658
|
+
/**
|
659
|
+
* Simulates a user pressing a key (without releasing it yet) by sending a native operating system keystroke.
|
660
|
+
* This function uses the java.awt.Robot class to send a keystroke; this more accurately simulates typing
|
661
|
+
* a key on the keyboard. It does not honor settings from the shiftKeyDown, controlKeyDown, altKeyDown and
|
662
|
+
* metaKeyDown commands, and does not target any particular HTML element. To send a keystroke to a particular
|
663
|
+
* element, focus on the element first before running this command.
|
664
|
+
*
|
665
|
+
* @param keycode an integer keycode number corresponding to a java.awt.event.KeyEvent; note that Java keycodes are NOT the same thing as JavaScript keycodes!
|
666
|
+
*/
|
667
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
668
|
+
};
|
669
|
+
|
670
|
+
Selenium.prototype.doKeyUpNative = function(keycode) {
|
671
|
+
/**
|
672
|
+
* Simulates a user releasing a key by sending a native operating system keystroke.
|
673
|
+
* This function uses the java.awt.Robot class to send a keystroke; this more accurately simulates typing
|
674
|
+
* a key on the keyboard. It does not honor settings from the shiftKeyDown, controlKeyDown, altKeyDown and
|
675
|
+
* metaKeyDown commands, and does not target any particular HTML element. To send a keystroke to a particular
|
676
|
+
* element, focus on the element first before running this command.
|
677
|
+
*
|
678
|
+
* @param keycode an integer keycode number corresponding to a java.awt.event.KeyEvent; note that Java keycodes are NOT the same thing as JavaScript keycodes!
|
679
|
+
*/
|
680
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
681
|
+
};
|
682
|
+
|
683
|
+
Selenium.prototype.doKeyPressNative = function(keycode) {
|
684
|
+
/**
|
685
|
+
* Simulates a user pressing and releasing a key by sending a native operating system keystroke.
|
686
|
+
* This function uses the java.awt.Robot class to send a keystroke; this more accurately simulates typing
|
687
|
+
* a key on the keyboard. It does not honor settings from the shiftKeyDown, controlKeyDown, altKeyDown and
|
688
|
+
* metaKeyDown commands, and does not target any particular HTML element. To send a keystroke to a particular
|
689
|
+
* element, focus on the element first before running this command.
|
690
|
+
*
|
691
|
+
* @param keycode an integer keycode number corresponding to a java.awt.event.KeyEvent; note that Java keycodes are NOT the same thing as JavaScript keycodes!
|
692
|
+
*/
|
693
|
+
// This doesn't really do anything on the JS side; we let the Selenium Server take care of this for us!
|
694
|
+
};
|
695
|
+
|