29#include <wx/filename.h>
33#include <wx/datetime.h>
37#define HISTORY_LOCK_TRACE wxT( "KICAD_HISTORY_LOCK" )
55 wxLogTrace(
HISTORY_LOCK_TRACE,
"Attempting to acquire lock for project: %s (timeout: %d sec)",
79 wxLogTrace(
HISTORY_LOCK_TRACE,
"Successfully acquired all locks for project: %s", aProjectPath );
96 git_repository_free(
m_repo );
120 return wxString::Format( wxS(
"%s@%s" ),
124 return wxEmptyString;
133 if( !wxFileName::Mkdir(
m_historyPath, 0777, wxPATH_MKDIR_FULL ) )
144 wxLogWarning(
"Detected stale lock file, removing it" );
152 m_fileLock = std::make_unique<LOCKFILE>( lockPath.GetFullPath(),
true );
157 _(
"History repository is locked by %s@%s" ),
181 const git_error* err = git_error_last();
183 _(
"Failed to open git repository: %s" ),
184 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
193 const git_error* err = git_error_last();
195 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
203 if( ignoreFile.FileExists() )
205 wxFFile f( ignoreFile.GetFullPath(), wxT(
"rb" ) );
208 if( f.IsOpened() && f.ReadAll( &content ) )
209 git_ignore_add_rule(
m_repo, content.utf8_str() );
221 m_lockError =
_(
"Cannot acquire index lock: repository not open" );
230 const git_error* err = git_error_last();
232 _(
"Failed to acquire git index lock (another operation in progress?): %s" ),
233 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
245 if( aStaleTimeoutSec <= 0 )
249 wxFileName lockPath( histPath, wxS(
".repo.lock" ) );
251 if( !lockPath.FileExists() )
255 if( !lockPath.GetTimes(
nullptr, &modTime,
nullptr ) )
258 wxDateTime now = wxDateTime::Now();
259 wxTimeSpan age = now - modTime;
262 (
int)age.GetSeconds().ToLong(), aStaleTimeoutSec );
264 return age.GetSeconds().ToLong() > aStaleTimeoutSec;
271 wxFileName lockPath( histPath, wxS(
".repo.lock" ) );
273 if( !lockPath.FileExists() )
277 wxFileName lockFilePath( histPath, wxS(
".repo" ) );
283 if( lockPath.FileExists() )
285 result = wxRemoveFile( lockPath.GetFullPath() );
289 wxLogError(
"Failed to remove stale lock: %s", lockPath.GetFullPath() );
292 if( lockFilePath.FileExists() )
294 bool lockFileResult = wxRemoveFile( lockFilePath.GetFullPath() );
296 wxLogTrace(
HISTORY_LOCK_TRACE,
"Removed stale lockfile: %s", lockFilePath.GetFullPath() );
315 git_repository_free(
m_repo );
static const ADVANCED_CFG & GetCfg()
Get the singleton instance's config, which is shared by all consumers.
wxString GetLockHolder() const
Get information about who currently holds the lock.
HISTORY_LOCK_MANAGER(const wxString &aProjectPath, int aStaleTimeoutSec=0)
Construct a lock manager and attempt to acquire locks.
void ReleaseRepository()
Release git repository and index handles early, but keep the file lock.
~HISTORY_LOCK_MANAGER()
Destructor releases all locks and closes git repository.
wxString GetLockError() const
Get error message describing why lock could not be acquired.
static bool BreakStaleLock(const wxString &aProjectPath)
Forcibly remove a stale lock file.
std::unique_ptr< LOCKFILE > m_fileLock
static bool IsLockStale(const wxString &aProjectPath, int aStaleTimeoutSec=0)
Check if a lock file exists and is stale (older than timeout).
bool IsLocked() const
Check if locks were successfully acquired.
virtual SETTINGS_MANAGER & GetSettingsManager() const
wxString GetLocalHistoryDirForPath(const wxString &aProjectPath) const
Resolve the local-history directory for a project given by its on-disk path.
int m_HistoryLockStaleTimeout
Stale lock timeout for local history repository locks, in seconds.
static const std::string LockFileExtension
static const std::string LockFilePrefix
static wxString historyPath(const wxString &aProjectPath)
#define HISTORY_LOCK_TRACE
PGM_BASE & Pgm()
The global program "get" accessor.
wxString result
Test unit parsing edge cases and error handling.