How To Combine SAP GUI Scripting And Selenium WebDriver
More and more loses the SAP GUI for Windows his unique position as primary UI and also more and more are desktop applications replaced with web applications. The SAP GUI for Windows offers for automation approaches the SAP GUI Scripting, a consolidated and technical reliable way. For web browser wrotes Former Member in his great blog about START (Simple Test Automation for Regression Tests) an UI Automation Framework to automate the testing of application. START bases on Selenium WebDrivers. So I think it could be advantageous to combine SAP GUI Scripting and Selenium.
Java
So I advanced my Scripting Tracker to create also Java code, to offer the possibilty to combine SAP GUI Scripting on this language platform. To use SAP GUI Scripting with Java you need the Java COM Bridge (JaCoB).
For the web activities I use in my context the chrome web driver, an x86 application, because I use the Chrome browser. So it is necessary to use the x86 (i586) version of the JDK, also for the SAP GUI for Windows. To record the activities on the web browser I use Firefox with Selenium IDE. You see it is a very heterogeneous development environment. But after all, it works. Here an example code how to combine SAP GUI for Windows activities with any activities in a web browser. In the example I combine information from the TAC SE16 from the table DEMO_CR_CUSTOMRS with a web dynpro application DEMO_WD_CAR_RENTAL – this is not really useful but shows the horizons.
//-Begin----------------------------------------------------------------
package de.stschnell;
//-Import-------------------------------------------------------------
import com.jacob.activeX.*;
import com.jacob.com.*;
import java.util.HashMap;
import java.util.Map;
import org.openqa.selenium.*;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.chrome.ChromeOptions;
import org.openqa.selenium.remote.*;
public class SAPGUIScriptingWithSelenium {
public static void main(String[] args) {
//-Variables--------------------------------------------------------
ActiveXComponent SAPROTWr, GUIApp, Connection, Session, Obj;
Dispatch ROTEntry;
Variant Value, ScriptEngine;
String cnt = "0";
ComThread.InitSTA();
//-Set SapGuiAuto = GetObject("SAPGUI")-----------------------------
SAPROTWr = new ActiveXComponent("SapROTWr.SapROTWrapper");
try {
ROTEntry = SAPROTWr.invoke("GetROTEntry", "SAPGUI").toDispatch();
//-Set application = SapGuiAuto.GetScriptingEngine----------------
ScriptEngine = Dispatch.call(ROTEntry, "GetScriptingEngine");
GUIApp = new ActiveXComponent(ScriptEngine.toDispatch());
//-Set connection = application.Children(0)-----------------------
Connection = new ActiveXComponent(
GUIApp.invoke("Children", 0).toDispatch()
);
//-Set session = connection.Children(0)---------------------------
Session = new ActiveXComponent(
Connection.invoke("Children", 0).toDispatch()
);
//-Open SE16------------------------------------------------------
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[0]/tbar[0]/okcd").toDispatch());
Obj.setProperty("text", "/nse16");
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[0]").toDispatch());
Obj.invoke("sendVKey", 0);
//-Open selection view for table DEMO_CR_CUSTOMRS-----------------
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[0]/usr/ctxtDATABROWSE-TABLENAME").toDispatch());
Obj.setProperty("text", "DEMO_CR_CUSTOMRS");
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[0]/tbar[1]/btn[7]").toDispatch());
Obj.invoke("press");
//-Open dialog "Number of Entries"--------------------------------
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[0]/tbar[1]/btn[31]").toDispatch());
Obj.invoke("press");
//-Get the number of entries--------------------------------------
Obj = new ActiveXComponent(Session.invoke("findById",
"wnd[1]/usr/txtG_DBCOUNT").toDispatch());
Value = Obj.getProperty("text");
cnt = Value.toString();
} catch (Exception e) {
System.out.println(
e.getMessage().toString()
);
} finally {
ComThread.Release();
}
//-If number of entries > 0-----------------------------------------
if (cnt != "0") {
//-Set path to chromedriver---------------------------------------
System.setProperty(
"webdriver.chrome.driver",
"C:/Program Files/Selenium/chromedriver.exe");
//-Set path to chrome browser-------------------------------------
Map<String, Object> chromeOptions = new HashMap<String, Object>();
chromeOptions.put("binary", "C:/Program Files/Google/Chrome/Application/chrome.exe");
DesiredCapabilities capabilities = DesiredCapabilities.chrome();
capabilities.setCapability(ChromeOptions.CAPABILITY, chromeOptions);
//-Opens a web browser window-------------------------------------
WebDriver driver = new ChromeDriver(capabilities);
//-Do your activities in the browser------------------------------
driver.get("http://nsp.stschnell.de:8630/sap/bc/webdynpro/sap/demo_wd_car_rental");
driver.findElement(By.id("WD1C")).clear();
driver.findElement(By.id("WD1C")).sendKeys("00000001");
driver.findElement(By.cssSelector("span.urBtnCntTxt")).click();
driver.close();
driver.quit();
}
System.exit(0);
}
}
//-End------------------------------------------------------------------
Here Scripting Tracker which records Java code from the SAP GUI for Windows activities:
Here Firefox with Selenium IDE to record web activities:
Here the SAP GUI for Windows with SAP GUI Scripting and Eclipse in debug mode side by side:
Here Chrome with Eclipse in debug mode side by side:
PowerShell
A longer time ago I wrote here about the possibility to use PowerShell with SAP GUI Scripting. Selenium offers, besides Java, also dotNET libraries. On this way it is possible to combine SAP GUI Scripting with the WebDriver also on the base of PowerShell.
Here the same example in PowerShell.
#-Begin-----------------------------------------------------------------
#-Includes------------------------------------------------------------
."$PSScriptRoot\COM.ps1"
#-Set SapGuiAuto = GetObject("SAPGUI")--------------------------------
$SapGuiAuto = Get-Object( , "SAPGUI")
If ($SapGuiAuto -isnot [__ComObject]) {
Exit
}
#-Set application = SapGuiAuto.GetScriptingEngine---------------------
$application = Invoke-Method $SapGuiAuto "GetScriptingEngine"
If ($application -isnot [__ComObject]) {
Free-Object $SapGuiAuto
Exit
}
#-Set connection = application.Children(0)----------------------------
$connection = Get-Property $application "Children" @(0)
If ($connection -eq $Null) {
Free-Object $SapGuiAuto
Exit
}
#-Set session = connection.Children(0)--------------------------------
$session = Get-Property $connection "Children" @(0)
If ($session -eq $Null) {
Free-Object $SapGuiAuto
Exit
}
#-Open SE16-----------------------------------------------------------
$ID = Invoke-Method $session "findById" @("wnd[0]/tbar[0]/okcd")
Set-Property $ID "text" @("/nse16")
$ID = Invoke-Method $session "findById" @("wnd[0]")
Invoke-Method $ID "sendVKey" @(0)
#-Open selection view for table DEMO_CR_CUSTOMRS----------------------
$ID = Invoke-Method $session "findById" @("wnd[0]/usr/ctxtDATABROWSE-TABLENAME")
Set-Property $ID "text" @("DEMO_CR_CUSTOMRS")
$ID = Invoke-Method $session "findById" @("wnd[0]/tbar[1]/btn[7]")
Invoke-Method $ID "press"
#-Open dialog "Number of Entries"-------------------------------------
$ID = Invoke-Method $session "findById" @("wnd[0]/tbar[1]/btn[31]")
Invoke-Method $ID "press"
#-Get the number of entries-------------------------------------------
$ID = Invoke-Method $session "findById" @("wnd[1]/usr/txtG_DBCOUNT")
$Value = Get-Property $ID "text"
#-If number of entries > 0--------------------------------------------
If ($Value -ne 0) {
#-Load libraries----------------------------------------------------
[System.Reflection.Assembly]::LoadFrom("C:\Program Files\Selenium\Selenium.WebDriverBackedSelenium.dll")
[System.Reflection.Assembly]::LoadFrom("C:\Program Files\Selenium\WebDriver.dll")
[System.Reflection.Assembly]::LoadFrom("C:\Program Files\Selenium\WebDriver.Support.dll")
#-Set path to chrome browser----------------------------------------
$Options = New-Object OpenQA.Selenium.Chrome.ChromeOptions
$Options.BinaryLocation = "C:/Program Files/Google/Chrome/Application/chrome.exe"
#-Opens a web browser window----------------------------------------
$WebDriver = New-Object OpenQA.Selenium.Chrome.ChromeDriver("C:\Program Files\Selenium", $Options)
$WebDriver.Url = "http://nsp.stschnell.de:8630/sap/bc/webdynpro/sap/demo_wd_car_rental"
#-Do your activities in the browser---------------------------------
$WebDriver.FindElementById("WD1C").Clear()
$WebDriver.FindElementById("WD1C").SendKeys("00000001")
$WebDriver.FindElementByCssSelector("span.urBtnCntTxt").click()
$WebDriver.Close()
$WebDriver.Quit()
}
#-End-------------------------------------------------------------------
On this ways is it possible to combine SAP GUI Scripting with e.g. WebDynpro or UI5 applications. So you can reach a higher integration level in your automation approaches.
Thanks for this.I was able to login and search for some objects in SAP UI.Now i am stucked in reading data from a grid.After searching system generate a grid.Now i want to read the data from the grid .
I can click on grid cells and rows but i can not read the data.How to read the data from grid.
sample code of clicking a row and cell
Obj = new ActiveXComponent(Session.invoke("findById", "wnd[0]/usr/cntlGRID1/shellcont/shell/shellcont[1]/shell").toDispatch()); Obj.setProperty("selectedRows", "0"); Obj = new ActiveXComponent(Session.invoke("findById", "wnd[0]/usr/cntlGRID1/shellcont/shell/shellcont[1]/shell").toDispatch()); Obj.invoke("doubleClickCurrentCell");
Hello Mohamed,
thanks for your reply.
Sorry but I have no Java sample code to read an ALV grid, but you can find here and here examples in VBScript. I assume that it should looks like this:
Let us know your results.
Best regards
Stefan
Thanks for the response.
Rows = Table.getProperty(“RowCount”) – 1;
Cols = Table.getProperty(“ColumnCount”) – 1;
above returns the no of column and rows.But
Table.invoke(“GetCellValue”, 1, 1);
this returns an exception.I have tried with different row and column id’s as well.
Exception – Method threw ‘com.jacob.com.ComFailException’ exception.
Detailed message -com.jacob.com.ComFailException: Invoke of: GetCellValue
Source: SAP Frontend Server
Description:
Stack trace –
0 = {StackTraceElement@1577} “com.jacob.com.Dispatch.invokev(Native Method)”
1 = {StackTraceElement@1578} “com.jacob.com.Dispatch.invokev(Dispatch.java:625)”
2 = {StackTraceElement@1579} “com.jacob.com.Dispatch.callN(Dispatch.java:453)”
3 = {StackTraceElement@1580} “com.jacob.com.Dispatch.call(Dispatch.java:541)”
4 = {StackTraceElement@1581} “com.jacob.activeX.ActiveXComponent.invoke(ActiveXComponent.java:453)”
Hello Mohamed,
try this:
Let us know your results.
Cheers
Stefan
Thanks for the reply.Appreciate it.The above code also returns the same error.But I am lucky enough to come across a code which worked for me.
Thanks for sharing your code.
Hi,
Sorry for interrupting again. Have you ever done automation for SAP MDM -SAP Master Data Management.
Would appreciate you thoughts and Ideas on it.
Hello Mohamed,
no, unfortunately I have never worked with MDM, so I can't give you hints, sorry.
Best regards
Stefan
Thanks ..
Hi ,
I wanted to automate SAP GUI transactions. this blog gave me many useful info.
I used the same Logon script shared on this blog but its not working.
Could you please help me on this?
thanks in Advance.
--Jaya
Hi ,
I wanted to automate SAP GUI transactions. this blog gave me many useful info.
I used the same Logon script shared on this blog but its not working. Any specific changes to be done? the script does not throw any error after resolving dependencies.
Could you please help me on this?
thanks in Advance.
–Jaya
Hi,
Sorry for interrupting again. Have you ever done automation for SAP MDM -SAP Master Data Management.
Would appreciate you thoughts and Ideas on it.
Hi - Can we use name for identifying the object instead of FindByID always? Please advise.
Hello Manoranjan,
welcome in the SAP Community.
Sure is that possible, here examples of different login variant:
Best regards
Stefan
Thanks a lot Stefan.
The examples shared are VB, any reference how that would be in Java?
One more help.. Is there any reference link which I can refer to?
Currently, I’m seeing issues for SAP grid where the ID would end at shell and post that, i need to write functions/methods to access the content.
Hello Manoranjan,
here a full example in Java which uses findByName method:
Best regards
Stefan
Hello Manoranjan,
I am no Java programmer, but as far as I understand your code correct it can't work. You are looping over the sessions but your xpath variable is set fix to ses[0]. Here my version of your code:
Best regards
Stefan
Thanks Stefan, will check the code shared.
Hi Stefan - I'm trying to scroll down on MM43 but sendVKey 82 is not active/configured on this screen.
Basically I'm trying to take a screenshot on the attributes which are not visible.
Could you please suggest if there is any alternate method to scroll down to a specific field on MM43?
Hello Manoranjan,
it would be great if you ask this as a new question in the forum and not as comment of this blog.
Thanks and best regards
Stefan
Hi Stefan,
I tried running the code mentioned in the blog but I am getting following error. Could you please help me.
Exception in thread “main” com.jacob.com.ComFailException: Can’t get object clsid from progid
at com.jacob.com.Dispatch.createInstanceNative(Native Method)
at com.jacob.com.Dispatch.<init>(Dispatch.java:99)
at com.jacob.activeX.ActiveXComponent.<init>(ActiveXComponent.java:58)
at testSAP.LoginSAP.main(LoginSAP.java:25)
Hello Neha,
is the SAP GUI Scripting installed on your computer?
Best regards
Stefan
Thanks Stefan.
Enabled the scripting in SAP GUI options and tried running the code mentioned in the blog. But it is failing at this line
SAPROTWr = new ActiveXComponent("SapROTWr.SapROTWrapper");
Error:
Exception in thread "main" com.jacob.com.ComFailException: Can't co-create object
at com.jacob.com.Dispatch.createInstanceNative(Native Method)
at com.jacob.com.Dispatch.<init>(Dispatch.java:99)
at com.jacob.activeX.ActiveXComponent.<init>(ActiveXComponent.java:58)
at testSAP.LoginSAP.main(LoginSAP.java:25)
Hello Neha,
do you use x86 version of Java? SAP Rot Wrapper is an x86 version and it is not possible to use it with x64 Java version.
Let us know your results.
Cheers
Stefan
Thanks Stefan. I tried with x86 version of Java. It worked!
Hi Stefan,
I have installed the Scripting Tracker from https://tracker.stschnell.de/
But I do not see the java option to select for the coding. Could you please help me here.
Hello Neha,
I disable the Java support with release 4 of Scripting Tracker. Java is since version 9 available only as x64 version. The only way to get access to the Windows Running Object Table (ROT) is via an x86 library, as I described above. So nobody has a chance to use this method with an actual Java release. From this perspective I see no reason to support Java furthermore.
Best regards
Stefan
Okay Stefan. Thanks!
But is there any possibility of getting any of the versions of Scripting Tracker below release 4?
Because I wanted the Java Scripting Engine which was more helpful as we do the SAP Automation using Java and can be integrated with the Hybris web apps as well which are automated using Selenium.
Hello Stefan
I appreciate your work very much!
I'd like to ask you for the very same thing, just to share a Scripting tracker version that supports Java plotter.
Thank you.
Hello Kacper Witos,
thanks for your post and your kindful words. I got many mails from users who wanted to have Java recording possibility back in the Tracker. This and the answer from Jude Bradley here, with the hint to note 2689304, changes the the things. “…as of patchlevel 8 of SAP GUI for Windows 7.50 and as of SAP GUI for Windows 7.60, the installation of SAP GUI for Windows automatically writes the respective registry values so that a usage of saprotwr.dll from 64bit processes is possible.”
You can find here Tracker with Java support.
Good automation, also to Neha Goje.
Best regards
Stefan
In VB Language sap connection string is connection = GUIApp.OpenConnection("XYZ [PUBLIC]", true)
But if same I have to do it with Java then What should be it.
Hi Stefan,
Thanks a lot for your continuous help in automating SAP GUI with selenium - java approach. I have got a blocker. If you could please share any code specific to drop down selection and the menu bar selection of GUI. Thanks so much!! Waiting for you response...
Hello Soumen,
welcome in the SAP Community.
To choose an entry in the menu bar you can try this:
With the number in the brackets you choose the menu, e.g. with "menu[0]" you choose the first menu.
To set an entry in a combo box (drop down selection) you can try this:
Here you choose the combo box (cmb) and set the key, in my example both German, country and language.
Best regards
Stefan
Another question Stephan. If you could please help. How UFT gets hold of every object of SAP GUI? How can we get the same information about SAP GUI objects in java also? for example: SAP GUI table column names, menubar etc?
Hello Soumen,
I don’t know UFT. But to get information about the controls of the SAP GUI for Windows you can use Scripting Tracker, as I described above. You can find here Tracker with Java support.
Best regards
Stefan
And if you could please share all the methods, functions etc to be used in SAP GUI Java combination so that we can do a full end to end flow of SAP by java and selenium.
Hello Soumen,
what are you exactly looking for?
Best regards
Stefan
Hi Stefan Schnell,
getting below mentioned exception, what would be the solution. please let me know.
"java.lang.IllegalStateException: Dispatch not hooked to windows memory"