25#include <wx/filename.h>
29#include <wx/datetime.h>
33#define HISTORY_LOCK_TRACE wxT( "KICAD_HISTORY_LOCK" )
51 wxLogTrace(
HISTORY_LOCK_TRACE,
"Attempting to acquire lock for project: %s (timeout: %d sec)",
75 wxLogTrace(
HISTORY_LOCK_TRACE,
"Successfully acquired all locks for project: %s", aProjectPath );
92 git_repository_free(
m_repo );
116 return wxString::Format( wxS(
"%s@%s" ),
120 return wxEmptyString;
129 if( !wxFileName::Mkdir(
m_historyPath, 0777, wxPATH_MKDIR_FULL ) )
140 wxLogWarning(
"Detected stale lock file, removing it" );
148 m_fileLock = std::make_unique<LOCKFILE>( lockPath.GetFullPath(),
true );
153 _(
"History repository is locked by %s@%s" ),
177 const git_error* err = git_error_last();
179 _(
"Failed to open git repository: %s" ),
180 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
189 const git_error* err = git_error_last();
191 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
199 if( ignoreFile.FileExists() )
201 wxFFile f( ignoreFile.GetFullPath(), wxT(
"rb" ) );
204 if( f.IsOpened() && f.ReadAll( &content ) )
205 git_ignore_add_rule(
m_repo, content.utf8_str() );
217 m_lockError =
_(
"Cannot acquire index lock: repository not open" );
226 const git_error* err = git_error_last();
228 _(
"Failed to acquire git index lock (another operation in progress?): %s" ),
229 err ? wxString::FromUTF8( err->message ) : wxString(
"Unknown error" ) );
241 if( aStaleTimeoutSec <= 0 )
245 wxFileName lockPath( histPath, wxS(
".repo.lock" ) );
247 if( !lockPath.FileExists() )
251 if( !lockPath.GetTimes(
nullptr, &modTime,
nullptr ) )
254 wxDateTime now = wxDateTime::Now();
255 wxTimeSpan age = now - modTime;
258 (
int)age.GetSeconds().ToLong(), aStaleTimeoutSec );
260 return age.GetSeconds().ToLong() > aStaleTimeoutSec;
267 wxFileName lockPath( histPath, wxS(
".repo.lock" ) );
269 if( !lockPath.FileExists() )
273 wxFileName lockFilePath( histPath, wxS(
".repo" ) );
279 if( lockPath.FileExists() )
281 result = wxRemoveFile( lockPath.GetFullPath() );
285 wxLogError(
"Failed to remove stale lock: %s", lockPath.GetFullPath() );
288 if( lockFilePath.FileExists() )
290 bool lockFileResult = wxRemoveFile( lockFilePath.GetFullPath() );
292 wxLogTrace(
HISTORY_LOCK_TRACE,
"Removed stale lockfile: %s", lockFilePath.GetFullPath() );
311 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.