mirror of
https://github.com/SAP/jenkins-library.git
synced 2025-01-18 05:18:24 +02:00
cd2ae89229
* Adjust local changed objects with commit history * fix the righ history entry * fix for local packages commit history * Adapt unit tests and fix for retry after 500 * Rename gctsExecuteABAPUnitTests to gctsExecuteABAPQualityChecks * Add unit test file for gCTSExecuteABAPUnitTest * add a test step * gcts Test change files * Add unit test for gCTS test * adapt gctsTEST * deletegcts * command * maintain old step * Adapt changes to documentation * fix for go generate
1897 lines
54 KiB
Go
1897 lines
54 KiB
Go
package cmd
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/xml"
|
|
"fmt"
|
|
"html"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/http/cookiejar"
|
|
"net/url"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/SAP/jenkins-library/pkg/command"
|
|
piperhttp "github.com/SAP/jenkins-library/pkg/http"
|
|
"github.com/SAP/jenkins-library/pkg/log"
|
|
"github.com/SAP/jenkins-library/pkg/telemetry"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var atcFailure, aUnitFailure bool
|
|
|
|
func gctsExecuteABAPQualityChecks(config gctsExecuteABAPQualityChecksOptions, telemetryData *telemetry.CustomData) {
|
|
|
|
// for command execution use Command
|
|
c := command.Command{}
|
|
// reroute command output to logging framework
|
|
c.Stdout(log.Writer())
|
|
c.Stderr(log.Writer())
|
|
|
|
httpClient := &piperhttp.Client{}
|
|
// error situations should stop execution through log.Entry().Fatal() call which leads to an os.Exit(1) in the end
|
|
err := rungctsExecuteABAPQualityChecks(&config, httpClient)
|
|
if err != nil {
|
|
log.Entry().WithError(err).Fatal("step execution failed")
|
|
}
|
|
|
|
if aUnitFailure || atcFailure {
|
|
|
|
log.Entry().Fatal("step execution failed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
func rungctsExecuteABAPQualityChecks(config *gctsExecuteABAPQualityChecksOptions, httpClient piperhttp.Sender) error {
|
|
|
|
const localChangedObjects = "localchangedobjects"
|
|
const remoteChangedObjects = "remotechangedobjects"
|
|
const localChangedPackages = "localchangedpackages"
|
|
const remoteChangedPackages = "remotechangedpackages"
|
|
const repository = "repository"
|
|
const packages = "packages"
|
|
|
|
cookieJar, cookieErr := cookiejar.New(nil)
|
|
if cookieErr != nil {
|
|
return errors.Wrap(cookieErr, "creating a cookie jar failed")
|
|
}
|
|
|
|
maxRetries := -1
|
|
clientOptions := piperhttp.ClientOptions{
|
|
CookieJar: cookieJar,
|
|
Username: config.Username,
|
|
Password: config.Password,
|
|
MaxRetries: maxRetries,
|
|
}
|
|
|
|
httpClient.SetOptions(clientOptions)
|
|
|
|
log.Entry().Infof("start of gctsExecuteABAPQualityChecks step with configuration values: %v", config)
|
|
|
|
var objects []repoObject
|
|
var err error
|
|
|
|
log.Entry().Info("scope:", config.Scope)
|
|
|
|
switch strings.ToLower(config.Scope) {
|
|
case localChangedObjects:
|
|
objects, err = getLocalObjects(config, httpClient)
|
|
case remoteChangedObjects:
|
|
objects, err = getRemoteObjects(config, httpClient)
|
|
case localChangedPackages:
|
|
objects, err = getLocalPackages(config, httpClient)
|
|
case remoteChangedPackages:
|
|
objects, err = getRemotePackages(config, httpClient)
|
|
case repository:
|
|
objects, err = getRepositoryObjects(config, httpClient)
|
|
case packages:
|
|
objects, err = getPackages(config, httpClient)
|
|
default:
|
|
log.Entry().Info("the specified scope does not exists, the default one will be used:" + repository)
|
|
objects, err = getRepositoryObjects(config, httpClient)
|
|
}
|
|
|
|
if err != nil {
|
|
log.Entry().WithError(err).Fatal("failure in get objects")
|
|
}
|
|
|
|
if objects == nil {
|
|
log.Entry().Warning("no object delta was found, therefore the step execution will stop")
|
|
return nil
|
|
|
|
}
|
|
|
|
log.Entry().Infof("objects to be checked:")
|
|
for _, object := range objects {
|
|
log.Entry().Info(object.Type, " ", object.Object)
|
|
}
|
|
|
|
if config.AUnitTest {
|
|
|
|
// wrapper for execution of AUnit Test
|
|
err := executeAUnitTest(config, httpClient, objects)
|
|
|
|
if err != nil {
|
|
log.Entry().WithError(err)
|
|
|
|
}
|
|
|
|
if aUnitFailure {
|
|
|
|
log.Entry().Error("unit test(s) has/have failed! Check " + config.AUnitResultsFileName + " for more information! If you have enabled Warnings-Next-Generation Plugin, you can see the issues there!")
|
|
|
|
} else {
|
|
|
|
log.Entry().Info("AUnit test run completed successfully. If there are any results from the run, the results are saved in " + config.AUnitResultsFileName)
|
|
|
|
}
|
|
}
|
|
|
|
if config.AtcCheck {
|
|
|
|
// wrapper for execution of ATCChecks
|
|
err = executeATCCheck(config, httpClient, objects)
|
|
|
|
if err != nil {
|
|
log.Entry().WithError(err).Fatal("execute ATC Check failed")
|
|
}
|
|
|
|
if atcFailure {
|
|
|
|
log.Entry().Error(" ATC issue(s) found! Check " + config.AtcResultsFileName + " for more information! If you have enabled Warnings-Next-Generation Plugin, you can see the issues there!")
|
|
|
|
} else {
|
|
|
|
log.Entry().Info("ATCCheck test run completed successfully. If there are any results from the run, the results are saved in " + config.AtcResultsFileName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func getLocalObjects(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
var localObjects []repoObject
|
|
var localObject repoObject
|
|
|
|
log.Entry().Info("get local changed objects started")
|
|
|
|
if config.Commit == "" {
|
|
|
|
return []repoObject{}, errors.Errorf("For scope: localChangedObjects you need to specify a commit")
|
|
|
|
}
|
|
|
|
history, err := getHistory(config, client)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get local changed objects failed")
|
|
}
|
|
|
|
if len(history.Result) == 0 {
|
|
|
|
return []repoObject{}, errors.Wrap(err, "no activities (from commit - to commit) were found")
|
|
}
|
|
|
|
fromCommit := history.Result[0].FromCommit
|
|
log.Entry().Info("from Commit: ", fromCommit)
|
|
toCommit := history.Result[0].ToCommit
|
|
log.Entry().Info("to Commit: ", toCommit)
|
|
|
|
// object delta between FromCommit and ToCommit retrieved from Activities Tab in gCTS
|
|
resp, err := getObjectDifference(config, fromCommit, toCommit, client)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get local changed objects failed")
|
|
}
|
|
|
|
for _, object := range resp.Objects {
|
|
localObject.Object = object.Name
|
|
localObject.Type = object.Type
|
|
localObjects = append(localObjects, localObject)
|
|
}
|
|
|
|
log.Entry().Info("get local changed objects finished")
|
|
|
|
return localObjects, nil
|
|
}
|
|
|
|
func getRemoteObjects(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
var remoteObjects []repoObject
|
|
var remoteObject repoObject
|
|
var currentRemoteCommit string
|
|
|
|
log.Entry().Info("get remote changed objects started")
|
|
|
|
if config.Commit == "" {
|
|
|
|
return []repoObject{}, errors.Errorf("For scope: remoteChangedObjects you need to specify a commit")
|
|
|
|
}
|
|
|
|
commitList, err := getCommitList(config, client)
|
|
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get remote changed objects failed")
|
|
}
|
|
|
|
for i, commit := range commitList.Commits {
|
|
if commit.ID == config.Commit {
|
|
currentRemoteCommit = commitList.Commits[i+1].ID
|
|
break
|
|
}
|
|
}
|
|
if currentRemoteCommit == "" {
|
|
return []repoObject{}, errors.New("current remote commit was not found")
|
|
|
|
}
|
|
log.Entry().Info("current commit in the remote repository: ", currentRemoteCommit)
|
|
// object delta between the commit that triggered the pipeline and the current commit in the remote repository
|
|
resp, err := getObjectDifference(config, currentRemoteCommit, config.Commit, client)
|
|
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get remote changed objects failed")
|
|
}
|
|
|
|
for _, object := range resp.Objects {
|
|
remoteObject.Object = object.Name
|
|
remoteObject.Type = object.Type
|
|
remoteObjects = append(remoteObjects, remoteObject)
|
|
}
|
|
|
|
log.Entry().Info("get remote changed objects finished")
|
|
|
|
return remoteObjects, nil
|
|
}
|
|
|
|
func getLocalPackages(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
var localPackages []repoObject
|
|
var localPackage repoObject
|
|
|
|
log.Entry().Info("get local changed packages started")
|
|
|
|
if config.Commit == "" {
|
|
|
|
return []repoObject{}, errors.Errorf("For scope: localChangedPackages you need to specify a commit")
|
|
|
|
}
|
|
|
|
history, err := getHistory(config, client)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get local changed objects failed")
|
|
}
|
|
|
|
if len(history.Result) == 0 {
|
|
|
|
return []repoObject{}, errors.Wrap(err, "no activities (from commit - to commit) were found")
|
|
}
|
|
|
|
fromCommit := history.Result[0].FromCommit
|
|
log.Entry().Info("from Commit: ", fromCommit)
|
|
toCommit := history.Result[0].ToCommit
|
|
log.Entry().Info("to Commit: ", toCommit)
|
|
|
|
// object delta between FromCommit and ToCommit retrieved from Activities Tab in gCTS
|
|
resp, err := getObjectDifference(config, fromCommit, config.Commit, client)
|
|
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get local changed packages failed")
|
|
|
|
}
|
|
|
|
myPackages := map[string]bool{}
|
|
|
|
// objects are resolved into packages(DEVC)
|
|
for _, object := range resp.Objects {
|
|
objInfo, err := getObjectInfo(config, client, object.Name, object.Type)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get local changed packages failed")
|
|
}
|
|
if myPackages[objInfo.Devclass] {
|
|
|
|
} else {
|
|
myPackages[objInfo.Devclass] = true
|
|
localPackage.Object = objInfo.Devclass
|
|
localPackage.Type = "DEVC"
|
|
localPackages = append(localPackages, localPackage)
|
|
}
|
|
|
|
}
|
|
|
|
log.Entry().Info("get local changed packages finished")
|
|
return localPackages, nil
|
|
}
|
|
|
|
func getRemotePackages(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
var remotePackages []repoObject
|
|
var remotePackage repoObject
|
|
var currentRemoteCommit string
|
|
|
|
log.Entry().Info("get remote changed packages started")
|
|
|
|
if config.Commit == "" {
|
|
|
|
return []repoObject{}, errors.Errorf("For scope: remoteChangedPackages you need to specify a commit")
|
|
|
|
}
|
|
|
|
commitList, err := getCommitList(config, client)
|
|
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get remote changed packages failed")
|
|
}
|
|
|
|
for i, commit := range commitList.Commits {
|
|
if commit.ID == config.Commit {
|
|
currentRemoteCommit = commitList.Commits[i+1].ID
|
|
break
|
|
}
|
|
}
|
|
|
|
if currentRemoteCommit == "" {
|
|
return []repoObject{}, errors.Wrap(err, "current remote commit was not found")
|
|
|
|
}
|
|
log.Entry().Info("current commit in the remote repository: ", currentRemoteCommit)
|
|
//object delta between the commit that triggered the pipeline and the current commit in the remote repository
|
|
resp, err := getObjectDifference(config, currentRemoteCommit, config.Commit, client)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get remote changed packages failed")
|
|
}
|
|
|
|
myPackages := map[string]bool{}
|
|
// objects are resolved into packages(DEVC)
|
|
for _, object := range resp.Objects {
|
|
objInfo, err := getObjectInfo(config, client, object.Name, object.Type)
|
|
if err != nil {
|
|
return []repoObject{}, errors.Wrap(err, "get remote changed packages failed")
|
|
}
|
|
if myPackages[objInfo.Devclass] {
|
|
|
|
} else {
|
|
myPackages[objInfo.Devclass] = true
|
|
remotePackage.Object = objInfo.Devclass
|
|
remotePackage.Type = "DEVC"
|
|
remotePackages = append(remotePackages, remotePackage)
|
|
}
|
|
|
|
}
|
|
log.Entry().Info("get remote changed packages finished")
|
|
return remotePackages, nil
|
|
}
|
|
|
|
func getRepositoryObjects(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
log.Entry().Info("get repository objects started")
|
|
|
|
var repoResp repoObjectResponse
|
|
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"/objects?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return []repoObject{}, errors.Wrap(httpErr, "could not get repository objects")
|
|
} else if resp == nil {
|
|
return []repoObject{}, errors.New("could not get repository objects: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &repoResp)
|
|
if parsingErr != nil {
|
|
return []repoObject{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
|
|
var repositoryObjects []repoObject
|
|
|
|
// remove object type DEVC, because it is already included in scope packages
|
|
// also if you run ATC Checks for DEVC together with other object types, ATC checks will run only for DEVC
|
|
for _, object := range repoResp.Objects {
|
|
|
|
if object.Type != "DEVC" {
|
|
repositoryObjects = append(repositoryObjects, object)
|
|
}
|
|
|
|
}
|
|
|
|
log.Entry().Info("get repository objects finished")
|
|
|
|
// all objects that are part of the local repository
|
|
return repositoryObjects, nil
|
|
}
|
|
|
|
func getPackages(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) ([]repoObject, error) {
|
|
|
|
var packages []repoObject
|
|
|
|
log.Entry().Info("get packages started")
|
|
|
|
var repoResp repoObjectResponse
|
|
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"/objects?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return []repoObject{}, errors.Wrap(httpErr, "get packages failed: could not get repository objects")
|
|
} else if resp == nil {
|
|
return []repoObject{}, errors.New("get packages failed: could not get repository objects: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &repoResp)
|
|
if parsingErr != nil {
|
|
return []repoObject{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
// chose only DEVC from repository objects
|
|
for _, object := range repoResp.Objects {
|
|
|
|
if object.Type == "DEVC" {
|
|
packages = append(packages, object)
|
|
}
|
|
|
|
}
|
|
|
|
log.Entry().Info("get packages finished")
|
|
return packages, nil
|
|
}
|
|
|
|
func discoverServer(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (*http.Header, error) {
|
|
|
|
url := config.Host +
|
|
"/sap/bc/adt/core/discovery?sap-client=" + config.Client
|
|
|
|
header := make(http.Header)
|
|
header.Add("Accept", "application/atomsvc+xml")
|
|
header.Add("x-csrf-token", "fetch")
|
|
header.Add("saml2", "disabled")
|
|
|
|
disc, httpErr := client.SendRequest("GET", url, nil, header, nil)
|
|
|
|
defer func() {
|
|
if disc != nil && disc.Body != nil {
|
|
disc.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return nil, errors.Wrap(httpErr, "discovery of the ABAP server failed")
|
|
} else if disc == nil || disc.Header == nil {
|
|
return nil, errors.New("discovery of the ABAP server failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
return &disc.Header, nil
|
|
}
|
|
|
|
func executeAUnitTest(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, objects []repoObject) error {
|
|
|
|
log.Entry().Info("execute ABAP Unit Test started")
|
|
|
|
var innerXml string
|
|
var result runResult
|
|
|
|
for _, object := range objects {
|
|
|
|
switch object.Type {
|
|
case "CLAS":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/oo/classes/` + object.Object + `"/>`
|
|
case "DEVC":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/packages/` + object.Object + `"/>`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var xmlBody = []byte(`<?xml version="1.0" encoding="UTF-8"?>
|
|
<aunit:runConfiguration xmlns:aunit="http://www.sap.com/adt/aunit">
|
|
<external>
|
|
<coverage active="false"/>
|
|
</external>
|
|
<options>
|
|
<uriType value="semantic"/>
|
|
<testDeterminationStrategy appendAssignedTestsPreview="true" assignedTests="false" sameProgram="true"/>
|
|
<testRiskLevels critical="true" dangerous="true" harmless="true"/>
|
|
<testDurations long="true" medium="true" short="true"/>
|
|
<withNavigationUri enabled="false"/>
|
|
</options>
|
|
<adtcore:objectSets xmlns:adtcore="http://www.sap.com/adt/core">
|
|
<objectSet kind="inclusive">
|
|
<adtcore:objectReferences>` +
|
|
innerXml +
|
|
`</adtcore:objectReferences>
|
|
</objectSet>
|
|
</adtcore:objectSets>
|
|
</aunit:runConfiguration>`)
|
|
|
|
resp, err := runAUnitTest(config, client, xmlBody)
|
|
if err != nil {
|
|
return errors.Wrap(err, "execute of Aunit test has failed")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyXML(resp, &result)
|
|
if parsingErr != nil {
|
|
log.Entry().Warning(parsingErr)
|
|
return nil
|
|
}
|
|
|
|
parsedRes, err := parseUnitResult(config, client, &result)
|
|
|
|
if err != nil {
|
|
log.Entry().Warning(err)
|
|
return nil
|
|
}
|
|
|
|
log.Entry().Info("execute ABAP Unit Test finished.", parsedRes.Text)
|
|
|
|
return nil
|
|
}
|
|
|
|
func runAUnitTest(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, xml []byte) (response *http.Response, err error) {
|
|
|
|
log.Entry().Info("run ABAP Unit Test started")
|
|
url := config.Host +
|
|
"/sap/bc/adt/abapunit/testruns?sap-client=" + config.Client
|
|
|
|
discHeader, discError := discoverServer(config, client)
|
|
|
|
if discError != nil {
|
|
return response, errors.Wrap(discError, "run of unit tests failed")
|
|
}
|
|
|
|
if discHeader.Get("X-Csrf-Token") == "" {
|
|
|
|
return response, errors.Errorf("could not retrieve x-csrf-token from server")
|
|
}
|
|
|
|
header := make(http.Header)
|
|
header.Add("x-csrf-token", discHeader.Get("X-Csrf-Token"))
|
|
header.Add("Accept", "application/xml")
|
|
header.Add("Content-Type", "application/vnd.sap.adt.abapunit.testruns.result.v1+xml")
|
|
|
|
response, httpErr := client.SendRequest("POST", url, bytes.NewBuffer(xml), header, nil)
|
|
|
|
if httpErr != nil {
|
|
return response, errors.Wrap(httpErr, "run of unit tests failed")
|
|
} else if response == nil {
|
|
return response, errors.New("run of unit tests failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
log.Entry().Info("run ABAP Unit Test finished")
|
|
return response, nil
|
|
}
|
|
|
|
func parseUnitResult(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, aUnitRunResult *runResult) (parsedResult checkstyle, err error) {
|
|
|
|
log.Entry().Info("parse ABAP Unit Result started")
|
|
|
|
var fileName string
|
|
var aUnitFile file
|
|
var aUnitError checkstyleError
|
|
|
|
parsedResult.Version = "1.0"
|
|
|
|
for _, program := range aUnitRunResult.Program {
|
|
|
|
objectType := program.Type[0:4]
|
|
objectName := program.Name
|
|
|
|
//syntax error in unit test or class
|
|
if program.Alerts.Alert.HasSyntaxErrors == "true" {
|
|
|
|
aUnitFailure = true
|
|
aUnitError.Source = objectName
|
|
aUnitError.Severity = "error"
|
|
log.Entry().Info("severity: ", aUnitError.Severity)
|
|
aUnitError.Message = html.UnescapeString(program.Alerts.Alert.Title + " " + program.Alerts.Alert.Details.Detail.AttrText)
|
|
log.Entry().Info("message: ", aUnitError.Message)
|
|
aUnitError.Line, err = findLine(config, client, program.Alerts.Alert.Stack.StackEntry.URI, objectName, objectType)
|
|
log.Entry().Error("line: ", aUnitError.Line)
|
|
if err != nil {
|
|
return parsedResult, errors.Wrap(err, "parse AUnit Result failed")
|
|
|
|
}
|
|
fileName, err = getFileName(config, client, program.Alerts.Alert.Stack.StackEntry.URI, objectName)
|
|
log.Entry().Error("file path: ", aUnitError.Line)
|
|
if err != nil {
|
|
return parsedResult, errors.Wrap(err, "parse AUnit Result failed")
|
|
|
|
}
|
|
|
|
aUnitFile.Error = append(aUnitFile.Error, aUnitError)
|
|
aUnitError = checkstyleError{}
|
|
log.Entry().Error("there is a syntax error", aUnitFile)
|
|
}
|
|
|
|
for _, testClass := range program.TestClasses.TestClass {
|
|
|
|
for _, testMethod := range testClass.TestMethods.TestMethod {
|
|
|
|
aUnitError.Source = testClass.Name + "/" + testMethod.Name
|
|
|
|
// unit test failure
|
|
if len(testMethod.Alerts.Alert) > 0 {
|
|
|
|
for _, testalert := range testMethod.Alerts.Alert {
|
|
|
|
switch testalert.Severity {
|
|
case "fatal":
|
|
log.Entry().Error("unit test " + aUnitError.Source + " has failed with severity fatal")
|
|
aUnitFailure = true
|
|
aUnitError.Severity = "error"
|
|
case "critical":
|
|
log.Entry().Error("unit test " + aUnitError.Source + " has failed with severity critical")
|
|
aUnitFailure = true
|
|
aUnitError.Severity = "error"
|
|
case "tolerable":
|
|
log.Entry().Warning("unit test " + aUnitError.Source + " has failed with severity warning")
|
|
aUnitError.Severity = "warning"
|
|
default:
|
|
aUnitError.Severity = "info"
|
|
|
|
}
|
|
|
|
//unit test message is spread in different elements
|
|
for _, detail := range testalert.Details.Detail {
|
|
aUnitError.Message = aUnitError.Message + " " + detail.AttrText
|
|
for _, subdetail := range detail.Details.Detail {
|
|
|
|
aUnitError.Message = html.UnescapeString(aUnitError.Message + " " + subdetail.AttrText)
|
|
log.Entry().Info("message: ", aUnitError.Message)
|
|
}
|
|
|
|
}
|
|
|
|
aUnitError.Line, err = findLine(config, client, testalert.Stack.StackEntry.URI, objectName, objectType)
|
|
log.Entry().Info("line: ", aUnitError.Line)
|
|
if err != nil {
|
|
|
|
log.Entry().Warning(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
aUnitFile.Error = append(aUnitFile.Error, aUnitError)
|
|
aUnitError = checkstyleError{}
|
|
|
|
} else {
|
|
|
|
log.Entry().Info("unit test:", aUnitError.Source, "- was successful")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fileName, err = getFileName(config, client, testClass.URI, objectName)
|
|
if err != nil {
|
|
return parsedResult, errors.Wrap(err, "parse AUnit Result failed")
|
|
|
|
}
|
|
}
|
|
|
|
aUnitFile.Name, err = constructPath(config, client, fileName, objectName, objectType)
|
|
log.Entry().Error("file path: ", aUnitFile.Name)
|
|
if err != nil {
|
|
|
|
return parsedResult, errors.Wrap(err, "parse AUnit Result failed")
|
|
}
|
|
parsedResult.File = append(parsedResult.File, aUnitFile)
|
|
aUnitFile = file{}
|
|
|
|
}
|
|
|
|
body, _ := xml.Marshal(parsedResult)
|
|
|
|
writeErr := ioutil.WriteFile(config.AUnitResultsFileName, body, 0644)
|
|
|
|
if writeErr != nil {
|
|
log.Entry().Error("file AUnitResults.xml could not be created")
|
|
return parsedResult, fmt.Errorf("handling unit test results failed: %w", writeErr)
|
|
}
|
|
|
|
log.Entry().Info("parse ABAP Unit Result finished")
|
|
return parsedResult, nil
|
|
|
|
}
|
|
|
|
func executeATCCheck(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, objects []repoObject) (error error) {
|
|
|
|
log.Entry().Info("execute ATC Check started")
|
|
|
|
var innerXml string
|
|
var result worklist
|
|
|
|
for _, object := range objects {
|
|
|
|
switch object.Type {
|
|
|
|
case "CLAS":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/oo/classes/` + object.Object + `"/>`
|
|
case "INTF":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/oo/interfaces/` + object.Object + `"/>`
|
|
case "DEVC":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/packages/` + object.Object + `"/>`
|
|
case "FUGR":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/functions/groups/` + object.Object + `/source/main"/>`
|
|
case "TABL":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/ddic/tables/` + object.Object + `/source/main"/>`
|
|
case "DTEL":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/ddic/dataelements/` + object.Object + `"/>`
|
|
case "DOMA":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/ddic/domains/` + object.Object + `"/>`
|
|
case "MSAG":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/messageclass/` + object.Object + `"/>`
|
|
case "PROG":
|
|
innerXml = innerXml + `<adtcore:objectReference adtcore:uri="/sap/bc/adt/programs/programs/` + object.Object + `/source/main"/>`
|
|
default:
|
|
log.Entry().Warning("object Type " + object.Type + " is not supported!")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var xmlBody = []byte(`<?xml version="1.0" encoding="UTF-8"?>
|
|
<atc:run xmlns:atc="http://www.sap.com/adt/atc"
|
|
maximumVerdicts="100">
|
|
<objectSets xmlns:adtcore="http://www.sap.com/adt/core">
|
|
<objectSet kind="inclusive">
|
|
<adtcore:objectReferences>` + innerXml +
|
|
`</adtcore:objectReferences>
|
|
</objectSet>
|
|
</objectSets>
|
|
</atc:run>`)
|
|
|
|
worklist, err := getWorklist(config, client)
|
|
if err != nil {
|
|
return errors.Wrap(err, "execution of ATC Checks failed")
|
|
}
|
|
|
|
err = startATCRun(config, client, xmlBody, worklist)
|
|
|
|
if err != nil {
|
|
return errors.Wrap(err, "execution of ATC Checks failed")
|
|
}
|
|
|
|
resp, err := getATCRun(config, client, worklist)
|
|
|
|
if err != nil {
|
|
return errors.Wrap(err, "execution of ATC Checks failed")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyXML(resp, &result)
|
|
if parsingErr != nil {
|
|
log.Entry().Warning(parsingErr)
|
|
return nil
|
|
}
|
|
|
|
atcRes, err := parseATCCheckResult(config, client, &result)
|
|
|
|
if err != nil {
|
|
log.Entry().Error(err)
|
|
return errors.Wrap(err, "execution of ATC Checks failed")
|
|
}
|
|
|
|
log.Entry().Info("execute ATC Checks finished.", atcRes.Text)
|
|
|
|
return nil
|
|
|
|
}
|
|
func startATCRun(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, xml []byte, worklistID string) (err error) {
|
|
|
|
log.Entry().Info("ATC Run started")
|
|
|
|
discHeader, discError := discoverServer(config, client)
|
|
if discError != nil {
|
|
return errors.Wrap(discError, "start of ATC run failed")
|
|
}
|
|
|
|
if discHeader.Get("X-Csrf-Token") == "" {
|
|
return errors.Errorf("could not retrieve x-csrf-token from server")
|
|
}
|
|
|
|
header := make(http.Header)
|
|
header.Add("x-csrf-token", discHeader.Get("X-Csrf-Token"))
|
|
header.Add("Accept", "application/xml")
|
|
|
|
url := config.Host +
|
|
"/sap/bc/adt/atc/runs?worklistId=" + worklistID + "&sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("POST", url, bytes.NewBuffer(xml), header, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return errors.Wrap(httpErr, "start of ATC run failed")
|
|
} else if resp == nil {
|
|
return errors.New("start of ATC run failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
log.Entry().Info("ATC Run finished")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func getATCRun(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, worklistID string) (response *http.Response, err error) {
|
|
|
|
log.Entry().Info("get ATC Run Results started")
|
|
|
|
header := make(http.Header)
|
|
|
|
url := config.Host +
|
|
"/sap/bc/adt/atc/worklists/" + worklistID + "?sap-client=" + config.Client
|
|
|
|
header.Add("Accept", "application/atc.worklist.v1+xml")
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, header, nil)
|
|
|
|
if httpErr != nil {
|
|
return response, errors.Wrap(httpErr, "get ATC run failed")
|
|
} else if resp == nil {
|
|
return response, errors.New("get ATC run failed: did not retrieve a HTTP response")
|
|
}
|
|
log.Entry().Info("get ATC Run Results finished")
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
func getWorklist(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (worklistID string, error error) {
|
|
|
|
url := config.Host +
|
|
"/sap/bc/adt/atc/worklists?checkVariant=" + config.AtcVariant + "&sap-client=" + config.Client
|
|
discHeader, discError := discoverServer(config, client)
|
|
|
|
if discError != nil {
|
|
return worklistID, errors.Wrap(discError, "get worklist failed")
|
|
}
|
|
|
|
if discHeader.Get("X-Csrf-Token") == "" {
|
|
return worklistID, errors.Errorf("could not retrieve x-csrf-token from server")
|
|
}
|
|
|
|
header := make(http.Header)
|
|
header.Add("x-csrf-token", discHeader.Get("X-Csrf-Token"))
|
|
header.Add("Accept", "*/*")
|
|
|
|
resp, httpErr := client.SendRequest("POST", url, nil, header, nil)
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return worklistID, errors.Wrap(httpErr, "get worklist failed")
|
|
} else if resp == nil {
|
|
return worklistID, errors.New("get worklist failed: did not retrieve a HTTP response")
|
|
}
|
|
location := resp.Header["Location"][0]
|
|
locationSlice := strings.Split(location, "/")
|
|
worklistID = locationSlice[len(locationSlice)-1]
|
|
log.Entry().Info("worklist id for ATC check: ", worklistID)
|
|
|
|
return worklistID, nil
|
|
}
|
|
|
|
func parseATCCheckResult(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, response *worklist) (atcResults checkstyle, error error) {
|
|
|
|
log.Entry().Info("parse ATC Check Result started")
|
|
|
|
var atcFile file
|
|
var subObject string
|
|
var aTCUnitError checkstyleError
|
|
|
|
atcResults.Version = "1.0"
|
|
|
|
for _, object := range response.Objects.Object {
|
|
|
|
objectType := object.Type
|
|
objectName := object.Name
|
|
|
|
for _, atcworklist := range object.Findings.Finding {
|
|
|
|
log.Entry().Info("there is atc finding for object type: ", objectType+" object name: "+objectName)
|
|
|
|
path, err := url.PathUnescape(atcworklist.Location)
|
|
|
|
if err != nil {
|
|
return atcResults, errors.Wrap(err, "conversion of ATC check results to CheckStyle has failed")
|
|
|
|
}
|
|
|
|
if len(atcworklist.Atcfinding) > 0 {
|
|
|
|
priority, err := strconv.Atoi(atcworklist.Priority)
|
|
|
|
if err != nil {
|
|
return atcResults, errors.Wrap(err, "conversion of ATC check results to CheckStyle has failed")
|
|
|
|
}
|
|
|
|
switch priority {
|
|
case 1:
|
|
atcFailure = true
|
|
aTCUnitError.Severity = "error"
|
|
log.Entry().Error("atc issue with priority: 1 ")
|
|
case 2:
|
|
atcFailure = true
|
|
aTCUnitError.Severity = "error"
|
|
log.Entry().Error("atc issue with priority: 2 ")
|
|
case 3:
|
|
aTCUnitError.Severity = "warning"
|
|
log.Entry().Warning("atc issue with priority: 3 ")
|
|
default:
|
|
aTCUnitError.Severity = "info"
|
|
log.Entry().Info("atc issue with low priority ")
|
|
}
|
|
|
|
log.Entry().Error("severity: ", aTCUnitError.Severity)
|
|
|
|
if aTCUnitError.Line == "" {
|
|
|
|
aTCUnitError.Line, err = findLine(config, client, path, objectName, objectType)
|
|
log.Entry().Info("line: ", aTCUnitError.Line)
|
|
|
|
if err != nil {
|
|
log.Entry().Info(path)
|
|
log.Entry().Warning(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if subObject != "" {
|
|
aTCUnitError.Source = objectName + "/" + strings.ToUpper(subObject)
|
|
} else {
|
|
aTCUnitError.Source = objectName
|
|
}
|
|
|
|
aTCUnitError.Message = html.UnescapeString(atcworklist.CheckTitle + " " + atcworklist.MessageTitle)
|
|
log.Entry().Info("message: ", aTCUnitError.Message)
|
|
atcFile.Error = append(atcFile.Error, aTCUnitError)
|
|
aTCUnitError = checkstyleError{}
|
|
}
|
|
|
|
if atcFile.Error[0].Message != "" {
|
|
|
|
fileName, err := getFileName(config, client, path, objectName)
|
|
|
|
if err != nil {
|
|
return atcResults, errors.Wrap(err, "conversion of ATC check results to CheckStyle has failed")
|
|
}
|
|
|
|
atcFile.Name, err = constructPath(config, client, fileName, objectName, objectType)
|
|
log.Entry().Info("file path: ", atcFile.Name)
|
|
if err != nil {
|
|
return atcResults, errors.Wrap(err, "conversion of ATC check results to CheckStyle has failed")
|
|
}
|
|
atcResults.File = append(atcResults.File, atcFile)
|
|
atcFile = file{}
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
atcBody, _ := xml.Marshal(atcResults)
|
|
|
|
writeErr := ioutil.WriteFile(config.AtcResultsFileName, atcBody, 0644)
|
|
|
|
if writeErr != nil {
|
|
log.Entry().Error("ATCResults.xml could not be created")
|
|
return atcResults, fmt.Errorf("handling atc results failed: %w", writeErr)
|
|
}
|
|
log.Entry().Info("parsing ATC check results to CheckStyle has finished.")
|
|
return atcResults, writeErr
|
|
}
|
|
|
|
func constructPath(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, fileName string, objectName string, objectType string) (filePath string, error error) {
|
|
|
|
targetDir, err := getTargetDir(config, client)
|
|
if err != nil {
|
|
return filePath, errors.Wrap(err, "path could not be constructed")
|
|
|
|
}
|
|
|
|
filePath = config.Workspace + "/" + targetDir + "/objects/" + strings.ToUpper(objectType) + "/" + strings.ToUpper(objectName) + "/" + fileName
|
|
return filePath, nil
|
|
|
|
}
|
|
|
|
func findLine(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, path string, objectName string, objectType string) (line string, error error) {
|
|
|
|
regexLine := regexp.MustCompile(`.start=\d*`)
|
|
regexMethod := regexp.MustCompile(`.name=[a-zA-Z0-9_-]*;`)
|
|
|
|
readableSource, err := checkReadableSource(config, client)
|
|
|
|
if err != nil {
|
|
|
|
return line, errors.Wrap(err, "could not find line in source code")
|
|
|
|
}
|
|
|
|
fileName, err := getFileName(config, client, path, objectName)
|
|
|
|
if err != nil {
|
|
|
|
return line, err
|
|
|
|
}
|
|
|
|
filePath, err := constructPath(config, client, fileName, objectName, objectType)
|
|
if err != nil {
|
|
return line, errors.Wrap(err, objectType+"/"+objectName+"could not find line in source code")
|
|
|
|
}
|
|
|
|
var absLine int
|
|
if readableSource {
|
|
|
|
// the error line that we get from UnitTest Run or ATC Check is not aligned for the readable source, we need to calculated it
|
|
rawfile, err := ioutil.ReadFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return line, errors.Wrapf(err, "could not find object in the workspace of your CI/CD tool ")
|
|
}
|
|
|
|
file := string(rawfile)
|
|
|
|
splittedfile := strings.Split(file, "\n")
|
|
|
|
// CLAS/OSO - is unique identifier for protection section in CLAS
|
|
if strings.Contains(path, "CLAS/OSO") {
|
|
|
|
for l, line := range splittedfile {
|
|
|
|
if strings.Contains(line, "protected section.") {
|
|
absLine = l
|
|
break
|
|
}
|
|
|
|
}
|
|
|
|
// CLAS/OM - is unique identifier for method section in CLAS
|
|
} else if strings.Contains(path, "CLAS/OM") {
|
|
|
|
methodName := regexMethod.FindString(path)
|
|
|
|
if methodName != "" {
|
|
methodName = methodName[len(`.name=`) : len(methodName)-1]
|
|
|
|
}
|
|
|
|
for line, linecontent := range splittedfile {
|
|
|
|
if strings.Contains(linecontent, "method"+" "+methodName) {
|
|
absLine = line
|
|
break
|
|
}
|
|
|
|
}
|
|
|
|
// CLAS/OSI - is unique identifier for private section in CLAS
|
|
} else if strings.Contains(path, "CLAS/OSI") {
|
|
|
|
for line, linecontent := range splittedfile {
|
|
|
|
if strings.Contains(linecontent, "private section.") {
|
|
absLine = line
|
|
break
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
errLine := regexLine.FindString(path)
|
|
|
|
if errLine != "" {
|
|
|
|
errLine, err := strconv.Atoi(errLine[len(`.start=`):])
|
|
if err == nil {
|
|
line = strconv.Itoa(absLine + errLine)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
// classic format
|
|
errLine := regexLine.FindString(path)
|
|
if errLine != "" {
|
|
line = errLine[len(`.start=`):]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return line, nil
|
|
}
|
|
func getFileName(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, path string, objName string) (fileName string, error error) {
|
|
|
|
readableSource, err := checkReadableSource(config, client)
|
|
if err != nil {
|
|
return fileName, errors.Wrap(err, "get file name has failed")
|
|
|
|
}
|
|
|
|
path, err = url.PathUnescape(path)
|
|
|
|
var fileExtension string
|
|
fileExtensionLength := 30 - len(objName)
|
|
for i := 0; i < fileExtensionLength; i++ {
|
|
fileExtension += "="
|
|
}
|
|
|
|
if err != nil {
|
|
return fileName, errors.Wrap(err, "get file name has failed")
|
|
|
|
}
|
|
|
|
// INTERFACES
|
|
regexInterface := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/interfaces\/\w*`)
|
|
intf := regexInterface.FindString(path)
|
|
if intf != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".intf.abap"
|
|
} else {
|
|
fileName = "REPS " + strings.ToUpper(objName) + fileExtension + "IU.abap"
|
|
}
|
|
|
|
}
|
|
// CLASSES DEFINITIONS
|
|
regexClasDef := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/includes\/definitions\/`)
|
|
clasDef := regexClasDef.FindString(path)
|
|
if clasDef != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.definitions.abap"
|
|
} else {
|
|
fileName = "CINC " + objName + fileExtension + "CCDEF.abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASSES IMPLEMENTATIONS
|
|
regexClasImpl := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/includes\/implementations\/`)
|
|
clasImpl := regexClasImpl.FindString(path)
|
|
if clasImpl != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.implementations.abap"
|
|
} else {
|
|
fileName = "CINC " + objName + fileExtension + "CCIMP.abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASSES MACROS
|
|
regexClasMacro := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/includes\/macros\/`)
|
|
clasMacro := regexClasMacro.FindString(path)
|
|
if clasMacro != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.macros.abap"
|
|
} else {
|
|
fileName = "CINC " + objName + fileExtension + "CCMAC.abap"
|
|
}
|
|
|
|
}
|
|
|
|
// TEST CLASSES
|
|
regexTestClass := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*#?\/?\w*\/?testclass`)
|
|
testClass := regexTestClass.FindString(path)
|
|
if testClass != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.testclasses.abap"
|
|
} else {
|
|
fileName = "CINC " + objName + fileExtension + "CCAU.abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASS PROTECTED
|
|
regexClasProtected := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/source\/main#type=CLAS\/OSO`)
|
|
classProtected := regexClasProtected.FindString(path)
|
|
if classProtected != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.abap"
|
|
} else {
|
|
fileName = "CPRO " + objName + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASS PRIVATE
|
|
regexClasPrivate := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/source\/main#type=CLAS\/OSI`)
|
|
classPrivate := regexClasPrivate.FindString(path)
|
|
if classPrivate != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.abap"
|
|
} else {
|
|
fileName = "CPRI " + objName + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASS METHOD
|
|
regexClasMethod := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/source\/main#type=CLAS\/OM`)
|
|
classMethod := regexClasMethod.FindString(path)
|
|
if classMethod != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.abap"
|
|
} else {
|
|
|
|
regexmethodName := regexp.MustCompile(`name=\w*`)
|
|
methodName := regexmethodName.FindString(path)
|
|
|
|
fileName = "METH " + methodName[len(`name=`):] + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// CLASS PUBLIC
|
|
regexClasPublic := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/\w*\/source\/main#start`)
|
|
classPublic := regexClasPublic.FindString(path)
|
|
if classPublic != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.abap"
|
|
} else {
|
|
fileName = "CPUB " + objName + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// FUNCTION INCLUDE
|
|
regexFuncIncl := regexp.MustCompile(`\/sap\/bc\/adt\/functions\/groups\/\w*\/includes/\w*`)
|
|
|
|
funcIncl := regexFuncIncl.FindString(path)
|
|
if funcIncl != "" && fileName == "" {
|
|
|
|
regexSubObj := regexp.MustCompile(`includes\/\w*`)
|
|
subObject := regexSubObj.FindString(path)
|
|
subObject = subObject[len(`includes/`):]
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".fugr." + strings.ToLower(subObject) + ".reps.abap"
|
|
} else {
|
|
fileName = "REPS " + strings.ToUpper(subObject) + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// FUNCTION GROUP
|
|
regexFuncGr := regexp.MustCompile(`\/sap\/bc\/adt\/functions\/groups\/\w*\/source\/main`)
|
|
|
|
funcGr := regexFuncGr.FindString(path)
|
|
if funcGr != "" && fileName == "" {
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".fugr.sapl" + strings.ToLower(objName) + ".reps.abap"
|
|
} else {
|
|
fileName = "REPS SAPL" + objName + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// FUNCTION MODULE
|
|
regexFuncMod := regexp.MustCompile(`\/sap\/bc\/adt\/functions\/groups\/\w*\/fmodules/\w*`)
|
|
funcMod := regexFuncMod.FindString(path)
|
|
if funcMod != "" && fileName == "" {
|
|
|
|
regexSubObj := regexp.MustCompile(`includes\/\w*`)
|
|
subObject := regexSubObj.FindString(path)
|
|
subObject = subObject[len(`includes/`):]
|
|
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(subObject) + ".func.abap"
|
|
} else {
|
|
fileName = "FUNC " + subObject + ".abap"
|
|
}
|
|
|
|
}
|
|
// CLAS
|
|
regexClas := regexp.MustCompile(`\/sap\/bc\/adt\/oo\/classes\/` + strings.ToLower(objName))
|
|
clas := regexClas.FindString(path)
|
|
if clas != "" && fileName == "" {
|
|
if readableSource {
|
|
|
|
fileName = strings.ToLower(objName) + ".clas.abap"
|
|
} else {
|
|
|
|
fileName = "CPUB " + objName + ".abap"
|
|
}
|
|
|
|
}
|
|
|
|
// PROGRAM
|
|
regexProg := regexp.MustCompile(`\/sap\/bc\/adt\/programs\/programs\/` + strings.ToLower(objName))
|
|
prog := regexProg.FindString(path)
|
|
if prog != "" && fileName == "" {
|
|
|
|
fileName = "REPS " + objName + ".abap"
|
|
|
|
}
|
|
|
|
// TABLES
|
|
regexTab := regexp.MustCompile(`\/sap\/bc\/adt\/ddic\/tables\/` + strings.ToLower(objName))
|
|
tab := regexTab.FindString(path)
|
|
if tab != "" && fileName == "" {
|
|
|
|
fileName = "TABL " + objName + ".asx.json"
|
|
|
|
}
|
|
|
|
return fileName, nil
|
|
|
|
}
|
|
|
|
func getTargetDir(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (string, error) {
|
|
|
|
var targetDir string
|
|
|
|
repository, err := getRepo(config, client)
|
|
|
|
if err != nil {
|
|
return targetDir, err
|
|
}
|
|
|
|
for _, config := range repository.Result.Config {
|
|
if config.Key == "VCS_TARGET_DIR" {
|
|
targetDir = config.Value
|
|
}
|
|
}
|
|
|
|
return targetDir, nil
|
|
|
|
}
|
|
|
|
func checkReadableSource(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (readableSource bool, error error) {
|
|
|
|
repoLayout, err := getRepositoryLayout(config, client)
|
|
if err != nil {
|
|
return readableSource, errors.Wrap(err, "could not check readable source format")
|
|
}
|
|
|
|
if repoLayout.Layout.ReadableSource == "true" || repoLayout.Layout.ReadableSource == "only" || repoLayout.Layout.ReadableSource == "all" {
|
|
|
|
readableSource = true
|
|
|
|
} else {
|
|
|
|
readableSource = false
|
|
|
|
}
|
|
|
|
return readableSource, nil
|
|
}
|
|
|
|
func getRepo(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (repositoryResponse, error) {
|
|
|
|
var repositoryResp repositoryResponse
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"?sap-client=" + config.Client
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return repositoryResponse{}, errors.Wrap(httpErr, "could not get repository")
|
|
} else if resp == nil {
|
|
return repositoryResponse{}, errors.New("could not get repository: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &repositoryResp)
|
|
if parsingErr != nil {
|
|
return repositoryResponse{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
|
|
return repositoryResp, nil
|
|
|
|
}
|
|
|
|
func getRepositoryLayout(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (layoutResponse, error) {
|
|
|
|
var repoLayoutResponse layoutResponse
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"/layout?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return layoutResponse{}, errors.Wrap(httpErr, "could not get repository layout")
|
|
} else if resp == nil {
|
|
return layoutResponse{}, errors.New("could not get repository layout: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &repoLayoutResponse)
|
|
if parsingErr != nil {
|
|
return layoutResponse{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
|
|
return repoLayoutResponse, nil
|
|
}
|
|
|
|
func getCommitList(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (commitResponse, error) {
|
|
|
|
var commitResp commitResponse
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"/getCommit?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return commitResponse{}, errors.Wrap(httpErr, "get repository history failed")
|
|
} else if resp == nil {
|
|
return commitResponse{}, errors.New("get repository history failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &commitResp)
|
|
if parsingErr != nil {
|
|
return commitResponse{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
|
|
return commitResp, nil
|
|
}
|
|
|
|
func getObjectDifference(config *gctsExecuteABAPQualityChecksOptions, fromCommit string, toCommit string, client piperhttp.Sender) (objectsResponse, error) {
|
|
var objectResponse objectsResponse
|
|
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository +
|
|
"/compareCommits?fromCommit=" + fromCommit + "&toCommit=" + toCommit + "&sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return objectsResponse{}, errors.Wrap(httpErr, "get object difference failed")
|
|
} else if resp == nil {
|
|
return objectsResponse{}, errors.New("get object difference failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &objectResponse)
|
|
if parsingErr != nil {
|
|
return objectsResponse{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
log.Entry().Info("get object differences: ", objectResponse.Objects)
|
|
return objectResponse, nil
|
|
}
|
|
|
|
func getObjectInfo(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender, objectName string, objectType string) (objectInfo, error) {
|
|
|
|
var objectMetInfoResponse objectInfo
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/objects/" + objectType + "/" + objectName +
|
|
"?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
|
|
if httpErr != nil {
|
|
return objectInfo{}, errors.Wrap(httpErr, "resolve package failed")
|
|
} else if resp == nil {
|
|
return objectInfo{}, errors.New("resolve package failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &objectMetInfoResponse)
|
|
if parsingErr != nil {
|
|
return objectInfo{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
return objectMetInfoResponse, nil
|
|
|
|
}
|
|
|
|
func getHistory(config *gctsExecuteABAPQualityChecksOptions, client piperhttp.Sender) (historyResponse, error) {
|
|
|
|
var historyResp historyResponse
|
|
url := config.Host +
|
|
"/sap/bc/cts_abapvcs/repository/" + config.Repository + "/getHistory?sap-client=" + config.Client
|
|
|
|
resp, httpErr := client.SendRequest("GET", url, nil, nil, nil)
|
|
|
|
defer func() {
|
|
if resp != nil && resp.Body != nil {
|
|
resp.Body.Close()
|
|
}
|
|
}()
|
|
if httpErr != nil {
|
|
return historyResponse{}, errors.Wrap(httpErr, "get history failed")
|
|
} else if resp == nil {
|
|
return historyResponse{}, errors.New("get history failed: did not retrieve a HTTP response")
|
|
}
|
|
|
|
parsingErr := piperhttp.ParseHTTPResponseBodyJSON(resp, &historyResp)
|
|
if parsingErr != nil {
|
|
return historyResponse{}, errors.Errorf("%v", parsingErr)
|
|
}
|
|
|
|
return historyResp, nil
|
|
}
|
|
|
|
type worklist struct {
|
|
XMLName xml.Name `xml:"worklist"`
|
|
Text string `xml:",chardata"`
|
|
ID string `xml:"id,attr"`
|
|
Timestamp string `xml:"timestamp,attr"`
|
|
UsedObjectSet string `xml:"usedObjectSet,attr"`
|
|
ObjectSetIsComplete string `xml:"objectSetIsComplete,attr"`
|
|
Atcworklist string `xml:"atcworklist,attr"`
|
|
ObjectSets struct {
|
|
Text string `xml:",chardata"`
|
|
ObjectSet []struct {
|
|
Text string `xml:",chardata"`
|
|
Name string `xml:"name,attr"`
|
|
Title string `xml:"title,attr"`
|
|
Kind string `xml:"kind,attr"`
|
|
} `xml:"objectSet"`
|
|
} `xml:"objectSets"`
|
|
Objects struct {
|
|
Text string `xml:",chardata"`
|
|
Object []struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Name string `xml:"name,attr"`
|
|
PackageName string `xml:"packageName,attr"`
|
|
Author string `xml:"author,attr"`
|
|
Atcobject string `xml:"atcobject,attr"`
|
|
Adtcore string `xml:"adtcore,attr"`
|
|
Findings struct {
|
|
Text string `xml:",chardata"`
|
|
Finding []struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Location string `xml:"location,attr"`
|
|
Processor string `xml:"processor,attr"`
|
|
LastChangedBy string `xml:"lastChangedBy,attr"`
|
|
Priority string `xml:"priority,attr"`
|
|
CheckId string `xml:"checkId,attr"`
|
|
CheckTitle string `xml:"checkTitle,attr"`
|
|
MessageId string `xml:"messageId,attr"`
|
|
MessageTitle string `xml:"messageTitle,attr"`
|
|
ExemptionApproval string `xml:"exemptionApproval,attr"`
|
|
ExemptionKind string `xml:"exemptionKind,attr"`
|
|
Checksum string `xml:"checksum,attr"`
|
|
QuickfixInfo string `xml:"quickfixInfo,attr"`
|
|
Atcfinding string `xml:"atcfinding,attr"`
|
|
Link struct {
|
|
Text string `xml:",chardata"`
|
|
Href string `xml:"href,attr"`
|
|
Rel string `xml:"rel,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Atom string `xml:"atom,attr"`
|
|
} `xml:"link"`
|
|
Quickfixes struct {
|
|
Text string `xml:",chardata"`
|
|
Manual string `xml:"manual,attr"`
|
|
Automatic string `xml:"automatic,attr"`
|
|
Pseudo string `xml:"pseudo,attr"`
|
|
} `xml:"quickfixes"`
|
|
} `xml:"finding"`
|
|
} `xml:"findings"`
|
|
} `xml:"object"`
|
|
} `xml:"objects"`
|
|
}
|
|
|
|
type runResult struct {
|
|
XMLName xml.Name `xml:"runResult"`
|
|
Text string `xml:",chardata"`
|
|
Aunit string `xml:"aunit,attr"`
|
|
Program []struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Name string `xml:"name,attr"`
|
|
URIType string `xml:"uriType,attr"`
|
|
Adtcore string `xml:"adtcore,attr"`
|
|
Alerts struct {
|
|
Text string `xml:",chardata"`
|
|
Alert struct {
|
|
Text string `xml:",chardata"`
|
|
HasSyntaxErrors string `xml:"hasSyntaxErrors,attr"`
|
|
Kind string `xml:"kind,attr"`
|
|
Severity string `xml:"severity,attr"`
|
|
Title string `xml:"title"`
|
|
Details struct {
|
|
Text string `xml:",chardata"`
|
|
Detail struct {
|
|
Text string `xml:",chardata"`
|
|
AttrText string `xml:"text,attr"`
|
|
} `xml:"detail"`
|
|
} `xml:"details"`
|
|
Stack struct {
|
|
Text string `xml:",chardata"`
|
|
StackEntry struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Description string `xml:"description,attr"`
|
|
} `xml:"stackEntry"`
|
|
} `xml:"stack"`
|
|
} `xml:"alert"`
|
|
} `xml:"alerts"`
|
|
|
|
TestClasses struct {
|
|
Text string `xml:",chardata"`
|
|
TestClass []struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Name string `xml:"name,attr"`
|
|
URIType string `xml:"uriType,attr"`
|
|
NavigationURI string `xml:"navigationUri,attr"`
|
|
DurationCategory string `xml:"durationCategory,attr"`
|
|
RiskLevel string `xml:"riskLevel,attr"`
|
|
TestMethods struct {
|
|
Text string `xml:",chardata"`
|
|
TestMethod []struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Name string `xml:"name,attr"`
|
|
ExecutionTime string `xml:"executionTime,attr"`
|
|
URIType string `xml:"uriType,attr"`
|
|
NavigationURI string `xml:"navigationUri,attr"`
|
|
Unit string `xml:"unit,attr"`
|
|
Alerts struct {
|
|
Text string `xml:",chardata"`
|
|
Alert []struct {
|
|
Text string `xml:",chardata"`
|
|
Kind string `xml:"kind,attr"`
|
|
Severity string `xml:"severity,attr"`
|
|
Title string `xml:"title"`
|
|
Details struct {
|
|
Text string `xml:",chardata"`
|
|
Detail []struct {
|
|
Text string `xml:",chardata"`
|
|
AttrText string `xml:"text,attr"`
|
|
Details struct {
|
|
Text string `xml:",chardata"`
|
|
Detail []struct {
|
|
Text string `xml:",chardata"`
|
|
AttrText string `xml:"text,attr"`
|
|
} `xml:"detail"`
|
|
} `xml:"details"`
|
|
} `xml:"detail"`
|
|
} `xml:"details"`
|
|
Stack struct {
|
|
Text string `xml:",chardata"`
|
|
StackEntry struct {
|
|
Text string `xml:",chardata"`
|
|
URI string `xml:"uri,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Name string `xml:"name,attr"`
|
|
Description string `xml:"description,attr"`
|
|
} `xml:"stackEntry"`
|
|
} `xml:"stack"`
|
|
} `xml:"alert"`
|
|
} `xml:"alerts"`
|
|
} `xml:"testMethod"`
|
|
} `xml:"testMethods"`
|
|
} `xml:"testClass"`
|
|
} `xml:"testClasses"`
|
|
} `xml:"program"`
|
|
}
|
|
|
|
type gctsException struct {
|
|
Message string `json:"message"`
|
|
Description string `json:"description"`
|
|
Code int `json:"code"`
|
|
}
|
|
|
|
type gctsLogs struct {
|
|
Time int `json:"time"`
|
|
User string `json:"user"`
|
|
Section string `json:"section"`
|
|
Action string `json:"action"`
|
|
Severity string `json:"severity"`
|
|
Message string `json:"message"`
|
|
Code string `json:"code"`
|
|
}
|
|
|
|
type commit struct {
|
|
ID string `json:"id"`
|
|
}
|
|
|
|
type commitResponse struct {
|
|
Commits []commit `json:"commits"`
|
|
ErrorLog []gctsLogs `json:"errorLog"`
|
|
Log []gctsLogs `json:"log"`
|
|
Exception gctsException `json:"exception"`
|
|
}
|
|
|
|
type objectInfo struct {
|
|
Pgmid string `json:"pgmid"`
|
|
Object string `json:"object"`
|
|
ObjName string `json:"objName"`
|
|
Srcsystem string `json:"srcsystem"`
|
|
Author string `json:"author"`
|
|
Devclass string `json:"devclass"`
|
|
}
|
|
|
|
type repoConfig struct {
|
|
Key string `json:"key"`
|
|
Value string `json:"value"`
|
|
Cprivate string `json:"cprivate"`
|
|
Cprotected string `json:"cprotected"`
|
|
Cvisible string `json:"cvisible"`
|
|
Category string `json:"category"`
|
|
Scope string `json:"scope"`
|
|
ChangedAt float64 `json:"changeAt"`
|
|
ChangedBy string `json:"changedBy"`
|
|
}
|
|
|
|
type repository struct {
|
|
Rid string `json:"rid"`
|
|
Name string `json:"name"`
|
|
Role string `json:"role"`
|
|
Type string `json:"type"`
|
|
Vsid string `json:"vsid"`
|
|
PrivateFlag string `json:"privateFlag"`
|
|
Status string `json:"status"`
|
|
Branch string `json:"branch"`
|
|
Url string `json:"url"`
|
|
CreatedBy string `json:"createdBy"`
|
|
CreatedDate string `json:"createdDate"`
|
|
Config []repoConfig `json:"config"`
|
|
Objects int `json:"objects"`
|
|
CurrentCommit string `json:"currentCommit"`
|
|
}
|
|
|
|
type repositoryResponse struct {
|
|
Result repository `json:"result"`
|
|
Exception gctsException `json:"exception"`
|
|
}
|
|
|
|
type objects struct {
|
|
Name string `json:"name"`
|
|
Type string `json:"type"`
|
|
Action string `json:"action"`
|
|
}
|
|
type objectsResponse struct {
|
|
Objects []objects `json:"objects"`
|
|
Log []gctsLogs `json:"log"`
|
|
Exception gctsException `json:"exception"`
|
|
ErrorLogs []gctsLogs `json:"errorLog"`
|
|
}
|
|
|
|
type repoObject struct {
|
|
Pgmid string `json:"pgmid"`
|
|
Object string `json:"object"`
|
|
Type string `json:"type"`
|
|
Description string `json:"description"`
|
|
}
|
|
|
|
type repoObjectResponse struct {
|
|
Objects []repoObject `json:"objects"`
|
|
Log []gctsLogs `json:"log"`
|
|
Exception gctsException `json:"exception"`
|
|
ErrorLogs []gctsLogs `json:"errorLog"`
|
|
}
|
|
|
|
type layout struct {
|
|
FormatVersion int `json:"formatVersion"`
|
|
Format string `json:"format"`
|
|
ObjectStorage string `json:"objectStorage"`
|
|
MetaInformation string `json:"metaInformation"`
|
|
TableContent string `json:"tableContent"`
|
|
Subdirectory string `json:"subdirectory"`
|
|
ReadableSource string `json:"readableSource"`
|
|
KeepClient string `json:"keepClient"`
|
|
}
|
|
|
|
type layoutResponse struct {
|
|
Layout layout `json:"layout"`
|
|
Log []gctsLogs `json:"log"`
|
|
Exception string `json:"exception"`
|
|
ErrorLogs []gctsLogs `json:"errorLog"`
|
|
}
|
|
|
|
type history struct {
|
|
Rid string `json:"rid"`
|
|
CheckoutTime int `json:"checkoutTime"`
|
|
FromCommit string `json:"fromCommit"`
|
|
ToCommit string `json:"toCommit"`
|
|
Caller string `json:"caller"`
|
|
Type string `json:"type"`
|
|
}
|
|
|
|
type historyResponse struct {
|
|
Result []history `xml:"result"`
|
|
Exception string `json:"exception"`
|
|
}
|
|
|
|
type checkstyleError struct {
|
|
Text string `xml:",chardata"`
|
|
Message string `xml:"message,attr"`
|
|
Source string `xml:"source,attr"`
|
|
Line string `xml:"line,attr"`
|
|
Severity string `xml:"severity,attr"`
|
|
}
|
|
|
|
type file struct {
|
|
Text string `xml:",chardata"`
|
|
Name string `xml:"name,attr"`
|
|
Error []checkstyleError `xml:"error"`
|
|
}
|
|
|
|
type checkstyle struct {
|
|
XMLName xml.Name `xml:"checkstyle"`
|
|
Text string `xml:",chardata"`
|
|
Version string `xml:"version,attr"`
|
|
File []file `xml:"file"`
|
|
}
|