Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
G
git2
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
lvzhengyang
git2
Commits
bd72425d
Commit
bd72425d
authored
Jul 18, 2012
by
nulltoken
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
reflog: introduce git_reflog_write()
parent
d284b3de
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
146 additions
and
61 deletions
+146
-61
include/git2/reflog.h
+9
-0
src/reflog.c
+113
-57
src/reflog.h
+1
-0
tests-clar/refs/reflog/drop.c
+19
-0
tests-clar/refs/reflog/reflog.c
+4
-4
No files found.
include/git2/reflog.h
View file @
bd72425d
...
@@ -33,6 +33,15 @@ GIT_BEGIN_DECL
...
@@ -33,6 +33,15 @@ GIT_BEGIN_DECL
GIT_EXTERN
(
int
)
git_reflog_read
(
git_reflog
**
reflog
,
git_reference
*
ref
);
GIT_EXTERN
(
int
)
git_reflog_read
(
git_reflog
**
reflog
,
git_reference
*
ref
);
/**
/**
* Write an existing in-memory reflog object back to disk
* using an atomic file lock.
*
* @param reflog an existing reflog object
* @return 0 or an error code
*/
GIT_EXTERN
(
int
)
git_reflog_write
(
git_reflog
*
reflog
);
/**
* Add a new entry to the reflog for the given reference
* Add a new entry to the reflog for the given reference
*
*
* If there is no reflog file for the given
* If there is no reflog file for the given
...
...
src/reflog.c
View file @
bd72425d
...
@@ -28,65 +28,83 @@ static int reflog_init(git_reflog **reflog, git_reference *ref)
...
@@ -28,65 +28,83 @@ static int reflog_init(git_reflog **reflog, git_reference *ref)
return
-
1
;
return
-
1
;
}
}
log
->
owner
=
git_reference_owner
(
ref
);
*
reflog
=
log
;
*
reflog
=
log
;
return
0
;
return
0
;
}
}
static
int
reflog_write
(
const
char
*
log_path
,
const
char
*
oid_old
,
static
int
serialize_reflog_entry
(
const
char
*
oid_new
,
const
git_signature
*
committer
,
git_buf
*
buf
,
const
char
*
msg
)
const
git_oid
*
oid_old
,
const
git_oid
*
oid_new
,
const
git_signature
*
committer
,
const
char
*
msg
)
{
{
int
error
;
char
raw_old
[
GIT_OID_HEXSZ
+
1
];
git_buf
log
=
GIT_BUF_INIT
;
char
raw_new
[
GIT_OID_HEXSZ
+
1
];
git_filebuf
fbuf
=
GIT_FILEBUF_INIT
;
bool
trailing_newline
=
false
;
assert
(
log_path
&&
oid_old
&&
oid_new
&&
committer
);
git_oid_tostr
(
raw_old
,
GIT_OID_HEXSZ
+
1
,
oid_old
);
git_oid_tostr
(
raw_new
,
GIT_OID_HEXSZ
+
1
,
oid_new
);
git_buf_clear
(
buf
);
git_buf_puts
(
buf
,
raw_old
);
git_buf_putc
(
buf
,
' '
);
git_buf_puts
(
buf
,
raw_new
);
git_signature__writebuf
(
buf
,
" "
,
committer
);
/* drop trailing LF */
git_buf_rtrim
(
buf
);
if
(
msg
)
{
if
(
msg
)
{
const
char
*
newline
=
strchr
(
msg
,
'\n'
);
const
char
*
newline
=
strchr
(
msg
,
'\n'
);
if
(
newline
)
{
if
(
*
(
newline
+
1
)
==
'\0'
)
if
(
newline
&&
newline
[
1
]
!=
'\0'
)
{
trailing_newline
=
true
;
giterr_set
(
GITERR_INVALID
,
"Reflog message cannot contain newline"
);
else
{
return
-
1
;
giterr_set
(
GITERR_INVALID
,
"Reflog message cannot contain newline"
);
return
-
1
;
}
}
}
git_buf_putc
(
buf
,
'\t'
);
git_buf_puts
(
buf
,
msg
);
/* drop potential trailing LF */
git_buf_rtrim
(
buf
);
}
}
git_buf_puts
(
&
log
,
oid_old
);
git_buf_putc
(
buf
,
'\n'
);
git_buf_putc
(
&
log
,
' '
);
return
git_buf_oom
(
buf
);
}
static
int
reflog_write
(
const
char
*
log_path
,
const
git_oid
*
oid_old
,
const
git_oid
*
oid_new
,
const
git_signature
*
committer
,
const
char
*
msg
)
{
int
error
=
-
1
;
git_buf
log
=
GIT_BUF_INIT
;
git_filebuf
fbuf
=
GIT_FILEBUF_INIT
;
git_buf_puts
(
&
log
,
oid_new
);
assert
(
log_path
&&
oid_old
&&
oid_new
&&
committer
);
git_signature__writebuf
(
&
log
,
" "
,
committer
);
if
(
serialize_reflog_entry
(
&
log
,
oid_old
,
oid_new
,
committer
,
msg
)
<
0
)
git_buf_truncate
(
&
log
,
log
.
size
-
1
);
/* drop LF */
goto
cleanup
;
if
(
msg
)
{
if
((
error
=
git_filebuf_open
(
&
fbuf
,
log_path
,
GIT_FILEBUF_APPEND
))
<
0
)
git_buf_putc
(
&
log
,
'\t'
);
goto
cleanup
;
git_buf_puts
(
&
log
,
msg
);
}
if
(
!
trailing_newline
)
if
(
(
error
=
git_filebuf_write
(
&
fbuf
,
log
.
ptr
,
log
.
size
))
<
0
)
g
it_buf_putc
(
&
log
,
'\n'
)
;
g
oto
cleanup
;
if
(
git_buf_oom
(
&
log
))
{
error
=
git_filebuf_commit
(
&
fbuf
,
GIT_REFLOG_FILE_MODE
);
git_buf_free
(
&
log
);
goto
success
;
return
-
1
;
}
error
=
git_filebuf_open
(
&
fbuf
,
log_path
,
GIT_FILEBUF_APPEND
);
cleanup:
if
(
!
error
)
{
git_filebuf_cleanup
(
&
fbuf
);
if
((
error
=
git_filebuf_write
(
&
fbuf
,
log
.
ptr
,
log
.
size
))
<
0
)
git_filebuf_cleanup
(
&
fbuf
);
else
error
=
git_filebuf_commit
(
&
fbuf
,
GIT_REFLOG_FILE_MODE
);
}
success:
git_buf_free
(
&
log
);
git_buf_free
(
&
log
);
return
error
;
return
error
;
}
}
...
@@ -184,6 +202,12 @@ void git_reflog_free(git_reflog *reflog)
...
@@ -184,6 +202,12 @@ void git_reflog_free(git_reflog *reflog)
git__free
(
reflog
);
git__free
(
reflog
);
}
}
static
int
retrieve_reflog_path
(
git_buf
*
path
,
git_reference
*
ref
)
{
return
git_buf_join_n
(
path
,
'/'
,
3
,
git_reference_owner
(
ref
)
->
path_repository
,
GIT_REFLOG_DIR
,
ref
->
name
);
}
int
git_reflog_read
(
git_reflog
**
reflog
,
git_reference
*
ref
)
int
git_reflog_read
(
git_reflog
**
reflog
,
git_reference
*
ref
)
{
{
int
error
;
int
error
;
...
@@ -196,8 +220,7 @@ int git_reflog_read(git_reflog **reflog, git_reference *ref)
...
@@ -196,8 +220,7 @@ int git_reflog_read(git_reflog **reflog, git_reference *ref)
if
(
reflog_init
(
&
log
,
ref
)
<
0
)
if
(
reflog_init
(
&
log
,
ref
)
<
0
)
return
-
1
;
return
-
1
;
error
=
git_buf_join_n
(
&
log_path
,
'/'
,
3
,
error
=
retrieve_reflog_path
(
&
log_path
,
ref
);
ref
->
owner
->
path_repository
,
GIT_REFLOG_DIR
,
ref
->
name
);
if
(
!
error
)
if
(
!
error
)
error
=
git_futils_readbuffer
(
&
log_file
,
log_path
.
ptr
);
error
=
git_futils_readbuffer
(
&
log_file
,
log_path
.
ptr
);
...
@@ -216,14 +239,53 @@ int git_reflog_read(git_reflog **reflog, git_reference *ref)
...
@@ -216,14 +239,53 @@ int git_reflog_read(git_reflog **reflog, git_reference *ref)
return
error
;
return
error
;
}
}
int
git_reflog_write
(
git_reflog
*
reflog
)
{
int
error
=
-
1
;
unsigned
int
i
;
git_reflog_entry
*
entry
;
git_buf
log_path
=
GIT_BUF_INIT
;
git_buf
log
=
GIT_BUF_INIT
;
git_filebuf
fbuf
=
GIT_FILEBUF_INIT
;
assert
(
reflog
);
if
(
git_buf_join_n
(
&
log_path
,
'/'
,
3
,
git_repository_path
(
reflog
->
owner
),
GIT_REFLOG_DIR
,
reflog
->
ref_name
)
<
0
)
return
-
1
;
if
((
error
=
git_filebuf_open
(
&
fbuf
,
git_buf_cstr
(
&
log_path
),
0
))
<
0
)
goto
cleanup
;
git_vector_foreach
(
&
reflog
->
entries
,
i
,
entry
)
{
if
(
serialize_reflog_entry
(
&
log
,
&
(
entry
->
oid_old
),
&
(
entry
->
oid_cur
),
entry
->
committer
,
entry
->
msg
)
<
0
)
goto
cleanup
;
if
((
error
=
git_filebuf_write
(
&
fbuf
,
log
.
ptr
,
log
.
size
))
<
0
)
goto
cleanup
;
}
error
=
git_filebuf_commit
(
&
fbuf
,
GIT_REFLOG_FILE_MODE
);
goto
success
;
cleanup:
git_filebuf_cleanup
(
&
fbuf
);
success:
git_buf_free
(
&
log
);
git_buf_free
(
&
log_path
);
return
error
;
}
int
git_reflog_append
(
git_reference
*
ref
,
const
git_oid
*
oid_old
,
int
git_reflog_append
(
git_reference
*
ref
,
const
git_oid
*
oid_old
,
const
git_signature
*
committer
,
const
char
*
msg
)
const
git_signature
*
committer
,
const
char
*
msg
)
{
{
int
error
;
int
error
;
char
old
[
GIT_OID_HEXSZ
+
1
];
char
new
[
GIT_OID_HEXSZ
+
1
];
git_buf
log_path
=
GIT_BUF_INIT
;
git_buf
log_path
=
GIT_BUF_INIT
;
git_reference
*
r
;
git_reference
*
r
;
git_oid
zero_oid
;
const
git_oid
*
oid
;
const
git_oid
*
oid
;
if
((
error
=
git_reference_resolve
(
&
r
,
ref
))
<
0
)
if
((
error
=
git_reference_resolve
(
&
r
,
ref
))
<
0
)
...
@@ -237,12 +299,7 @@ int git_reflog_append(git_reference *ref, const git_oid *oid_old,
...
@@ -237,12 +299,7 @@ int git_reflog_append(git_reference *ref, const git_oid *oid_old,
return
-
1
;
return
-
1
;
}
}
git_oid_tostr
(
new
,
GIT_OID_HEXSZ
+
1
,
oid
);
error
=
retrieve_reflog_path
(
&
log_path
,
ref
);
git_reference_free
(
r
);
error
=
git_buf_join_n
(
&
log_path
,
'/'
,
3
,
ref
->
owner
->
path_repository
,
GIT_REFLOG_DIR
,
ref
->
name
);
if
(
error
<
0
)
if
(
error
<
0
)
goto
cleanup
;
goto
cleanup
;
...
@@ -260,14 +317,14 @@ int git_reflog_append(git_reference *ref, const git_oid *oid_old,
...
@@ -260,14 +317,14 @@ int git_reflog_append(git_reference *ref, const git_oid *oid_old,
if
(
error
<
0
)
if
(
error
<
0
)
goto
cleanup
;
goto
cleanup
;
if
(
oid_old
)
if
(
!
oid_old
)
{
git_oid_tostr
(
old
,
sizeof
(
old
),
oid_old
);
git_oid_fromstr
(
&
zero_oid
,
GIT_OID_HEX_ZERO
);
else
error
=
reflog_write
(
log_path
.
ptr
,
&
zero_oid
,
oid
,
committer
,
msg
);
memmove
(
old
,
GIT_OID_HEX_ZERO
,
sizeof
(
old
));
}
else
error
=
reflog_write
(
log_path
.
ptr
,
oid_old
,
oid
,
committer
,
msg
);
error
=
reflog_write
(
log_path
.
ptr
,
old
,
new
,
committer
,
msg
);
cleanup:
cleanup:
git_reference_free
(
r
);
git_buf_free
(
&
log_path
);
git_buf_free
(
&
log_path
);
return
error
;
return
error
;
}
}
...
@@ -281,7 +338,7 @@ int git_reflog_rename(git_reference *ref, const char *new_name)
...
@@ -281,7 +338,7 @@ int git_reflog_rename(git_reference *ref, const char *new_name)
assert
(
ref
&&
new_name
);
assert
(
ref
&&
new_name
);
if
(
git_buf_joinpath
(
&
temp_path
,
ref
->
owner
->
path_repository
,
GIT_REFLOG_DIR
)
<
0
)
if
(
git_buf_joinpath
(
&
temp_path
,
git_reference_owner
(
ref
)
->
path_repository
,
GIT_REFLOG_DIR
)
<
0
)
return
-
1
;
return
-
1
;
if
(
git_buf_joinpath
(
&
old_path
,
git_buf_cstr
(
&
temp_path
),
ref
->
name
)
<
0
)
if
(
git_buf_joinpath
(
&
old_path
,
git_buf_cstr
(
&
temp_path
),
ref
->
name
)
<
0
)
...
@@ -329,8 +386,7 @@ int git_reflog_delete(git_reference *ref)
...
@@ -329,8 +386,7 @@ int git_reflog_delete(git_reference *ref)
int
error
;
int
error
;
git_buf
path
=
GIT_BUF_INIT
;
git_buf
path
=
GIT_BUF_INIT
;
error
=
git_buf_join_n
(
error
=
retrieve_reflog_path
(
&
path
,
ref
);
&
path
,
'/'
,
3
,
ref
->
owner
->
path_repository
,
GIT_REFLOG_DIR
,
ref
->
name
);
if
(
!
error
&&
git_path_exists
(
path
.
ptr
))
if
(
!
error
&&
git_path_exists
(
path
.
ptr
))
error
=
p_unlink
(
path
.
ptr
);
error
=
p_unlink
(
path
.
ptr
);
...
...
src/reflog.h
View file @
bd72425d
...
@@ -30,6 +30,7 @@ struct git_reflog_entry {
...
@@ -30,6 +30,7 @@ struct git_reflog_entry {
struct
git_reflog
{
struct
git_reflog
{
char
*
ref_name
;
char
*
ref_name
;
git_repository
*
owner
;
git_vector
entries
;
git_vector
entries
;
};
};
...
...
tests-clar/refs/reflog/drop.c
View file @
bd72425d
...
@@ -109,3 +109,22 @@ void test_refs_reflog_drop__can_drop_all_the_entries(void)
...
@@ -109,3 +109,22 @@ void test_refs_reflog_drop__can_drop_all_the_entries(void)
cl_assert_equal_i
(
0
,
git_reflog_entrycount
(
g_reflog
));
cl_assert_equal_i
(
0
,
git_reflog_entrycount
(
g_reflog
));
}
}
void
test_refs_reflog_drop__can_persist_deletion_on_disk
(
void
)
{
git_reference
*
ref
;
cl_assert
(
entrycount
>
2
);
cl_git_pass
(
git_reference_lookup
(
&
ref
,
g_repo
,
g_reflog
->
ref_name
));
cl_git_pass
(
git_reflog_entry_drop
(
g_reflog
,
entrycount
-
1
,
1
));
cl_assert_equal_i
(
entrycount
-
1
,
git_reflog_entrycount
(
g_reflog
));
cl_git_pass
(
git_reflog_write
(
g_reflog
));
git_reflog_free
(
g_reflog
);
git_reflog_read
(
&
g_reflog
,
ref
);
git_reference_free
(
ref
);
cl_assert_equal_i
(
entrycount
-
1
,
git_reflog_entrycount
(
g_reflog
));
}
tests-clar/refs/reflog/reflog.c
View file @
bd72425d
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
static
const
char
*
new_ref
=
"refs/heads/test-reflog"
;
static
const
char
*
new_ref
=
"refs/heads/test-reflog"
;
static
const
char
*
current_master_tip
=
"a65fedf39aefe402d3bb6e24df4d4f5fe4547750"
;
static
const
char
*
current_master_tip
=
"a65fedf39aefe402d3bb6e24df4d4f5fe4547750"
;
static
const
char
*
commit_msg
=
"commit: bla bla"
;
#define commit_msg "commit: bla bla"
static
git_repository
*
g_repo
;
static
git_repository
*
g_repo
;
...
@@ -57,13 +57,13 @@ void test_refs_reflog_reflog__append_then_read(void)
...
@@ -57,13 +57,13 @@ void test_refs_reflog_reflog__append_then_read(void)
cl_git_pass
(
git_reflog_append
(
ref
,
NULL
,
committer
,
NULL
));
cl_git_pass
(
git_reflog_append
(
ref
,
NULL
,
committer
,
NULL
));
cl_git_fail
(
git_reflog_append
(
ref
,
NULL
,
committer
,
"no ancestor NULL for an existing reflog"
));
cl_git_fail
(
git_reflog_append
(
ref
,
NULL
,
committer
,
"no ancestor NULL for an existing reflog"
));
cl_git_fail
(
git_reflog_append
(
ref
,
NULL
,
committer
,
"no
\n
newline"
));
cl_git_fail
(
git_reflog_append
(
ref
,
NULL
,
committer
,
"no
inner
\n
newline"
));
cl_git_pass
(
git_reflog_append
(
ref
,
&
oid
,
committer
,
commit_msg
));
cl_git_pass
(
git_reflog_append
(
ref
,
&
oid
,
committer
,
commit_msg
"
\n
"
));
/* Reopen a new instance of the repository */
/* Reopen a new instance of the repository */
cl_git_pass
(
git_repository_open
(
&
repo2
,
"testrepo.git"
));
cl_git_pass
(
git_repository_open
(
&
repo2
,
"testrepo.git"
));
/* Lookup the pre
iv
ously created branch */
/* Lookup the pre
vi
ously created branch */
cl_git_pass
(
git_reference_lookup
(
&
lookedup_ref
,
repo2
,
new_ref
));
cl_git_pass
(
git_reference_lookup
(
&
lookedup_ref
,
repo2
,
new_ref
));
/* Read and parse the reflog for this branch */
/* Read and parse the reflog for this branch */
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment