mirror of
https://github.com/stashapp/stash.git
synced 2025-12-06 08:26:00 +01:00
* Bump golang.org/x/text from 0.3.7 to 0.3.8 Bumps [golang.org/x/text](https://github.com/golang/text) from 0.3.7 to 0.3.8. - [Release notes](https://github.com/golang/text/releases) - [Commits](https://github.com/golang/text/compare/v0.3.7...v0.3.8) --- updated-dependencies: - dependency-name: golang.org/x/text dependency-type: direct:production ... Signed-off-by: dependabot[bot] <support@github.com> * Update go dependencies * Update x/net --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
207 lines
3.9 KiB
Go
207 lines
3.9 KiB
Go
package rewrite
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"go/ast"
|
|
"go/token"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/99designs/gqlgen/internal/code"
|
|
"golang.org/x/tools/go/packages"
|
|
)
|
|
|
|
type Rewriter struct {
|
|
pkg *packages.Package
|
|
files map[string]string
|
|
copied map[ast.Decl]bool
|
|
}
|
|
|
|
func New(dir string) (*Rewriter, error) {
|
|
importPath := code.ImportPathForDir(dir)
|
|
if importPath == "" {
|
|
return nil, fmt.Errorf("import path not found for directory: %q", dir)
|
|
}
|
|
pkgs, err := packages.Load(&packages.Config{
|
|
Mode: packages.NeedSyntax | packages.NeedTypes,
|
|
}, importPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(pkgs) == 0 {
|
|
return nil, fmt.Errorf("package not found for importPath: %s", importPath)
|
|
}
|
|
|
|
return &Rewriter{
|
|
pkg: pkgs[0],
|
|
files: map[string]string{},
|
|
copied: map[ast.Decl]bool{},
|
|
}, nil
|
|
}
|
|
|
|
func (r *Rewriter) getSource(start, end token.Pos) string {
|
|
startPos := r.pkg.Fset.Position(start)
|
|
endPos := r.pkg.Fset.Position(end)
|
|
|
|
if startPos.Filename != endPos.Filename {
|
|
panic("cant get source spanning multiple files")
|
|
}
|
|
|
|
file := r.getFile(startPos.Filename)
|
|
return file[startPos.Offset:endPos.Offset]
|
|
}
|
|
|
|
func (r *Rewriter) getFile(filename string) string {
|
|
if _, ok := r.files[filename]; !ok {
|
|
b, err := os.ReadFile(filename)
|
|
if err != nil {
|
|
panic(fmt.Errorf("unable to load file, already exists: %w", err))
|
|
}
|
|
|
|
r.files[filename] = string(b)
|
|
|
|
}
|
|
|
|
return r.files[filename]
|
|
}
|
|
|
|
func (r *Rewriter) GetPrevDecl(structname string, methodname string) *ast.FuncDecl {
|
|
for _, f := range r.pkg.Syntax {
|
|
for _, d := range f.Decls {
|
|
d, isFunc := d.(*ast.FuncDecl)
|
|
if !isFunc {
|
|
continue
|
|
}
|
|
if d.Name.Name != methodname {
|
|
continue
|
|
}
|
|
if d.Recv == nil || len(d.Recv.List) == 0 {
|
|
continue
|
|
}
|
|
recv := d.Recv.List[0].Type
|
|
if star, isStar := recv.(*ast.StarExpr); isStar {
|
|
recv = star.X
|
|
}
|
|
ident, ok := recv.(*ast.Ident)
|
|
if !ok {
|
|
continue
|
|
}
|
|
if ident.Name != structname {
|
|
continue
|
|
}
|
|
r.copied[d] = true
|
|
return d
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rewriter) GetMethodComment(structname string, methodname string) string {
|
|
d := r.GetPrevDecl(structname, methodname)
|
|
if d != nil {
|
|
return d.Doc.Text()
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (r *Rewriter) GetMethodBody(structname string, methodname string) string {
|
|
d := r.GetPrevDecl(structname, methodname)
|
|
if d != nil {
|
|
return r.getSource(d.Body.Pos()+1, d.Body.End()-1)
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (r *Rewriter) MarkStructCopied(name string) {
|
|
for _, f := range r.pkg.Syntax {
|
|
for _, d := range f.Decls {
|
|
d, isGen := d.(*ast.GenDecl)
|
|
if !isGen {
|
|
continue
|
|
}
|
|
if d.Tok != token.TYPE || len(d.Specs) == 0 {
|
|
continue
|
|
}
|
|
|
|
spec, isTypeSpec := d.Specs[0].(*ast.TypeSpec)
|
|
if !isTypeSpec {
|
|
continue
|
|
}
|
|
|
|
if spec.Name.Name != name {
|
|
continue
|
|
}
|
|
|
|
r.copied[d] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r *Rewriter) ExistingImports(filename string) []Import {
|
|
filename, err := filepath.Abs(filename)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
for _, f := range r.pkg.Syntax {
|
|
pos := r.pkg.Fset.Position(f.Pos())
|
|
|
|
if filename != pos.Filename {
|
|
continue
|
|
}
|
|
|
|
var imps []Import
|
|
for _, i := range f.Imports {
|
|
name := ""
|
|
if i.Name != nil {
|
|
name = i.Name.Name
|
|
}
|
|
path, err := strconv.Unquote(i.Path.Value)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
imps = append(imps, Import{name, path})
|
|
}
|
|
return imps
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rewriter) RemainingSource(filename string) string {
|
|
filename, err := filepath.Abs(filename)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
for _, f := range r.pkg.Syntax {
|
|
pos := r.pkg.Fset.Position(f.Pos())
|
|
|
|
if filename != pos.Filename {
|
|
continue
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
|
|
for _, d := range f.Decls {
|
|
if r.copied[d] {
|
|
continue
|
|
}
|
|
|
|
if d, isGen := d.(*ast.GenDecl); isGen && d.Tok == token.IMPORT {
|
|
continue
|
|
}
|
|
|
|
buf.WriteString(r.getSource(d.Pos(), d.End()))
|
|
buf.WriteString("\n")
|
|
}
|
|
|
|
return strings.TrimSpace(buf.String())
|
|
}
|
|
return ""
|
|
}
|
|
|
|
type Import struct {
|
|
Alias string
|
|
ImportPath string
|
|
}
|