/parse/parse.go

http://github.com/jacobsa/igo · Go · 121 lines · 62 code · 17 blank · 42 comment · 11 complexity · a77f4e734c918757591590ae91d87b5a MD5 · raw file

  1. // Copyright 2010 Aaron Jacobs. All rights reserved.
  2. // See the LICENSE file for licensing details.
  3. // The parse package offers utility functions for extracting dependency
  4. // information from Go source files.
  5. package parse
  6. import (
  7. "go/ast"
  8. "go/parser"
  9. "igo/set"
  10. "regexp"
  11. "strings"
  12. )
  13. // GetPackageName returns the package name from the supplied .go file source
  14. // code, or the empty string if it could not be properly parsed.
  15. func GetPackageName(source string) string {
  16. fileNode, err := parser.ParseFile("", source, nil, parser.ImportsOnly)
  17. if err != nil {
  18. return ""
  19. }
  20. return fileNode.Name.Name()
  21. }
  22. // GetImports parses the supplied source code for a .go file and returns a set
  23. // of package names that the file depends upon.
  24. //
  25. // For example, if source looks like the following:
  26. //
  27. // import (
  28. // "./bar/baz"
  29. // "fmt"
  30. // "os"
  31. // )
  32. //
  33. // func DoSomething() {
  34. // ...
  35. // }
  36. //
  37. // then the result will be { "./bar/baz", "fmt", "os" }.
  38. //
  39. // An attempt is made to return the imports for the file even if there is a
  40. // syntax error elsewhere in the file.
  41. func GetImports(source string) *set.StringSet {
  42. node, err := parser.ParseFile("", source, nil, parser.ImportsOnly)
  43. if err != nil {
  44. return &set.StringSet{}
  45. }
  46. var visitor importVisitor
  47. ast.Walk(&visitor, node)
  48. return &visitor.imports
  49. }
  50. type importVisitor struct {
  51. imports set.StringSet
  52. }
  53. var importRegexp *regexp.Regexp = regexp.MustCompile(`"(.+)"`)
  54. func (v *importVisitor) Visit(node interface{}) ast.Visitor {
  55. switch t := node.(type) {
  56. case *ast.ImportSpec:
  57. component := node.(*ast.ImportSpec).Path
  58. matches := importRegexp.MatchStrings(string(component.Value))
  59. if len(matches) < 2 {
  60. // skipping this?
  61. } else {
  62. v.imports.Insert(matches[1])
  63. }
  64. }
  65. return v
  66. }
  67. // GetTestFunctions parses the supplied source code for a .go file and returns
  68. // a set of test function names contained within it. Test functions are summed
  69. // to begin with the prefix "Test".
  70. //
  71. // For example, if source looks like the following:
  72. //
  73. // import "testing"
  74. //
  75. // func DoSomething() {
  76. // ...
  77. // }
  78. //
  79. // func TestBlah(t *testing.T) { ... }
  80. // func TestAsdf(t *testing.T) { ... }
  81. //
  82. // then the result will be { "TestBlah", "TestAsdf" }.
  83. func GetTestFunctions(source string) *set.StringSet {
  84. node, err := parser.ParseFile("", source, nil, 0)
  85. if err != nil {
  86. return &set.StringSet{}
  87. }
  88. var visitor testFunctionVisitor
  89. ast.Walk(&visitor, node)
  90. return &visitor.testFunctions
  91. }
  92. type testFunctionVisitor struct {
  93. testFunctions set.StringSet
  94. }
  95. func (v *testFunctionVisitor) Visit(node interface{}) ast.Visitor {
  96. switch t := node.(type) {
  97. case *ast.FuncDecl:
  98. name := node.(*ast.FuncDecl).Name.Obj.Name
  99. if strings.HasPrefix(name, "Test") {
  100. v.testFunctions.Insert(name)
  101. }
  102. }
  103. return v
  104. }