mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
gitRepo.cpp
Go to the documentation of this file.
1/** \file gitRepo.cpp
2 * \author Jared R. Males
3 * \brief Interrogate the current state of a git repository (definitions)
4 * \ingroup utils_files
5 */
6
7//***********************************************************************//
8// Copyright 2021 Jared R. Males (jaredmales@gmail.com)
9//
10// This file is part of mxlib.
11//
12// mxlib is free software: you can redistribute it and/or modify
13// it under the terms of the GNU General Public License as published by
14// the Free Software Foundation, either version 3 of the License, or
15// (at your option) any later version.
16//
17// mxlib is distributed in the hope that it will be useful,
18// but WITHOUT ANY WARRANTY; without even the implied warranty of
19// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20// GNU General Public License for more details.
21//
22// You should have received a copy of the GNU General Public License
23// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
24//***********************************************************************//
25
26#include "mxError.hpp"
27#include "sys/gitRepo.hpp"
28
31#include "ioutils/fileUtils.hpp"
32
33namespace mx
34{
35namespace sys
36{
37
41
42gitRepo::gitRepo( const std::string &d )
43{
44 dir( d );
45}
46
47void gitRepo::dir( const std::string &d )
48{
49 m_dir = d;
50
51 getGitName();
52 getGitHash();
55}
56
57std::string gitRepo::dir()
58{
59 return m_dir;
60}
61
62std::string gitRepo::gitDir()
63{
64 return m_dir + "/.git";
65}
66
68{
69 int retVal;
70 std::vector<std::string> stdOut, stdErr;
71
72 if( ipc::runCommand( retVal, stdOut, stdErr, { "git", "--git-dir=" + gitDir(), "rev-parse", "--show-toplevel" } ) <
73 0 )
74 {
75 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error running git command" );
76 return -1;
77 }
78
79 if( stdErr.size() > 0 )
80 {
81 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error returned by git" );
82
83 for( size_t n = 0; n < stdErr.size(); ++n )
84 {
85 std::cerr << "err: " << stdErr[n] << "\n";
86 }
87
88 return -1;
89 }
90
91 if( stdOut.size() < 1 )
92 {
93 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "nothing returned by git" );
94 return -1;
95 }
96
97 if( stdOut.size() > 1 )
98 {
99 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "too much returned by git" );
100 return -1;
101 }
102
103 m_name = ioutils::pathFilename( stdOut[0] );
104
105 return 0;
106}
107
109{
110 int retVal;
111 std::vector<std::string> stdOut, stdErr;
112
113 if( ipc::runCommand( retVal,
114 stdOut,
115 stdErr,
116 { "git", "--git-dir=" + gitDir(), "--work-tree=" + dir(), "log", "-1", "--format=%H" } ) < 0 )
117 {
118 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error running git command" );
119 return -1;
120 }
121
122 if( stdErr.size() > 0 )
123 {
124 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error returned by git" );
125
126 for( size_t n = 0; n < stdErr.size(); ++n )
127 {
128 std::cerr << "err: " << stdErr[n] << "\n";
129 }
130
131 return -1;
132 }
133
134 if( stdOut.size() < 1 )
135 {
136 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "nothing returned by git" );
137 return -1;
138 }
139
140 if( stdOut.size() > 1 )
141 {
142 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "too much returned by git" );
143 return -1;
144 }
145
146 m_hash = stdOut[0];
147
148 return 0;
149}
150
152{
153 int retVal;
154 std::vector<std::string> stdOut, stdErr;
155
156 if( ipc::runCommand( retVal,
157 stdOut,
158 stdErr,
159 { "git", "--git-dir=" + gitDir(), "--work-tree=" + dir(), "diff-index", "HEAD", "--" } ) < 0 )
160 {
161 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error running git command" );
162 return -1;
163 }
164
165 if( stdErr.size() > 0 )
166 {
167 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error returned by git" );
168 for( size_t n = 0; n < stdErr.size(); ++n )
169 {
170 std::cerr << "err: " << stdErr[n] << "\n";
171 }
172
173 return -1;
174 }
175
176 m_modified = ( stdOut.size() > 0 ); // rv;
177
178 return 0;
179}
180
182{
183 int retVal;
184 std::vector<std::string> stdOut, stdErr;
185
186 if( ipc::runCommand(
187 retVal, stdOut, stdErr, { "git", "--git-dir=" + gitDir(), "--work-tree=" + dir(), "status" } ) < 0 )
188 {
189 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error running git command" );
190 return -1;
191 }
192
193 if( stdErr.size() > 0 )
194 {
195 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "error returned by git" );
196
197 for( size_t n = 0; n < stdErr.size(); ++n )
198 {
199 std::cerr << "err: " << stdErr[n] << "\n";
200 }
201
202 return -1;
203 }
204
205 if( stdOut.size() < 1 )
206 {
207 mxError( "mx::sys::gitRepo::getGitName", MXE_PROCERR, "nothing returned by git" );
208 return -1;
209 }
210
211 m_modifiedFiles.clear();
212 m_deletedFiles.clear();
213 m_renamedFiles.clear();
214 m_renamedFiles2.clear();
215 m_untrackedFiles.clear();
216
217 for( size_t n = 0; n < stdOut.size(); ++n )
218 {
219 if( stdOut[n].find( "On b" ) != std::string::npos )
220 {
221 m_branch = stdOut[n].substr( sizeof( "On branch" ) );
222 }
223 else if( stdOut[n].find( "Changes not" ) != std::string::npos )
224 {
225 // Changes not staged for commit
226
227 ++n;
228 while( stdOut[n].size() > 0 )
229 {
230 if( n >= stdOut.size() )
231 break;
232
233 if( stdOut[n].find( "mod" ) != std::string::npos )
234 {
235 m_modifiedFiles.insert( ioutils::removeWhiteSpace( stdOut[n].substr( sizeof( "modified:" ) ) ) );
236 }
237 else if( stdOut[n].find( "del" ) != std::string::npos )
238 {
239 m_deletedFiles.insert( ioutils::removeWhiteSpace( stdOut[n].substr( sizeof( "deleted:" ) ) ) );
240 }
241 else if( stdOut[n].find( "ren" ) != std::string::npos )
242 {
243 size_t a = stdOut[n].find( " -> " );
244
246 stdOut[n].substr( sizeof( "renamed:" ), a - sizeof( "renamed:" ) ) ) );
247 m_renamedFiles2.insert( ioutils::removeWhiteSpace( stdOut[n].substr( a + sizeof( " ->" ) ) ) );
248 }
249 ++n;
250 }
251 }
252 else if( stdOut[n].find( "Changes to" ) != std::string::npos )
253 {
254 // Changes to be committed
255
256 ++n;
257 while( stdOut[n].size() > 0 )
258 {
259 if( n >= stdOut.size() )
260 break;
261
262 if( stdOut[n].find( "mod" ) != std::string::npos )
263 {
264 m_modifiedFiles.insert( ioutils::removeWhiteSpace( stdOut[n].substr( sizeof( "modified:" ) ) ) );
265 }
266 else if( stdOut[n].find( "del" ) != std::string::npos )
267 {
268 m_deletedFiles.insert( ioutils::removeWhiteSpace( stdOut[n].substr( sizeof( "deleted:" ) ) ) );
269 }
270 else if( stdOut[n].find( "ren" ) != std::string::npos )
271 {
272 size_t a = stdOut[n].find( " -> " );
273
275 stdOut[n].substr( sizeof( "renamed: " ), a - sizeof( "renamed:" ) ) ) );
276 m_renamedFiles2.insert( ioutils::removeWhiteSpace( stdOut[n].substr( a + sizeof( " ->" ) ) ) );
277 }
278 ++n;
279 }
280 }
281 else if( stdOut[n].find( "Untracked" ) != std::string::npos )
282 {
283 // Untracked files:
284
285 ++n;
286 ++n;
287 while( stdOut[n].size() > 0 )
288 {
289 if( n >= stdOut.size() )
290 break;
291 m_untrackedFiles.insert( ioutils::removeWhiteSpace( stdOut[n] ) );
292 ++n;
293 }
294 }
295 }
296
297 return 0;
298}
299
300std::string gitRepo::name()
301{
302 return m_name;
303}
304
305std::string gitRepo::branch()
306{
307 return m_branch;
308}
309
310std::string gitRepo::hash()
311{
312 return m_hash;
313}
314
316{
317 return m_modified;
318}
319
320bool gitRepo::isNotCommitted( const std::string &file )
321{
322 return ( m_modifiedFiles.count( file ) + m_deletedFiles.count( file ) + m_renamedFiles.count( file ) +
323 m_renamedFiles2.count( file ) + m_untrackedFiles.count( file ) >
324 0 );
325}
326
327} // namespace sys
328} // namespace mx
int getGitName()
Get the name of the git repo.
Definition gitRepo.cpp:67
std::string gitDir()
Get the current repo's .git directory.
Definition gitRepo.cpp:62
std::set< std::string > m_renamedFiles
Files which git lists as renamed-from.
Definition gitRepo.hpp:66
std::string m_branch
The current branch.
Definition gitRepo.hpp:60
std::set< std::string > m_renamedFiles2
Files which git lists as renamed-to.
Definition gitRepo.hpp:67
bool isNotCommitted(const std::string &file)
Check whether a file is listed as not committed.
Definition gitRepo.cpp:320
std::string m_dir
The directory of the git repository.
Definition gitRepo.hpp:56
int getGitHash()
Get the name of the current commit hash.
Definition gitRepo.cpp:108
std::string branch()
Get the current branch.
Definition gitRepo.cpp:305
std::string name()
Get the repo's name.
Definition gitRepo.cpp:300
std::string m_name
The repo name.
Definition gitRepo.hpp:59
std::set< std::string > m_untrackedFiles
Files which git lists as untracked.
Definition gitRepo.hpp:68
std::set< std::string > m_deletedFiles
Files which git lists as deleted.
Definition gitRepo.hpp:65
std::set< std::string > m_modifiedFiles
Files which git lists as modified.
Definition gitRepo.hpp:64
std::string dir()
Get the current directory.
Definition gitRepo.cpp:57
std::string m_hash
The complete commit hash.
Definition gitRepo.hpp:61
bool modified()
Get whether the repo is modified.
Definition gitRepo.cpp:315
gitRepo()
Default c'tor.
Definition gitRepo.cpp:38
int getGitModified()
Get the modification status of the repo.
Definition gitRepo.cpp:151
bool m_modified
The modification status, true or false.
Definition gitRepo.hpp:62
int getGitFileState()
Get the list of modified files, and the branch name.
Definition gitRepo.cpp:181
std::string hash()
Get the current commit hash.
Definition gitRepo.cpp:310
Declarations of utilities for working with files.
Interrogate the current state of a git repository (declarations)
int runCommand(int &retVal, std::vector< std::string > &commandOutput, std::vector< std::string > &commandStderr, const std::vector< std::string > &commandList)
Runs a command (with parameters) passed in using fork/exec.
std::string pathFilename(const std::string &fname)
Get the base filename.
Definition fileUtils.cpp:71
void removeWhiteSpace(std::string &outstr, const std::string &instr)
Remove all white space from a string.
Declares and defines the mxlib error reporting system.
The mxlib c++ namespace.
Definition mxError.hpp:106
Process interface facilities.
Utilities for working with strings.