2020-04-05 09:20:50 +03:00
// Copyright 2020 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2016-12-26 04:16:37 +03:00
package models
import (
2019-02-12 16:07:31 +03:00
"crypto/sha256"
"encoding/hex"
2016-12-26 04:16:37 +03:00
"errors"
2019-02-12 16:07:31 +03:00
"fmt"
"io"
2020-09-08 18:45:10 +03:00
"path"
2017-12-11 07:37:04 +03:00
2019-08-15 17:46:21 +03:00
"code.gitea.io/gitea/modules/timeutil"
2019-10-28 21:31:55 +03:00
"xorm.io/builder"
2016-12-26 04:16:37 +03:00
)
// LFSMetaObject stores metadata for LFS tracked files.
type LFSMetaObject struct {
2019-08-15 17:46:21 +03:00
ID int64 ` xorm:"pk autoincr" `
Oid string ` xorm:"UNIQUE(s) INDEX NOT NULL" `
Size int64 ` xorm:"NOT NULL" `
RepositoryID int64 ` xorm:"UNIQUE(s) INDEX NOT NULL" `
Existing bool ` xorm:"-" `
CreatedUnix timeutil . TimeStamp ` xorm:"created" `
2016-12-26 04:16:37 +03:00
}
2020-09-08 18:45:10 +03:00
// RelativePath returns the relative path of the lfs object
func ( m * LFSMetaObject ) RelativePath ( ) string {
if len ( m . Oid ) < 5 {
return m . Oid
}
return path . Join ( m . Oid [ 0 : 2 ] , m . Oid [ 2 : 4 ] , m . Oid [ 4 : ] )
}
2019-02-12 16:07:31 +03:00
// Pointer returns the string representation of an LFS pointer file
func ( m * LFSMetaObject ) Pointer ( ) string {
return fmt . Sprintf ( "%s\n%s%s\nsize %d\n" , LFSMetaFileIdentifier , LFSMetaFileOidPrefix , m . Oid , m . Size )
}
2016-12-26 04:16:37 +03:00
// LFSTokenResponse defines the JSON structure in which the JWT token is stored.
// This structure is fetched via SSH and passed by the Git LFS client to the server
// endpoint for authorization.
type LFSTokenResponse struct {
Header map [ string ] string ` json:"header" `
Href string ` json:"href" `
}
var (
// ErrLFSObjectNotExist is returned from lfs models functions in order
// to differentiate between database and missing object errors.
ErrLFSObjectNotExist = errors . New ( "LFS Meta object does not exist" )
)
const (
// LFSMetaFileIdentifier is the string appearing at the first line of LFS pointer files.
// https://github.com/git-lfs/git-lfs/blob/master/docs/spec.md
LFSMetaFileIdentifier = "version https://git-lfs.github.com/spec/v1"
// LFSMetaFileOidPrefix appears in LFS pointer files on a line before the sha256 hash.
LFSMetaFileOidPrefix = "oid sha256:"
)
// NewLFSMetaObject stores a given populated LFSMetaObject structure in the database
// if it is not already present.
func NewLFSMetaObject ( m * LFSMetaObject ) ( * LFSMetaObject , error ) {
var err error
2019-01-23 11:56:51 +03:00
sess := x . NewSession ( )
defer sess . Close ( )
if err = sess . Begin ( ) ; err != nil {
return nil , err
}
has , err := sess . Get ( m )
2016-12-26 04:16:37 +03:00
if err != nil {
return nil , err
}
if has {
m . Existing = true
2019-01-23 11:56:51 +03:00
return m , sess . Commit ( )
2016-12-26 04:16:37 +03:00
}
if _ , err = sess . Insert ( m ) ; err != nil {
return nil , err
}
return m , sess . Commit ( )
}
2019-02-12 16:07:31 +03:00
// GenerateLFSOid generates a Sha256Sum to represent an oid for arbitrary content
func GenerateLFSOid ( content io . Reader ) ( string , error ) {
h := sha256 . New ( )
if _ , err := io . Copy ( h , content ) ; err != nil {
return "" , err
}
sum := h . Sum ( nil )
return hex . EncodeToString ( sum ) , nil
}
2016-12-26 04:16:37 +03:00
// GetLFSMetaObjectByOid selects a LFSMetaObject entry from database by its OID.
// It may return ErrLFSObjectNotExist or a database error. If the error is nil,
// the returned pointer is a valid LFSMetaObject.
2017-10-30 15:11:56 +03:00
func ( repo * Repository ) GetLFSMetaObjectByOid ( oid string ) ( * LFSMetaObject , error ) {
2016-12-26 04:16:37 +03:00
if len ( oid ) == 0 {
return nil , ErrLFSObjectNotExist
}
2017-10-30 15:11:56 +03:00
m := & LFSMetaObject { Oid : oid , RepositoryID : repo . ID }
2016-12-26 04:16:37 +03:00
has , err := x . Get ( m )
if err != nil {
return nil , err
} else if ! has {
return nil , ErrLFSObjectNotExist
}
return m , nil
}
// RemoveLFSMetaObjectByOid removes a LFSMetaObject entry from database by its OID.
// It may return ErrLFSObjectNotExist or a database error.
2019-10-28 21:31:55 +03:00
func ( repo * Repository ) RemoveLFSMetaObjectByOid ( oid string ) ( int64 , error ) {
2016-12-26 04:16:37 +03:00
if len ( oid ) == 0 {
2019-10-28 21:31:55 +03:00
return 0 , ErrLFSObjectNotExist
2016-12-26 04:16:37 +03:00
}
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-12-26 04:16:37 +03:00
if err := sess . Begin ( ) ; err != nil {
2019-10-28 21:31:55 +03:00
return - 1 , err
2016-12-26 04:16:37 +03:00
}
2017-10-30 15:11:56 +03:00
m := & LFSMetaObject { Oid : oid , RepositoryID : repo . ID }
2016-12-26 04:16:37 +03:00
if _ , err := sess . Delete ( m ) ; err != nil {
2019-10-28 21:31:55 +03:00
return - 1 , err
}
count , err := sess . Count ( & LFSMetaObject { Oid : oid } )
if err != nil {
return count , err
}
return count , sess . Commit ( )
}
// GetLFSMetaObjects returns all LFSMetaObjects associated with a repository
func ( repo * Repository ) GetLFSMetaObjects ( page , pageSize int ) ( [ ] * LFSMetaObject , error ) {
sess := x . NewSession ( )
defer sess . Close ( )
if page >= 0 && pageSize > 0 {
start := 0
if page > 0 {
start = ( page - 1 ) * pageSize
}
sess . Limit ( pageSize , start )
}
lfsObjects := make ( [ ] * LFSMetaObject , 0 , pageSize )
return lfsObjects , sess . Find ( & lfsObjects , & LFSMetaObject { RepositoryID : repo . ID } )
}
// CountLFSMetaObjects returns a count of all LFSMetaObjects associated with a repository
func ( repo * Repository ) CountLFSMetaObjects ( ) ( int64 , error ) {
return x . Count ( & LFSMetaObject { RepositoryID : repo . ID } )
}
// LFSObjectAccessible checks if a provided Oid is accessible to the user
func LFSObjectAccessible ( user * User , oid string ) ( bool , error ) {
if user . IsAdmin {
count , err := x . Count ( & LFSMetaObject { Oid : oid } )
return ( count > 0 ) , err
}
2020-01-13 20:33:46 +03:00
cond := accessibleRepositoryCondition ( user )
2019-10-28 21:31:55 +03:00
count , err := x . Where ( cond ) . Join ( "INNER" , "repository" , "`lfs_meta_object`.repository_id = `repository`.id" ) . Count ( & LFSMetaObject { Oid : oid } )
return ( count > 0 ) , err
}
// LFSAutoAssociate auto associates accessible LFSMetaObjects
func LFSAutoAssociate ( metas [ ] * LFSMetaObject , user * User , repoID int64 ) error {
sess := x . NewSession ( )
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
oids := make ( [ ] interface { } , len ( metas ) )
oidMap := make ( map [ string ] * LFSMetaObject , len ( metas ) )
for i , meta := range metas {
oids [ i ] = meta . Oid
oidMap [ meta . Oid ] = meta
}
cond := builder . NewCond ( )
if ! user . IsAdmin {
cond = builder . In ( "`lfs_meta_object`.repository_id" ,
2020-01-13 20:33:46 +03:00
builder . Select ( "`repository`.id" ) . From ( "repository" ) . Where ( accessibleRepositoryCondition ( user ) ) )
2019-10-28 21:31:55 +03:00
}
newMetas := make ( [ ] * LFSMetaObject , 0 , len ( metas ) )
if err := sess . Cols ( "oid" ) . Where ( cond ) . In ( "oid" , oids ... ) . GroupBy ( "oid" ) . Find ( & newMetas ) ; err != nil {
return err
}
for i := range newMetas {
newMetas [ i ] . Size = oidMap [ newMetas [ i ] . Oid ] . Size
newMetas [ i ] . RepositoryID = repoID
}
if _ , err := sess . InsertMulti ( newMetas ) ; err != nil {
2016-12-26 04:16:37 +03:00
return err
}
return sess . Commit ( )
}
2020-09-08 18:45:10 +03:00
// IterateLFS iterates lfs object
func IterateLFS ( f func ( mo * LFSMetaObject ) error ) error {
var start int
const batchSize = 100
for {
var mos = make ( [ ] * LFSMetaObject , 0 , batchSize )
if err := x . Limit ( batchSize , start ) . Find ( & mos ) ; err != nil {
return err
}
if len ( mos ) == 0 {
return nil
}
start += len ( mos )
for _ , mo := range mos {
if err := f ( mo ) ; err != nil {
return err
}
}
}
}