| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 | // Copyright 2015 The Gogs Authors. All rights reserved.// Use of this source code is governed by a MIT-style// license that can be found in the LICENSE file.package dbimport (	"fmt"	"io/ioutil"	"net/url"	"os"	"path"	"path/filepath"	"strings"	"github.com/unknwon/com"	"github.com/gogs/git-module"	"gogs.io/gogs/internal/conf"	"gogs.io/gogs/internal/sync")var wikiWorkingPool = sync.NewExclusivePool()// ToWikiPageURL formats a string to corresponding wiki URL name.func ToWikiPageURL(name string) string {	return url.QueryEscape(name)}// ToWikiPageName formats a URL back to corresponding wiki page name,// and removes leading characters './' to prevent changing files// that are not belong to wiki repository.func ToWikiPageName(urlString string) string {	name, _ := url.QueryUnescape(urlString)	return strings.Replace(strings.TrimLeft(path.Clean("/"+name), "/"), "/", " ", -1)}// WikiCloneLink returns clone URLs of repository wiki.func (repo *Repository) WikiCloneLink() (cl *CloneLink) {	return repo.cloneLink(true)}// WikiPath returns wiki data path by given user and repository name.func WikiPath(userName, repoName string) string {	return filepath.Join(UserPath(userName), strings.ToLower(repoName)+".wiki.git")}func (repo *Repository) WikiPath() string {	return WikiPath(repo.MustOwner().Name, repo.Name)}// HasWiki returns true if repository has wiki.func (repo *Repository) HasWiki() bool {	return com.IsDir(repo.WikiPath())}// InitWiki initializes a wiki for repository,// it does nothing when repository already has wiki.func (repo *Repository) InitWiki() error {	if repo.HasWiki() {		return nil	}	if err := git.Init(repo.WikiPath(), git.InitOptions{Bare: true}); err != nil {		return fmt.Errorf("init repository: %v", err)	} else if err = createDelegateHooks(repo.WikiPath()); err != nil {		return fmt.Errorf("createDelegateHooks: %v", err)	}	return nil}func (repo *Repository) LocalWikiPath() string {	return filepath.Join(conf.Server.AppDataPath, "tmp", "local-wiki", com.ToStr(repo.ID))}// UpdateLocalWiki makes sure the local copy of repository wiki is up-to-date.func (repo *Repository) UpdateLocalWiki() error {	return UpdateLocalCopyBranch(repo.WikiPath(), repo.LocalWikiPath(), "master", true)}func discardLocalWikiChanges(localPath string) error {	return discardLocalRepoBranchChanges(localPath, "master")}// updateWikiPage adds new page to repository wiki.func (repo *Repository) updateWikiPage(doer *User, oldTitle, title, content, message string, isNew bool) (err error) {	wikiWorkingPool.CheckIn(com.ToStr(repo.ID))	defer wikiWorkingPool.CheckOut(com.ToStr(repo.ID))	if err = repo.InitWiki(); err != nil {		return fmt.Errorf("InitWiki: %v", err)	}	localPath := repo.LocalWikiPath()	if err = discardLocalWikiChanges(localPath); err != nil {		return fmt.Errorf("discardLocalWikiChanges: %v", err)	} else if err = repo.UpdateLocalWiki(); err != nil {		return fmt.Errorf("UpdateLocalWiki: %v", err)	}	title = ToWikiPageName(title)	filename := path.Join(localPath, title+".md")	// If not a new file, show perform update not create.	if isNew {		if com.IsExist(filename) {			return ErrWikiAlreadyExist{filename}		}	} else {		os.Remove(path.Join(localPath, oldTitle+".md"))	}	// SECURITY: if new file is a symlink to non-exist critical file,	// attack content can be written to the target file (e.g. authorized_keys2)	// as a new page operation.	// So we want to make sure the symlink is removed before write anything.	// The new file we created will be in normal text format.	os.Remove(filename)	if err = ioutil.WriteFile(filename, []byte(content), 0666); err != nil {		return fmt.Errorf("WriteFile: %v", err)	}	if len(message) == 0 {		message = "Update page '" + title + "'"	}	if err = git.RepoAdd(localPath, git.AddOptions{All: true}); err != nil {		return fmt.Errorf("add all changes: %v", err)	} else if err = git.RepoCommit(localPath, doer.NewGitSig(), message); err != nil {		return fmt.Errorf("commit changes: %v", err)	} else if err = git.RepoPush(localPath, "origin", "master"); err != nil {		return fmt.Errorf("push: %v", err)	}	return nil}func (repo *Repository) AddWikiPage(doer *User, title, content, message string) error {	return repo.updateWikiPage(doer, "", title, content, message, true)}func (repo *Repository) EditWikiPage(doer *User, oldTitle, title, content, message string) error {	return repo.updateWikiPage(doer, oldTitle, title, content, message, false)}func (repo *Repository) DeleteWikiPage(doer *User, title string) (err error) {	wikiWorkingPool.CheckIn(com.ToStr(repo.ID))	defer wikiWorkingPool.CheckOut(com.ToStr(repo.ID))	localPath := repo.LocalWikiPath()	if err = discardLocalWikiChanges(localPath); err != nil {		return fmt.Errorf("discardLocalWikiChanges: %v", err)	} else if err = repo.UpdateLocalWiki(); err != nil {		return fmt.Errorf("UpdateLocalWiki: %v", err)	}	title = ToWikiPageName(title)	filename := path.Join(localPath, title+".md")	os.Remove(filename)	message := "Delete page '" + title + "'"	if err = git.RepoAdd(localPath, git.AddOptions{All: true}); err != nil {		return fmt.Errorf("add all changes: %v", err)	} else if err = git.RepoCommit(localPath, doer.NewGitSig(), message); err != nil {		return fmt.Errorf("commit changes: %v", err)	} else if err = git.RepoPush(localPath, "origin", "master"); err != nil {		return fmt.Errorf("push: %v", err)	}	return nil}
 |