@ -16,99 +16,181 @@
package doc
package doc
import (
import (
"archive/tar"
"bytes"
"compress/gzip"
"io"
"net/http"
"net/http"
"os"
"path"
"path"
"regexp"
"regexp"
"strings"
"github.com/Unknwon/gowalker/utils"
)
)
var (
var (
b itbucketPattern = regexp . MustCompile ( ` ^bitbucket\.org/(?P<owner>[a-z0-9A-Z_.\-]+)/(?P<repo>[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]*)?$ ` )
B itbucketPattern = regexp . MustCompile ( ` ^bitbucket\.org/(?P<owner>[a-z0-9A-Z_.\-]+)/(?P<repo>[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]*)?$ ` )
bitbucketEtagRe = regexp . MustCompile ( ` ^(hg|git)- ` )
bitbucketEtagRe = regexp . MustCompile ( ` ^(hg|git)- ` )
)
)
func getBitbucketDoc ( client * http . Client , match map [ string ] string , savedEtag string ) ( * Package , error ) {
// GetBitbucketDoc downloads tarball from bitbucket.org.
func GetBitbucketDoc ( client * http . Client , match map [ string ] string , installGOPATH , commit string , cmdFlags map [ string ] bool ) ( * Package , [ ] string , error ) {
if m := bitbucketEtagRe . FindStringSubmatch ( savedEtag ) ; m != nil {
// Check version control.
if m := bitbucketEtagRe . FindStringSubmatch ( commit ) ; m != nil {
match [ "vcs" ] = m [ 1 ]
match [ "vcs" ] = m [ 1 ]
} else {
} else {
var repo struct {
var repo struct {
Scm string
Scm string
}
}
if err := httpGetJSON ( client , expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}" , match ) , & repo ) ; err != nil {
if err := httpGetJSON ( client , expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}" , match ) , & repo ) ; err != nil {
return nil , err
return nil , nil , err
}
}
match [ "vcs" ] = repo . Scm
match [ "vcs" ] = repo . Scm
}
}
// bundle and snapshot will have commit 'B' and 'S',
// but does not need to download dependencies.
isCheckImport := len ( commit ) == 0
// Check if download with specific revision.
if isCheckImport || len ( commit ) == 1 {
tags := make ( map [ string ] string )
tags := make ( map [ string ] string )
for _ , nodeType := range [ ] string { "branches" , "tags" } {
for _ , nodeType := range [ ] string { "branches" , "tags" } {
var nodes map [ string ] struct {
var nodes map [ string ] struct {
Node string
Node string
}
}
if err := httpGetJSON ( client , expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}" , match , nodeType ) , & nodes ) ; err != nil {
if err := httpGetJSON ( client , expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}" , match , nodeType ) , & nodes ) ; err != nil {
return nil , err
return nil , nil , err
}
}
for t , n := range nodes {
for t , n := range nodes {
tags [ t ] = n . Node
tags [ t ] = n . Node
}
}
}
}
// Check revision tag.
var err error
var err error
match [ "tag" ] , match [ "commit" ] , err = bestTag ( tags , defaultTags [ match [ "vcs" ] ] )
match [ "tag" ] , match [ "commit" ] , err = bestTag ( tags , defaultTags [ match [ "vcs" ] ] )
if err != nil {
if err != nil {
return nil , err
return nil , nil , err
}
} else {
match [ "commit" ] = commit
}
}
// Check revision tag.
// We use .tar.gz here.
etag := expand ( "{vcs}-{commit}" , match )
// zip : https://bitbucket.org/{owner}/{repo}/get/{commit}.zip
if etag == savedEtag {
// tarball : https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz
return nil , errNotModified
// Downlaod archive.
p , err := httpGetBytes ( client , expand ( "https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz" , match ) , nil )
if err != nil {
return nil , nil , err
}
importPath := "bitbucket.org/" + expand ( "{owner}/{repo}" , match )
installPath := installGOPATH + "/src/" + importPath
// Remove old files.
os . RemoveAll ( installPath + "/" )
// Create destination directory.
os . MkdirAll ( installPath + "/" , os . ModePerm )
gzr , err := gzip . NewReader ( bytes . NewReader ( p ) )
if err != nil {
return nil , nil , err
}
}
defer gzr . Close ( )
tr := tar . NewReader ( gzr )
var node struct {
var autoPath string // Auto path is the root path that generated by bitbucket.org.
Files [ ] struct {
// Get source file data.
Path string
dirs := make ( [ ] string , 0 , 5 )
for {
h , err := tr . Next ( )
if err == io . EOF {
break
} else if err != nil {
return nil , nil , err
}
}
Directories [ ] string
fn := h . FileInfo ( ) . Name ( )
// In case that we find directory, usually we should not.
if ! strings . HasSuffix ( fn , "/" ) {
// Check root path.
if len ( autoPath ) == 0 {
autoPath = fn [ : strings . Index ( fn , "/" ) ]
}
}
absPath := strings . Replace ( fn , autoPath , installPath , 1 )
if err := httpGetJSON ( client , expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/src/{tag}{dir}/" , match ) , & node ) ; err != nil {
// Create diretory before create file.
return nil , err
dir := path . Dir ( absPath )
if ! checkDir ( dir , dirs ) {
dirs = append ( dirs , dir )
os . MkdirAll ( dir + "/" , os . ModePerm )
}
}
// Get source file data.
// Get data from archive.
files := make ( [ ] * source , 0 , 5 )
fbytes := make ( [ ] byte , h . Size )
for _ , f := range node . Files {
if _ , err := io . ReadFull ( tr , fbytes ) ; err != nil {
_ , name := path . Split ( f . Path )
return nil , nil , err
if utils . IsDocFile ( name ) {
}
files = append ( files , & source {
name : name ,
// Write data to file
browseURL : expand ( "https://bitbucket.org/{owner}/{repo}/src/{tag}/{0}" , match , f . Path ) ,
fw , err := os . Create ( absPath )
rawURL : expand ( "https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/raw/{tag}/{0}" , match , f . Path ) ,
if err != nil {
} )
return nil , nil , err
}
_ , err = fw . Write ( fbytes )
fw . Close ( )
if err != nil {
return nil , nil , err
}
}
}
pkg := & Package {
ImportPath : importPath ,
AbsPath : installPath ,
Commit : commit ,
}
var imports [ ] string
// Check if need to check imports.
if isCheckImport {
rootdir , err := os . Open ( installPath + "/" )
if err != nil {
return nil , nil , err
}
}
defer rootdir . Close ( )
dirs , err := rootdir . Readdir ( 0 )
if err != nil {
return nil , nil , err
}
}
if len ( files ) == 0 && len ( node . Directories ) == 0 {
for _ , d := range dirs {
return nil , NotFoundError { "Directory tree does not contain Go files and subdirs." }
if d . IsDir ( ) {
absPath := installPath + "/" + d . Name ( ) + "/"
imports , err = checkImports ( absPath , importPath )
if err != nil {
return nil , nil , err
}
}
}
}
}
// Fetch file from VCS.
return pkg , imports , err
if err := fetchFiles ( client , files , nil ) ; err != nil {
return nil , err
}
}
// Start generating data.
// checkDir checks if current directory has been saved.
w := & walker {
func checkDir ( dir string , dirs [ ] string ) bool {
lineFmt : "#cl-%d" ,
for _ , d := range dirs {
pdoc : & Package {
if dir == d {
ImportPath : match [ "importPath" ] ,
return true
ProjectName : match [ "repo" ] ,
}
Etag : etag ,
Dirs : node . Directories ,
} ,
}
}
return w . build ( files )
return false
}
}