2018-08-12 19:31:27 +10:00
package app
import (
2019-03-16 11:31:09 +11:00
"bufio"
2020-08-27 22:19:03 +10:00
"errors"
2019-03-02 20:32:16 +11:00
"fmt"
2018-08-12 19:31:27 +10:00
"io"
2018-08-13 23:35:01 +10:00
"io/ioutil"
2020-09-26 10:23:10 +10:00
"log"
2018-08-13 21:16:21 +10:00
"os"
2018-12-08 16:54:54 +11:00
"path/filepath"
2020-09-18 21:00:03 +10:00
"regexp"
2020-08-27 22:19:03 +10:00
"strconv"
2019-02-19 23:36:29 +11:00
"strings"
2018-08-12 19:31:27 +10:00
2020-09-26 10:23:10 +10:00
"github.com/aybabtme/humanlog"
2018-08-12 19:31:27 +10:00
"github.com/jesseduffield/lazygit/pkg/commands"
2020-09-29 19:10:57 +10:00
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
2018-08-12 19:31:27 +10:00
"github.com/jesseduffield/lazygit/pkg/config"
2020-09-27 16:17:26 +10:00
"github.com/jesseduffield/lazygit/pkg/env"
2018-08-13 20:26:02 +10:00
"github.com/jesseduffield/lazygit/pkg/gui"
2018-08-14 15:26:25 +02:00
"github.com/jesseduffield/lazygit/pkg/i18n"
2020-12-21 09:37:48 +11:00
"github.com/jesseduffield/lazygit/pkg/secureexec"
2018-08-19 23:28:29 +10:00
"github.com/jesseduffield/lazygit/pkg/updates"
2018-08-25 15:55:49 +10:00
"github.com/sirupsen/logrus"
2018-08-12 19:31:27 +10:00
)
// App struct
type App struct {
closers [ ] io . Closer
2019-03-03 12:44:10 +11:00
Config config . AppConfigurer
Log * logrus . Entry
2020-09-29 19:10:57 +10:00
OSCommand * oscommands . OSCommand
2019-03-03 12:44:10 +11:00
GitCommand * commands . GitCommand
Gui * gui . Gui
2020-10-04 11:00:48 +11:00
Tr * i18n . TranslationSet
2019-03-03 12:44:10 +11:00
Updater * updates . Updater // may only need this on the Gui
ClientContext string
2018-08-13 21:16:21 +10:00
}
2019-07-07 19:15:11 +01:00
type errorMapping struct {
originalError string
newError string
}
2018-08-26 15:46:18 +10:00
func newProductionLogger ( config config . AppConfigurer ) * logrus . Logger {
2018-08-13 21:16:21 +10:00
log := logrus . New ( )
2018-08-26 15:46:18 +10:00
log . Out = ioutil . Discard
2019-03-16 10:37:31 +11:00
log . SetLevel ( logrus . ErrorLevel )
2018-08-26 15:46:18 +10:00
return log
}
2019-03-16 10:37:31 +11:00
func getLogLevel ( ) logrus . Level {
strLevel := os . Getenv ( "LOG_LEVEL" )
level , err := logrus . ParseLevel ( strLevel )
if err != nil {
return logrus . DebugLevel
}
return level
}
2020-09-26 10:23:10 +10:00
func newDevelopmentLogger ( configurer config . AppConfigurer ) * logrus . Logger {
2020-10-03 14:54:55 +10:00
logger := logrus . New ( )
logger . SetLevel ( getLogLevel ( ) )
logPath , err := config . LogPath ( )
if err != nil {
log . Fatal ( err )
}
file , err := os . OpenFile ( logPath , os . O_CREATE | os . O_WRONLY | os . O_APPEND , 0666 )
2018-08-13 21:16:21 +10:00
if err != nil {
panic ( "unable to log to file" ) // TODO: don't panic (also, remove this call to the `panic` function)
}
2020-10-03 14:54:55 +10:00
logger . SetOutput ( file )
return logger
2018-08-12 19:31:27 +10:00
}
2018-08-26 15:46:18 +10:00
func newLogger ( config config . AppConfigurer ) * logrus . Entry {
var log * logrus . Logger
2019-02-18 21:29:43 +11:00
if config . GetDebug ( ) || os . Getenv ( "DEBUG" ) == "TRUE" {
2018-12-08 16:54:54 +11:00
log = newDevelopmentLogger ( config )
2018-08-26 15:46:18 +10:00
} else {
log = newProductionLogger ( config )
}
2018-12-07 18:52:31 +11:00
// highly recommended: tail -f development.log | humanlog
// https://github.com/aybabtme/humanlog
2018-12-04 19:50:11 +11:00
log . Formatter = & logrus . JSONFormatter { }
2018-08-26 15:46:18 +10:00
return log . WithFields ( logrus . Fields {
"debug" : config . GetDebug ( ) ,
"version" : config . GetVersion ( ) ,
"commit" : config . GetCommit ( ) ,
"buildDate" : config . GetBuildDate ( ) ,
} )
}
2019-02-18 19:42:23 +11:00
// NewApp bootstrap a new application
2020-03-29 10:11:15 +11:00
func NewApp ( config config . AppConfigurer , filterPath string ) ( * App , error ) {
2021-03-30 22:17:42 +11:00
2018-08-12 19:31:27 +10:00
app := & App {
closers : [ ] io . Closer { } ,
Config : config ,
}
var err error
2018-08-13 21:16:21 +10:00
app . Log = newLogger ( config )
2020-10-04 11:00:48 +11:00
app . Tr = i18n . NewTranslationSet ( app . Log )
2018-08-14 22:12:07 +10:00
2019-02-18 21:29:43 +11:00
// if we are being called in 'demon' mode, we can just return here
2019-03-03 12:44:10 +11:00
app . ClientContext = os . Getenv ( "LAZYGIT_CLIENT_COMMAND" )
if app . ClientContext != "" {
2019-02-18 21:29:43 +11:00
return app , nil
}
2020-09-29 19:10:57 +10:00
app . OSCommand = oscommands . NewOSCommand ( app . Log , config )
2019-02-18 21:29:43 +11:00
2018-09-07 09:41:15 +10:00
app . Updater , err = updates . NewUpdater ( app . Log , config , app . OSCommand , app . Tr )
2018-08-12 19:31:27 +10:00
if err != nil {
2018-08-18 19:43:58 +10:00
return app , err
2018-08-12 19:31:27 +10:00
}
2019-03-16 11:31:09 +11:00
2020-08-16 22:49:37 +10:00
showRecentRepos , err := app . setupRepo ( )
if err != nil {
2019-03-16 11:31:09 +11:00
return app , err
}
2019-02-18 21:29:43 +11:00
app . GitCommand , err = commands . NewGitCommand ( app . Log , app . OSCommand , app . Tr , app . Config )
2018-08-19 23:28:29 +10:00
if err != nil {
return app , err
}
2020-09-27 15:36:04 +10:00
2020-08-16 22:49:37 +10:00
app . Gui , err = gui . NewGui ( app . Log , app . GitCommand , app . OSCommand , app . Tr , config , app . Updater , filterPath , showRecentRepos )
2018-08-13 20:26:02 +10:00
if err != nil {
2018-08-18 19:43:58 +10:00
return app , err
2018-08-13 20:26:02 +10:00
}
2018-08-12 19:31:27 +10:00
return app , nil
}
2020-08-27 22:19:03 +10:00
func ( app * App ) validateGitVersion ( ) error {
output , err := app . OSCommand . RunCommandWithOutput ( "git --version" )
// if we get an error anywhere here we'll show the same status
2020-10-04 11:00:48 +11:00
minVersionError := errors . New ( app . Tr . MinGitVersionError )
2020-08-27 22:19:03 +10:00
if err != nil {
return minVersionError
}
2020-09-18 21:00:03 +10:00
if isGitVersionValid ( output ) {
return nil
}
return minVersionError
}
func isGitVersionValid ( versionStr string ) bool {
// output should be something like: 'git version 2.23.0 (blah)'
re := regexp . MustCompile ( ` [^\d]+([\d\.]+) ` )
matches := re . FindStringSubmatch ( versionStr )
if len ( matches ) == 0 {
return false
}
gitVersion := matches [ 1 ]
2020-08-27 22:19:03 +10:00
majorVersion , err := strconv . Atoi ( gitVersion [ 0 : 1 ] )
if err != nil {
2020-09-18 21:00:03 +10:00
return false
2020-08-27 22:19:03 +10:00
}
if majorVersion < 2 {
2020-09-18 21:00:03 +10:00
return false
2020-08-27 22:19:03 +10:00
}
2020-09-18 21:00:03 +10:00
return true
2020-08-27 22:19:03 +10:00
}
2020-08-16 22:49:37 +10:00
func ( app * App ) setupRepo ( ) ( bool , error ) {
2020-08-27 22:19:03 +10:00
if err := app . validateGitVersion ( ) ; err != nil {
return false , err
}
2020-09-27 16:17:26 +10:00
if env . GetGitDirEnv ( ) != "" {
2020-09-27 15:36:04 +10:00
// we've been given the git dir directly. We'll verify this dir when initializing our GitCommand object
return false , nil
}
2019-03-16 11:31:09 +11:00
// if we are not in a git repo, we ask if we want to `git init`
2021-03-30 22:17:42 +11:00
if err := commands . VerifyInGitRepo ( app . OSCommand ) ; err != nil {
2020-05-28 07:20:13 +03:00
cwd , err := os . Getwd ( )
if err != nil {
2020-08-16 22:49:37 +10:00
return false , err
2019-03-16 11:31:09 +11:00
}
2020-05-28 07:20:13 +03:00
info , _ := os . Stat ( filepath . Join ( cwd , ".git" ) )
if info != nil && info . IsDir ( ) {
2020-08-16 22:49:37 +10:00
return false , err // Current directory appears to be a git repository.
2020-05-28 07:20:13 +03:00
}
2020-11-24 21:21:11 +10:00
shouldInitRepo := true
notARepository := app . Config . GetUserConfig ( ) . NotARepository
2020-11-24 22:02:31 +10:00
if notARepository == "prompt" {
2020-11-24 21:21:11 +10:00
// Offer to initialize a new repository in current directory.
fmt . Print ( app . Tr . CreateRepo )
response , _ := bufio . NewReader ( os . Stdin ) . ReadString ( '\n' )
if strings . Trim ( response , " \n" ) != "y" {
shouldInitRepo = false
}
2020-11-24 22:02:31 +10:00
} else if notARepository == "skip" {
2020-11-24 21:21:11 +10:00
shouldInitRepo = false
}
if ! shouldInitRepo {
2020-08-16 22:49:37 +10:00
// check if we have a recent repo we can open
recentRepos := app . Config . GetAppState ( ) . RecentRepos
if len ( recentRepos ) > 0 {
var err error
// try opening each repo in turn, in case any have been deleted
for _ , repoDir := range recentRepos {
if err = os . Chdir ( repoDir ) ; err == nil {
return true , nil
}
}
return false , err
}
2019-03-16 11:31:09 +11:00
os . Exit ( 1 )
}
if err := app . OSCommand . RunCommand ( "git init" ) ; err != nil {
2020-08-16 22:49:37 +10:00
return false , err
2019-03-16 11:31:09 +11:00
}
}
2021-03-30 22:17:42 +11:00
2020-08-16 22:49:37 +10:00
return false , nil
2019-03-16 11:31:09 +11:00
}
2019-02-18 19:42:23 +11:00
func ( app * App ) Run ( ) error {
2019-03-03 12:44:10 +11:00
if app . ClientContext == "INTERACTIVE_REBASE" {
2019-02-18 21:29:43 +11:00
return app . Rebase ( )
}
2019-03-03 12:44:10 +11:00
if app . ClientContext == "EXIT_IMMEDIATELY" {
os . Exit ( 0 )
}
2021-04-03 13:43:43 +11:00
err := app . Gui . RunAndHandleError ( )
2019-07-07 19:15:11 +01:00
return err
2019-02-18 19:42:23 +11:00
}
2020-09-27 15:36:04 +10:00
func gitDir ( ) string {
2020-09-27 16:17:26 +10:00
dir := env . GetGitDirEnv ( )
2020-09-27 15:36:04 +10:00
if dir == "" {
return ".git"
}
return dir
}
2019-02-19 23:36:29 +11:00
// Rebase contains logic for when we've been run in demon mode, meaning we've
// given lazygit as a command for git to call e.g. to edit a file
2019-02-18 21:29:43 +11:00
func ( app * App ) Rebase ( ) error {
2019-02-19 23:36:29 +11:00
app . Log . Info ( "Lazygit invoked as interactive rebase demon" )
app . Log . Info ( "args: " , os . Args )
if strings . HasSuffix ( os . Args [ 1 ] , "git-rebase-todo" ) {
2019-03-02 13:22:02 +11:00
if err := ioutil . WriteFile ( os . Args [ 1 ] , [ ] byte ( os . Getenv ( "LAZYGIT_REBASE_TODO" ) ) , 0644 ) ; err != nil {
return err
}
2020-09-27 15:36:04 +10:00
} else if strings . HasSuffix ( os . Args [ 1 ] , filepath . Join ( gitDir ( ) , "COMMIT_EDITMSG" ) ) { // TODO: test
2019-02-19 23:36:29 +11:00
// if we are rebasing and squashing, we'll see a COMMIT_EDITMSG
// but in this case we don't need to edit it, so we'll just return
} else {
app . Log . Info ( "Lazygit demon did not match on any use cases" )
}
2019-02-18 21:29:43 +11:00
return nil
}
2018-08-12 19:31:27 +10:00
// Close closes any resources
func ( app * App ) Close ( ) error {
for _ , closer := range app . closers {
err := closer . Close ( )
if err != nil {
return err
}
}
return nil
}
2019-07-07 19:15:11 +01:00
// KnownError takes an error and tells us whether it's an error that we know about where we can print a nicely formatted version of it rather than panicking with a stack trace
func ( app * App ) KnownError ( err error ) ( string , bool ) {
errorMessage := err . Error ( )
2020-10-04 11:00:48 +11:00
knownErrorMessages := [ ] string { app . Tr . MinGitVersionError }
2020-08-27 22:19:03 +10:00
for _ , message := range knownErrorMessages {
if errorMessage == message {
return message , true
}
}
2019-07-07 19:15:11 +01:00
mappings := [ ] errorMapping {
{
2020-08-16 22:49:37 +10:00
originalError : "fatal: not a git repository" ,
2020-10-04 11:00:48 +11:00
newError : app . Tr . NotARepository ,
2019-07-07 19:15:11 +01:00
} ,
}
for _ , mapping := range mappings {
if strings . Contains ( errorMessage , mapping . originalError ) {
return mapping . newError , true
}
}
return "" , false
}
2020-09-26 10:23:10 +10:00
func TailLogs ( ) {
2020-10-03 14:54:55 +10:00
logFilePath , err := config . LogPath ( )
if err != nil {
log . Fatal ( err )
}
2020-09-26 10:23:10 +10:00
fmt . Printf ( "Tailing log file %s\n\n" , logFilePath )
2020-10-03 14:54:55 +10:00
_ , err = os . Stat ( logFilePath )
2020-09-26 10:23:10 +10:00
if err != nil {
if os . IsNotExist ( err ) {
log . Fatal ( "Log file does not exist. Run `lazygit --debug` first to create the log file" )
}
log . Fatal ( err )
}
2020-12-21 09:37:48 +11:00
cmd := secureexec . Command ( "tail" , "-f" , logFilePath )
2020-09-26 10:23:10 +10:00
stdout , _ := cmd . StdoutPipe ( )
if err := cmd . Start ( ) ; err != nil {
log . Fatal ( err )
}
opts := humanlog . DefaultOptions
opts . Truncates = false
if err := humanlog . Scanner ( stdout , os . Stdout , opts ) ; err != nil {
log . Fatal ( err )
}
if err := cmd . Wait ( ) ; err != nil {
log . Fatal ( err )
}
os . Exit ( 0 )
}