selenium-core-runner 0.0.3
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.
- 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
|
+
|