Skip to content
This repository has been archived by the owner on Dec 8, 2022. It is now read-only.

feat(*): add SDK core #1

Merged
merged 26 commits into from
Jun 20, 2022
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
e095afe
feat(*): add SDK core
burntcarrot Jun 6, 2022
d768047
tests(*): integration tests using tree-sitter
burntcarrot Jun 7, 2022
5ad1b10
ci: add golangci-lint configuration
burntcarrot Jun 7, 2022
ceef642
fix(sdk/utils): sanitize markdown
burntcarrot Jun 7, 2022
bd4a0fe
refactor(*): refactor SDK and add tests
burntcarrot Jun 11, 2022
318ccaa
tests(*): test TOML generation
burntcarrot Jun 12, 2022
a0928fa
fix(utils): check return value while creating directory
burntcarrot Jun 12, 2022
da20033
refactor(*): refactor SDK and add tests
burntcarrot Jun 12, 2022
7ce40b7
docs(*): add guide for writing an analyzer
burntcarrot Jun 12, 2022
3a0c99a
chore: add README
burntcarrot Jun 12, 2022
f7d3b7e
refactor(*): refactor SDK and cleanup
burntcarrot Jun 13, 2022
6bd2540
fix(analyzers): use REPO_ROOT and TOOLBOX_PATH environment variables
burntcarrot Jun 14, 2022
a0ce3b1
chore: add gitignore
burntcarrot Jun 14, 2022
7fc2940
docs(*): update guide
burntcarrot Jun 14, 2022
adfb69d
docs(*): add guide for writing CSS analyzer
burntcarrot Jun 14, 2022
a101b86
tests(analysistest): add integration test for CSS
burntcarrot Jun 14, 2022
bb2135c
chore: fix DeepSource issues
burntcarrot Jun 14, 2022
180769b
refactor(*): refactor tests and add guides
burntcarrot Jun 15, 2022
e5a7554
docs(guides): fix TOML fields for issues.toml
burntcarrot Jun 15, 2022
a74d7ce
ci: install csslint
burntcarrot Jun 15, 2022
e9e4724
feat(analysistest): add all tree-sitter languages
burntcarrot Jun 15, 2022
f7d38a1
Apply suggestions from code review
sourya-deepsource Jun 15, 2022
0513907
tests(*): add unit tests
burntcarrot Jun 17, 2022
1034696
chore: fix merge conflicts
burntcarrot Jun 17, 2022
1112721
refactor(*): allow issue parsing via structs
burntcarrot Jun 17, 2022
ad43322
refactor(*): pass env vars through params
burntcarrot Jun 20, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 14 additions & 0 deletions .deepsource.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
version = 1

exclude_patterns = ["**/testdata/src/**"]

[[analyzers]]
name = "go"
enabled = true

[analyzers.meta]
import_root = "github.com/deepsourcelabs/deepsource-go"

[[transformers]]
name = "gofumpt"
enabled = true
20 changes: 20 additions & 0 deletions .github/workflows/tests.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
name: Tests
on: [push, pull_request]
jobs:
tests:
name: tests
runs-on: ubuntu-latest
strategy:
matrix:
go-version: [1.17.x]
steps:
- name: Install Go
uses: actions/setup-go@v3
with:
go-version: ${{ matrix.go-version }}
- name: Checkout code
uses: actions/checkout@v3
- name: Install analyzers
run: go install honnef.co/go/tools/cmd/staticcheck@latest
- name: Run tests
run: go test -v ./...
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
.envrc
36 changes: 36 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
<p align="center">
<img src="https://cms.deepsource.io/logo-wordmark-dark.svg" />
</p>

<p align="center">
<a href="https://deepsource.io/docs">Documentation</a> |
<a href="https://deepsource.io/signup">Get Started</a> |
<a href="https://discuss.deepsource.io/">Discuss</a>
</p>

<p align="center">
DeepSource helps you ship good quality code.
</p>

</p>

---

# DeepSource Go SDK

<a href="https://pkg.go.dev/github.com/deepsourcelabs/deepsource-go"><img src="https://godoc.org/github.com/deepsourcelabs/deepsource-go?status.svg" /></a>

Go SDK for [DeepSource](https://deepsource.io/).

The Go SDK makes it easier for developers to integrate an existing analyzer with DeepSource.

## Guides

Here are some extensive guides on working with the SDK:

- [Writing custom analyzers](guides/writing-analyzers.md)
- [Writing a CSS Analyzer](guides/css-analyzer.md)

## Community

Interested in DeepSource and want to chat with the community? Feel free to join our [Discord server](http://deepsource.io/discord).
142 changes: 142 additions & 0 deletions analyzers/analysistest/analysistest.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
package analysistest

import (
"context"
"errors"
"os"
"path/filepath"
"regexp"
"sort"
"strings"

"github.com/deepsourcelabs/deepsource-go/analyzers/types"
sitter "github.com/smacker/go-tree-sitter"
"github.com/smacker/go-tree-sitter/css"
"github.com/smacker/go-tree-sitter/golang"
)

// ParsedIssue represents an issue parsed using tree-sitter.
type ParsedIssue struct {
IssueCode string
Line int
}

// Verify compares the generated report and parsed issues using tree-sitter.
func Verify(report types.AnalysisReport, filename string) error {
burntcarrot marked this conversation as resolved.
Show resolved Hide resolved
parser := sitter.NewParser()

// get language
lang, err := getLanguage(filename)
if err != nil {
return err
}
parser.SetLanguage(lang)

// read report
content, err := os.ReadFile(filename)
if err != nil {
return err
}

// generate tree
ctx := context.Background()
tree, err := parser.ParseCtx(ctx, nil, content)
if err != nil {
return err
}

// create a query for fetching comments
queryStr := "(comment) @comment"
query, err := sitter.NewQuery([]byte(queryStr), lang)
if err != nil {
return err
}

// execute query on root node
qc := sitter.NewQueryCursor()
n := tree.RootNode()
qc.Exec(query, n)
defer qc.Close()

var parsedIssues []ParsedIssue

// iterate over matches
for {
m, ok := qc.NextMatch()
if !ok {
break
}

for _, c := range m.Captures {
// get node content
node := c.Node
nodeContent := node.Content(content)

// check if the comment contains raise annotation
if strings.Contains(nodeContent, "raise") {
// find match using expression
exp := regexp.MustCompile(`.+ raise: `)
submatches := exp.FindStringSubmatch(nodeContent)

if len(submatches) != 0 {
substrings := exp.Split(nodeContent, -1)
if len(substrings) > 1 {
issueCodes := strings.Split(substrings[1], ",")
// add issue to parsedIssues
for _, issueCode := range issueCodes {
parsedIssue := ParsedIssue{IssueCode: strings.TrimSpace(issueCode), Line: int(node.StartPoint().Row) + 1}
parsedIssues = append(parsedIssues, parsedIssue)
}
}
}
}
}
}

// if number of issues don't match, exit early.
if len(parsedIssues) != len(report.Issues) {
return errors.New("mismatch between the number of reported issues and parsed issues")
}

// compare the report's issues and parsed issues
match := compareReport(parsedIssues, report)
if !match {
return errors.New("mismatch between parsed issue and report issue")
}

return nil
}

// getLanguage is a helper for fetching a tree-sitter language based on the file's extension.
func getLanguage(filename string) (*sitter.Language, error) {
extension := filepath.Ext(filename)

switch extension {
case ".go":
return golang.GetLanguage(), nil
case ".css":
return css.GetLanguage(), nil
default:
return nil, errors.New("language not supported")
}
}

// compareReport is a helper which checks if the parsed issues are identical to the issues present in the report.
func compareReport(parsedIssues []ParsedIssue, report types.AnalysisReport) bool {
// sort report and parsedIssues by line number
sort.Slice(parsedIssues, func(i, j int) bool {
return parsedIssues[i].Line < parsedIssues[j].Line
})

sort.Slice(report.Issues, func(i, j int) bool {
return report.Issues[i].Location.Position.Begin.Line < report.Issues[j].Location.Position.Begin.Line
})

for i, issue := range report.Issues {
if (parsedIssues[i].Line != issue.Location.Position.Begin.Line) && (parsedIssues[i].IssueCode != issue.IssueCode) {
return false
}
}

return true
}
90 changes: 90 additions & 0 deletions analyzers/processors/regex.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
package processors

import (
"bytes"
"errors"
"regexp"
"strconv"
"strings"

"github.com/deepsourcelabs/deepsource-go/analyzers/types"
)

// IssueCodeProcessor is used when an analyzer doesn't support issue codes. IssueCodeProcessor takes in the content of the "issue_code" named group and returns an appropriate issue code. If not implemented, it fallbacks to using the content as the issue code.
type IssueCodeProcessor func(string) string

// RegexProcessor utilizes regular expressions for processing.
type RegexProcessor struct {
Pattern string
IssueCodeProcessor IssueCodeProcessor
burntcarrot marked this conversation as resolved.
Show resolved Hide resolved
}

func (r *RegexProcessor) Process(buf bytes.Buffer) (types.AnalysisReport, error) {
var issues []types.Issue

// trim newline from buffer output
lines := strings.Split(buf.String(), "\n")

for _, line := range lines {
// trim spaces
line = strings.TrimSpace(line)
if line == "" {
break
}

exp, err := regexp.Compile(r.Pattern)
if err != nil {
return types.AnalysisReport{}, err
}

// get groups
groupNames := exp.SubexpNames()

var issue types.Issue
groups := exp.FindAllStringSubmatch(strings.TrimSuffix(line, "\n"), -1)
for groupIdx, content := range groups[0] {
groupName := groupNames[groupIdx]

// populate issue using named groups
switch groupName {
case "filename":
issue.Location.Path = content
case "line":
line, err := strconv.Atoi(content)
if err != nil {
return types.AnalysisReport{}, err
}
issue.Location.Position.Begin.Line = line
case "column":
col, err := strconv.Atoi(content)
if err != nil {
return types.AnalysisReport{}, err
}
issue.Location.Position.Begin.Column = col
case "message":
issue.IssueText = content
case "issue_code":
if r.IssueCodeProcessor == nil {
issue.IssueCode = content
} else {
issue.IssueCode = r.IssueCodeProcessor(content)
}
default:
continue
}
}
if len(groups) == 0 {
return types.AnalysisReport{}, errors.New("failed to parse message")
}

issues = append(issues, issue)
}

// populate report
report := types.AnalysisReport{
Issues: issues,
}

// return report
return report, nil
}
Loading