-
Notifications
You must be signed in to change notification settings - Fork 101
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
draft: parser xml file to tree node and generate test case #535
Draft
zhou9584
wants to merge
11
commits into
main
Choose a base branch
from
zhoule/analysisSmartTestResult
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Draft
Changes from all commits
Commits
Show all changes
11 commits
Select commit
Hold shift + click to select a range
3f17c18
draft: parser xml file to tree node
zhou9584 2f11560
Update PageNode.java
zhou9584 1b980a6
update Prompt
zhou9584 7affe36
update
DexterDreeeam 42c3a4e
Maestro case generation
zhou9584 e5108b7
Merge branch 'main' into zhoule/analysisSmartTestResult
zhou9584 ddf93f0
rebuild case generation code
zhou9584 69220de
Merge branch 'main' into zhoule/analysisSmartTestResult
zhou9584 2146fdb
Update build.gradle
zhou9584 af3ce99
add an api to download cases
zhou9584 7b412ed
Merge branch 'main' into zhoule/analysisSmartTestResult
zhou9584 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
41 changes: 41 additions & 0 deletions
41
center/src/main/java/com/microsoft/hydralab/center/service/LongChainExample.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
// Copyright (c) Microsoft Corporation. | ||
// Licensed under the MIT License. | ||
|
||
package com.microsoft.hydralab.center.service; | ||
|
||
import dev.langchain4j.model.input.structured.StructuredPrompt; | ||
|
||
/** | ||
* @author zhoule | ||
* @date 07/13/2023 | ||
*/ | ||
|
||
public class LongChainExample { | ||
|
||
@StructuredPrompt({ | ||
"I want you to act as a software tester. I will provide a route map of a mobile application and it will be your job to write a test case. ", | ||
"The case should be in maestro script format. This is a maestro example", | ||
"{{maestroExample}}", | ||
"Firstly I will introduce the format of the route map.", | ||
"1. It is a unidirectional ordered graph in xml format, the nodes attribute are the pages of app and the id property of each node is the unique id of page. " + | ||
"By the way the id of node equals -1 means the app has not been opened.", | ||
"2. The edges attributes means the only way of jumping from a page to another page. The source property is the unique id of original page and the target property " + | ||
"is the unique id of the page after jumping. The attvalue of each edge means the operation type such launch app, click button, click testview etc..", | ||
"The commands that maestro supported is in the site https://maestro.mobile.dev/api-reference/commands.", | ||
"Requirements:", | ||
"1. the case should start from node which id is -1.", | ||
"2. the case must follow the direction of the edge.", | ||
"3. the case should jump as many pages as possible of the app.", | ||
"4. the page can be visited only once", | ||
"5. you can't use the back command", | ||
"6. add comment to case declare current page id", | ||
"The first route map is {{routeMap}}", | ||
"please generate a maestro script for this route map." | ||
}) | ||
static class MaestroCaseGeneration { | ||
|
||
String maestroExample; | ||
String routeMap; | ||
} | ||
|
||
} |
112 changes: 112 additions & 0 deletions
112
...rc/main/java/com/microsoft/hydralab/center/service/generation/AbstractCaseGeneration.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
// Copyright (c) Microsoft Corporation. | ||
// Licensed under the MIT License. | ||
|
||
package com.microsoft.hydralab.center.service.generation; | ||
|
||
import com.microsoft.hydralab.common.util.PageNode; | ||
import org.dom4j.Document; | ||
import org.dom4j.DocumentException; | ||
import org.dom4j.Element; | ||
import org.dom4j.io.SAXReader; | ||
import org.springframework.util.StringUtils; | ||
|
||
import java.io.File; | ||
import java.util.HashMap; | ||
import java.util.List; | ||
import java.util.Map; | ||
|
||
/** | ||
* @author zhoule | ||
* @date 07/21/2023 | ||
*/ | ||
|
||
public abstract class AbstractCaseGeneration { | ||
public PageNode parserXMLToPageNode(String xmlFilePath) { | ||
// read xml file, get page node and action info | ||
Document document = null; | ||
SAXReader saxReader = new SAXReader(); | ||
try { | ||
document = saxReader.read(xmlFilePath); | ||
} catch (DocumentException e) { | ||
throw new RuntimeException(e); | ||
} | ||
List<Element> pages = document.getRootElement().element("graph").element("nodes").elements("node"); | ||
List<Element> actions = document.getRootElement().element("graph").element("edges").elements("edge"); | ||
|
||
Map<Integer, PageNode> pageNodes = new HashMap<>(); | ||
// init page node | ||
for (Element page : pages) { | ||
PageNode pageNode = new PageNode(); | ||
int id = Integer.parseInt(page.attributeValue("id")); | ||
pageNode.setId(id); | ||
pageNodes.put(id, pageNode); | ||
} | ||
// init action info | ||
for (Element action : actions) { | ||
int source = Integer.parseInt(action.attributeValue("source")); | ||
int target = Integer.parseInt(action.attributeValue("target")); | ||
if (source == target) { | ||
continue; | ||
} | ||
int actionId = Integer.parseInt(action.attributeValue("id")); | ||
//link action to page | ||
pageNodes.get(source).getActionInfoList().add(parserAction(action)); | ||
//link page to page | ||
pageNodes.get(source).getChildPageNodeMap().put(actionId, pageNodes.get(target)); | ||
} | ||
return pageNodes.get(0); | ||
} | ||
|
||
private PageNode.ActionInfo parserAction(Element element) { | ||
PageNode.ActionInfo actionInfo = new PageNode.ActionInfo(); | ||
Map<String, Object> arguments = new HashMap<>(); | ||
actionInfo.setId(Integer.parseInt(element.attributeValue("id"))); | ||
actionInfo.setActionType("click"); | ||
|
||
PageNode.ElementInfo elementInfo = new PageNode.ElementInfo(); | ||
String sourceCode = element.element("attvalues").element("attvalue").attributeValue("value"); | ||
elementInfo.setText(extractElementAttr("Text", sourceCode)); | ||
elementInfo.setClassName(extractElementAttr("Class", sourceCode)); | ||
elementInfo.setClickable(Boolean.parseBoolean(extractElementAttr("Clickable", sourceCode))); | ||
elementInfo.setResourceId(extractElementAttr("ResourceID", sourceCode)); | ||
actionInfo.setTestElement(elementInfo); | ||
if (!StringUtils.isEmpty(elementInfo.getText())) { | ||
arguments.put("defaultValue", elementInfo.getText()); | ||
} else if (!StringUtils.isEmpty(elementInfo.getResourceId())) { | ||
arguments.put("id", elementInfo.getResourceId()); | ||
} | ||
actionInfo.setArguments(arguments); | ||
return actionInfo; | ||
} | ||
|
||
private String extractElementAttr(String attrName, String elementStr) { | ||
String[] attrs = elementStr.split(attrName + ": "); | ||
if (attrs.length > 1 && !attrs[1].startsWith(",")) { | ||
return attrs[1].split(",")[0]; | ||
} | ||
return ""; | ||
} | ||
|
||
/** | ||
* explore all path of page node | ||
* | ||
* @param pageNode | ||
* @param nodePath | ||
* @param action | ||
* @param explorePaths | ||
*/ | ||
public void explorePageNodePath(PageNode pageNode, String nodePath, String action, List<PageNode.ExplorePath> explorePaths) { | ||
if (pageNode.getChildPageNodeMap().isEmpty()) { | ||
explorePaths.add(new PageNode.ExplorePath(nodePath + "_" + pageNode.getId(), action)); | ||
return; | ||
} | ||
for (Map.Entry<Integer, PageNode> entry : pageNode.getChildPageNodeMap().entrySet()) { | ||
explorePageNodePath(entry.getValue(), StringUtils.isEmpty(nodePath) ? String.valueOf(pageNode.getId()) : nodePath + "_" + pageNode.getId(), | ||
StringUtils.isEmpty(action) ? String.valueOf(entry.getKey()) : action + "," + entry.getKey(), explorePaths); | ||
} | ||
} | ||
|
||
public abstract File generateCaseFile(PageNode pageNode, List<PageNode.ExplorePath> explorePaths); | ||
|
||
public abstract File generateCaseFile(PageNode pageNode, PageNode.ExplorePath explorePaths, File caseFolder); | ||
} |
107 changes: 107 additions & 0 deletions
107
...n/java/com/microsoft/hydralab/center/service/generation/MaestroCaseGenerationService.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,107 @@ | ||
// Copyright (c) Microsoft Corporation. | ||
// Licensed under the MIT License. | ||
|
||
package com.microsoft.hydralab.center.service.generation; | ||
|
||
import com.microsoft.hydralab.center.util.CenterConstant; | ||
import com.microsoft.hydralab.common.util.DateUtil; | ||
import com.microsoft.hydralab.common.util.FileUtil; | ||
import com.microsoft.hydralab.common.util.HydraLabRuntimeException; | ||
import com.microsoft.hydralab.common.util.PageNode; | ||
import org.springframework.stereotype.Service; | ||
|
||
import java.io.File; | ||
import java.util.Date; | ||
import java.util.List; | ||
import java.util.Map; | ||
|
||
/** | ||
* @author zhoule | ||
* @date 07/14/2023 | ||
*/ | ||
|
||
@Service | ||
public class MaestroCaseGenerationService extends AbstractCaseGeneration { | ||
/** | ||
* generate maestro case files and zip them | ||
* | ||
* @param pageNode | ||
* @param explorePaths | ||
* @return | ||
*/ | ||
@Override | ||
public File generateCaseFile(PageNode pageNode, List<PageNode.ExplorePath> explorePaths) { | ||
// create temp folder to store case files | ||
File tempFolder = new File(CenterConstant.CENTER_TEMP_FILE_DIR, DateUtil.fileNameDateFormat.format(new Date())); | ||
if (!tempFolder.exists()) { | ||
tempFolder.mkdirs(); | ||
} | ||
// generate case files | ||
for (PageNode.ExplorePath explorePath : explorePaths) { | ||
generateCaseFile(pageNode, explorePath, tempFolder); | ||
} | ||
if (tempFolder.listFiles().length == 0) { | ||
return null; | ||
} | ||
// zip temp folder | ||
File zipFile = new File(tempFolder.getParent() + "/" + tempFolder.getName() + ".zip"); | ||
FileUtil.zipFile(tempFolder.getAbsolutePath(), zipFile.getAbsolutePath()); | ||
FileUtil.deleteFile(tempFolder); | ||
return zipFile; | ||
} | ||
|
||
@Override | ||
public File generateCaseFile(PageNode pageNode, PageNode.ExplorePath explorePath, File caseFolder) { | ||
File maestroCaseFile = new File(caseFolder, explorePath.getPath() + ".yaml"); | ||
String caseContent = buildConfigSection(pageNode.getPageName()); | ||
caseContent += buildDelimiter(); | ||
caseContent += buildCommandSection("launch", null); | ||
String[] actionIds = explorePath.getActions().split(","); | ||
PageNode pageNodeCopy = pageNode; | ||
for (String actionId : actionIds) { | ||
PageNode.ActionInfo action = pageNodeCopy.getActionInfoList().stream().filter(actionInfo -> actionInfo.getId() == Integer.parseInt(actionId)).findFirst().get(); | ||
caseContent += buildCommandSection(action.getActionType(), action.getArguments()); | ||
pageNodeCopy = pageNodeCopy.getChildPageNodeMap().get(Integer.parseInt(actionId)); | ||
} | ||
caseContent += buildCommandSection("stop", null); | ||
FileUtil.writeToFile(caseContent, maestroCaseFile.getAbsolutePath()); | ||
return maestroCaseFile; | ||
} | ||
|
||
private String buildConfigSection(String appId) { | ||
return "appId: " + appId + "\n"; | ||
} | ||
|
||
private String buildDelimiter() { | ||
return "---\n"; | ||
} | ||
|
||
private String buildCommandSection(String actionType, Map<String, Object> arguments) { | ||
String command = "-"; | ||
switch (actionType) { | ||
case "launch": | ||
command = command + " launchApp\n"; | ||
break; | ||
case "click": | ||
command = command + " tapOn:"; | ||
if (arguments.size() == 0) { | ||
throw new HydraLabRuntimeException("arguments is empty"); | ||
} | ||
if (arguments.containsKey("defaultValue")) { | ||
command = command + " " + arguments.get("defaultValue") + "\n"; | ||
break; | ||
} | ||
command = command + "\n"; | ||
for (String key : arguments.keySet()) { | ||
command = command + " " + key + ": \"" + arguments.get(key) + "\"\n"; | ||
} | ||
break; | ||
case "stop": | ||
command = command + " stopApp\n"; | ||
break; | ||
default: | ||
throw new HydraLabRuntimeException("Unsupported action type: " + actionType); | ||
} | ||
return command; | ||
} | ||
} |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why using compileOnly?