2018-08-12 19:31:27 +10:00
package app
import (
2019-03-16 11:31:09 +11:00
"bufio"
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"
2018-08-13 21:16:21 +10:00
"os"
2018-12-08 16:54:54 +11:00
"path/filepath"
2019-02-19 23:36:29 +11:00
"strings"
2018-08-12 19:31:27 +10:00
"github.com/jesseduffield/lazygit/pkg/commands"
"github.com/jesseduffield/lazygit/pkg/config"
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"
2018-08-19 23:28:29 +10:00
"github.com/jesseduffield/lazygit/pkg/updates"
2018-12-08 16:54:54 +11:00
"github.com/shibukawa/configdir"
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
OSCommand * commands . OSCommand
GitCommand * commands . GitCommand
Gui * gui . Gui
Tr * i18n . Localizer
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
}
2018-12-08 16:54:54 +11:00
func globalConfigDir ( ) string {
configDirs := configdir . New ( "jesseduffield" , "lazygit" )
configDir := configDirs . QueryFolders ( configdir . Global ) [ 0 ]
return configDir . Path
}
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
}
2018-12-08 16:54:54 +11:00
func newDevelopmentLogger ( config config . AppConfigurer ) * logrus . Logger {
2018-08-26 15:46:18 +10:00
log := logrus . New ( )
2019-03-16 10:37:31 +11:00
log . SetLevel ( getLogLevel ( ) )
2018-12-08 16:54:54 +11:00
file , err := os . OpenFile ( filepath . Join ( globalConfigDir ( ) , "development.log" ) , 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)
}
2018-08-14 18:02:27 +10:00
log . SetOutput ( file )
2018-08-13 21:16:21 +10:00
return log
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 ) {
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 )
2018-08-20 21:04:04 +02:00
app . Tr = i18n . NewLocalizer ( 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
}
app . OSCommand = commands . NewOSCommand ( app . Log , config )
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-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-16 22:49:37 +10:00
func ( app * App ) setupRepo ( ) ( bool , error ) {
2019-03-16 11:31:09 +11:00
// if we are not in a git repo, we ask if we want to `git init`
if err := app . OSCommand . RunCommand ( "git status" ) ; 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
}
// Offer to initialize a new repository in current directory.
2019-03-16 11:31:09 +11:00
fmt . Print ( app . Tr . SLocalize ( "CreateRepo" ) )
response , _ := bufio . NewReader ( os . Stdin ) . ReadString ( '\n' )
if strings . Trim ( response , " \n" ) != "y" {
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
}
}
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 )
}
2019-07-07 19:15:11 +01:00
err := app . Gui . RunWithSubprocesses ( )
return err
2019-02-18 19:42:23 +11:00
}
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
}
2019-02-19 23:36:29 +11:00
} else if strings . HasSuffix ( os . Args [ 1 ] , ".git/COMMIT_EDITMSG" ) {
// 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 ( )
mappings := [ ] errorMapping {
{
2020-08-16 22:49:37 +10:00
originalError : "fatal: not a git repository" ,
2019-07-07 19:15:11 +01:00
newError : app . Tr . SLocalize ( "notARepository" ) ,
} ,
}
for _ , mapping := range mappings {
if strings . Contains ( errorMessage , mapping . originalError ) {
return mapping . newError , true
}
}
return "" , false
}