2020-02-02 05:17:44 +03:00
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2020-02-02 05:17:44 +03:00
package cmd
import (
"errors"
"fmt"
"os"
"path/filepath"
"strings"
2023-04-12 13:16:45 +03:00
"code.gitea.io/gitea/modules/assetfs"
2020-02-02 05:17:44 +03:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/options"
"code.gitea.io/gitea/modules/public"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/templates"
2021-07-15 18:46:07 +03:00
"code.gitea.io/gitea/modules/util"
2020-02-02 05:17:44 +03:00
"github.com/gobwas/glob"
"github.com/urfave/cli"
)
// Cmdembedded represents the available extract sub-command.
var (
Cmdembedded = cli . Command {
Name : "embedded" ,
Usage : "Extract embedded resources" ,
Description : "A command for extracting embedded resources, like templates and images" ,
Subcommands : [ ] cli . Command {
subcmdList ,
subcmdView ,
subcmdExtract ,
} ,
}
subcmdList = cli . Command {
Name : "list" ,
Usage : "List files matching the given pattern" ,
Action : runList ,
Flags : [ ] cli . Flag {
cli . BoolFlag {
Name : "include-vendored,vendor" ,
Usage : "Include files under public/vendor as well" ,
} ,
} ,
}
subcmdView = cli . Command {
Name : "view" ,
Usage : "View a file matching the given pattern" ,
Action : runView ,
Flags : [ ] cli . Flag {
cli . BoolFlag {
Name : "include-vendored,vendor" ,
Usage : "Include files under public/vendor as well" ,
} ,
} ,
}
subcmdExtract = cli . Command {
Name : "extract" ,
Usage : "Extract resources" ,
Action : runExtract ,
Flags : [ ] cli . Flag {
cli . BoolFlag {
Name : "include-vendored,vendor" ,
Usage : "Include files under public/vendor as well" ,
} ,
cli . BoolFlag {
Name : "overwrite" ,
Usage : "Overwrite files if they already exist" ,
} ,
cli . BoolFlag {
Name : "rename" ,
Usage : "Rename files as {name}.bak if they already exist (overwrites previous .bak)" ,
} ,
cli . BoolFlag {
Name : "custom" ,
Usage : "Extract to the 'custom' directory as per app.ini" ,
} ,
cli . StringFlag {
Name : "destination,dest-dir" ,
Usage : "Extract to the specified directory" ,
} ,
} ,
}
2023-04-12 13:16:45 +03:00
matchedAssetFiles [ ] assetFile
2020-02-02 05:17:44 +03:00
)
2023-04-12 13:16:45 +03:00
type assetFile struct {
fs * assetfs . LayeredFS
name string
path string
2020-02-02 05:17:44 +03:00
}
func initEmbeddedExtractor ( c * cli . Context ) error {
2023-04-12 13:16:45 +03:00
// FIXME: there is a bug, if the user runs `gitea embedded` with a different user or root,
// The setting.Init (loadRunModeFrom) will fail and do log.Fatal
// But the console logger has been deleted, so nothing is printed, the user sees nothing and Gitea just exits.
2020-02-02 05:17:44 +03:00
// Silence the console logger
log . DelNamedLogger ( "console" )
log . DelNamedLogger ( log . DEFAULT )
// Read configuration file
2023-05-04 06:55:35 +03:00
setting . Init ( & setting . Options {
AllowEmpty : true ,
} )
2020-02-02 05:17:44 +03:00
2023-04-12 13:16:45 +03:00
patterns , err := compileCollectPatterns ( c . Args ( ) )
2020-02-02 05:17:44 +03:00
if err != nil {
return err
}
2023-04-12 13:16:45 +03:00
collectAssetFilesByPattern ( c , patterns , "options" , options . BuiltinAssets ( ) )
collectAssetFilesByPattern ( c , patterns , "public" , public . BuiltinAssets ( ) )
collectAssetFilesByPattern ( c , patterns , "templates" , templates . BuiltinAssets ( ) )
2020-02-02 05:17:44 +03:00
return nil
}
func runList ( c * cli . Context ) error {
if err := runListDo ( c ) ; err != nil {
fmt . Fprintf ( os . Stderr , "%v\n" , err )
return err
}
return nil
}
func runView ( c * cli . Context ) error {
if err := runViewDo ( c ) ; err != nil {
fmt . Fprintf ( os . Stderr , "%v\n" , err )
return err
}
return nil
}
func runExtract ( c * cli . Context ) error {
if err := runExtractDo ( c ) ; err != nil {
fmt . Fprintf ( os . Stderr , "%v\n" , err )
return err
}
return nil
}
func runListDo ( c * cli . Context ) error {
if err := initEmbeddedExtractor ( c ) ; err != nil {
return err
}
2023-04-12 13:16:45 +03:00
for _ , a := range matchedAssetFiles {
fmt . Println ( a . path )
2020-02-02 05:17:44 +03:00
}
return nil
}
func runViewDo ( c * cli . Context ) error {
if err := initEmbeddedExtractor ( c ) ; err != nil {
return err
}
2023-04-12 13:16:45 +03:00
if len ( matchedAssetFiles ) == 0 {
return fmt . Errorf ( "no files matched the given pattern" )
} else if len ( matchedAssetFiles ) > 1 {
return fmt . Errorf ( "too many files matched the given pattern, try to be more specific" )
2020-02-02 05:17:44 +03:00
}
2023-04-12 13:16:45 +03:00
data , err := matchedAssetFiles [ 0 ] . fs . ReadFile ( matchedAssetFiles [ 0 ] . name )
2020-02-02 05:17:44 +03:00
if err != nil {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "%s: %w" , matchedAssetFiles [ 0 ] . path , err )
2020-02-02 05:17:44 +03:00
}
if _ , err = os . Stdout . Write ( data ) ; err != nil {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "%s: %w" , matchedAssetFiles [ 0 ] . path , err )
2020-02-02 05:17:44 +03:00
}
return nil
}
func runExtractDo ( c * cli . Context ) error {
if err := initEmbeddedExtractor ( c ) ; err != nil {
return err
}
if len ( c . Args ( ) ) == 0 {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "a list of pattern of files to extract is mandatory (e.g. '**' for all)" )
2020-02-02 05:17:44 +03:00
}
destdir := "."
if c . IsSet ( "destination" ) {
destdir = c . String ( "destination" )
} else if c . Bool ( "custom" ) {
destdir = setting . CustomPath
fmt . Println ( "Using app.ini at" , setting . CustomConf )
}
fi , err := os . Stat ( destdir )
if errors . Is ( err , os . ErrNotExist ) {
// In case Windows users attempt to provide a forward-slash path
wdestdir := filepath . FromSlash ( destdir )
if wfi , werr := os . Stat ( wdestdir ) ; werr == nil {
destdir = wdestdir
fi = wfi
err = nil
}
}
if err != nil {
return fmt . Errorf ( "%s: %s" , destdir , err )
} else if ! fi . IsDir ( ) {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "destination %q is not a directory" , destdir )
2020-02-02 05:17:44 +03:00
}
fmt . Printf ( "Extracting to %s:\n" , destdir )
overwrite := c . Bool ( "overwrite" )
rename := c . Bool ( "rename" )
2023-04-12 13:16:45 +03:00
for _ , a := range matchedAssetFiles {
2020-02-02 05:17:44 +03:00
if err := extractAsset ( destdir , a , overwrite , rename ) ; err != nil {
// Non-fatal error
2023-04-12 13:16:45 +03:00
fmt . Fprintf ( os . Stderr , "%s: %v" , a . path , err )
2020-02-02 05:17:44 +03:00
}
}
return nil
}
2023-04-12 13:16:45 +03:00
func extractAsset ( d string , a assetFile , overwrite , rename bool ) error {
dest := filepath . Join ( d , filepath . FromSlash ( a . path ) )
2020-02-02 05:17:44 +03:00
dir := filepath . Dir ( dest )
2023-04-12 13:16:45 +03:00
data , err := a . fs . ReadFile ( a . name )
2020-02-02 05:17:44 +03:00
if err != nil {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "%s: %w" , a . path , err )
2020-02-02 05:17:44 +03:00
}
if err := os . MkdirAll ( dir , os . ModePerm ) ; err != nil {
2022-10-24 22:29:17 +03:00
return fmt . Errorf ( "%s: %w" , dir , err )
2020-02-02 05:17:44 +03:00
}
2022-01-20 20:46:10 +03:00
perms := os . ModePerm & 0 o666
2020-02-02 05:17:44 +03:00
fi , err := os . Lstat ( dest )
if err != nil {
if ! errors . Is ( err , os . ErrNotExist ) {
2022-10-24 22:29:17 +03:00
return fmt . Errorf ( "%s: %w" , dest , err )
2020-02-02 05:17:44 +03:00
}
} else if ! overwrite && ! rename {
fmt . Printf ( "%s already exists; skipped.\n" , dest )
return nil
} else if ! fi . Mode ( ) . IsRegular ( ) {
return fmt . Errorf ( "%s already exists, but it's not a regular file" , dest )
} else if rename {
2021-07-15 18:46:07 +03:00
if err := util . Rename ( dest , dest + ".bak" ) ; err != nil {
2023-04-12 13:16:45 +03:00
return fmt . Errorf ( "error creating backup for %s: %w" , dest , err )
2020-02-02 05:17:44 +03:00
}
// Attempt to respect file permissions mask (even if user:group will be set anew)
perms = fi . Mode ( )
}
file , err := os . OpenFile ( dest , os . O_WRONLY | os . O_TRUNC | os . O_CREATE , perms )
if err != nil {
2022-10-24 22:29:17 +03:00
return fmt . Errorf ( "%s: %w" , dest , err )
2020-02-02 05:17:44 +03:00
}
defer file . Close ( )
if _ , err = file . Write ( data ) ; err != nil {
2022-10-24 22:29:17 +03:00
return fmt . Errorf ( "%s: %w" , dest , err )
2020-02-02 05:17:44 +03:00
}
fmt . Println ( dest )
return nil
}
2023-04-12 13:16:45 +03:00
func collectAssetFilesByPattern ( c * cli . Context , globs [ ] glob . Glob , path string , layer * assetfs . Layer ) {
fs := assetfs . Layered ( layer )
files , err := fs . ListAllFiles ( "." , true )
if err != nil {
log . Error ( "Error listing files in %q: %v" , path , err )
return
}
for _ , name := range files {
if path == "public" &&
strings . HasPrefix ( name , "vendor/" ) &&
! c . Bool ( "include-vendored" ) {
continue
}
matchName := path + "/" + name
for _ , g := range globs {
if g . Match ( matchName ) {
matchedAssetFiles = append ( matchedAssetFiles , assetFile { fs : fs , name : name , path : path + "/" + name } )
break
2020-02-02 05:17:44 +03:00
}
}
}
}
2023-04-12 13:16:45 +03:00
func compileCollectPatterns ( args [ ] string ) ( [ ] glob . Glob , error ) {
2020-02-02 05:17:44 +03:00
if len ( args ) == 0 {
args = [ ] string { "**" }
}
pat := make ( [ ] glob . Glob , len ( args ) )
for i := range args {
if g , err := glob . Compile ( args [ i ] , '/' ) ; err != nil {
2022-10-24 22:29:17 +03:00
return nil , fmt . Errorf ( "'%s': Invalid glob pattern: %w" , args [ i ] , err )
2023-04-12 13:16:45 +03:00
} else { //nolint:revive
2020-02-02 05:17:44 +03:00
pat [ i ] = g
}
}
return pat , nil
}