* changes for adding new features --snapshot,
@ 2025-12-18 5:02 Abdullah
2025-12-19 10:41 ` Phillip Wood
0 siblings, 1 reply; 3+ messages in thread
From: Abdullah @ 2025-12-18 5:02 UTC (permalink / raw)
To: git
---
builtin/commit.c | 2 +
snapshot.c | 108 +++++++++++++++++++++++++++++++++++++++++++++++
snapshot.h | 13 ++++++
3 files changed, 123 insertions(+)
create mode 100644 snapshot.c
create mode 100644 snapshot.h
diff --git a/builtin/commit.c b/builtin/commit.c
index 0243f17d53..e880409be7 100644
--- a/builtin/commit.c
+++ b/builtin/commit.c
@@ -43,6 +43,7 @@
#include "commit-graph.h"
#include "pretty.h"
#include "trailer.h"
+#include "snapshot.c"
static const char * const builtin_commit_usage[] = {
N_("git commit [-a | --interactive | --patch] [-s] [-v] [-u[<mode>]]
[--amend]\n"
@@ -1545,6 +1546,7 @@ struct repository *repo UNUSED)
int fd;
struct object_id oid;
static struct option builtin_status_options[] = {
+ OPT_BOOLEAN(0, "snapshot", &opts.snapshot, "Create snapshot of staged files")
OPT__VERBOSE(&verbose, N_("be verbose")),
OPT_SET_INT('s', "short", &status_format,
N_("show status concisely"), STATUS_FORMAT_SHORT),
diff --git a/snapshot.c b/snapshot.c
new file mode 100644
index 0000000000..8c117a7314
--- /dev/null
+++ b/snapshot.c
@@ -0,0 +1,108 @@
+#include "snapshot.h"
+#include "run-command.h"
+#include "strbuf.h"
+#include "utf8.h"
+#include <time.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/stat.h>
+#include <errno.h>
+
+#define DEFAULT_SNAPSHOT_DIR ".git/snapshots"
+/*
+## Idea
+The `--snapshot` flag would create a copy of all staged files in a
timestamped folder before the commit is made. Users could configure
the snapshot folder, defaulting to `.git/snapshots`.
+
+## Example Usage
+
+# Enable snapshot feature
+git config snapshot.enable true
+
+# Optional: configure snapshot folder
+git config snapshot.folder "../snapshots"
+
+# Stage files and commit with snapshot
+git add .
+git commit -m "Initial commit" --snapshot
+
+# Example snapshot folder created:
+# ../snapshots/2025-12-17_14-30-00_Initial_commit/
+
+# Notes:
+these codes are never tested, I send it just to know if it fits to
your development
+*/
+
+// Get snapshot folder from config, or fallback to default
+static char *get_snapshot_dir(void) {
+ const char *cfg_dir = git_config_get_string("snapshot.folder");
+ if (cfg_dir && *cfg_dir)
+ return strdup(cfg_dir);
+ return strdup(DEFAULT_SNAPSHOT_DIR);
+}
+
+// Check if snapshot is enabled
+int snapshot_enabled(void) {
+ const char *val = git_config_get_string("snapshot.enable");
+ return val && strcmp(val, "true") == 0;
+}
+
+// Get current timestamp for folder name
+static void get_timestamp(char *buffer, size_t size) {
+ time_t now = time(NULL);
+ struct tm *tm_info = localtime(&now);
+ strftime(buffer, size, "%Y-%m-%d_%H-%M-%S", tm_info);
+}
+
+// Create snapshot of staged files
+int create_snapshot(const char *message) {
+ char timestamp[64];
+ char snapshot_path[1024];
+
+ char *snapshot_dir = get_snapshot_dir();
+ if (!snapshot_dir) {
+ fprintf(stderr, "Failed to get snapshot directory\n");
+ return -1;
+ }
+
+ // Ensure base folder exists
+ if (mkdir(snapshot_dir, 0755) != 0 && errno != EEXIST) {
+ perror("mkdir snapshot_dir");
+ free(snapshot_dir);
+ return -1;
+ }
+
+ get_timestamp(timestamp, sizeof(timestamp));
+
+ // Sanitize commit message
+ char sanitized_msg[512];
+ snprintf(sanitized_msg, sizeof(sanitized_msg), "%s", message);
+ for (char *p = sanitized_msg; *p; p++)
+ if (*p == ' ') *p = '_';
+
+ // Create snapshot folder path
+ snprintf(snapshot_path, sizeof(snapshot_path), "%s/%s_%s",
snapshot_dir, timestamp, sanitized_msg);
+
+ if (mkdir(snapshot_path, 0755) != 0) {
+ perror("mkdir snapshot_path");
+ free(snapshot_dir);
+ return -1;
+ }
+
+ // Copy staged files
+ char cmd[2048];
+ snprintf(cmd, sizeof(cmd),
+ "git diff --name-only --cached | xargs -I{} cp --parents
{} \"%s\"",
+ snapshot_path);
+
+ int ret = system(cmd);
+ if (ret != 0) {
+ fprintf(stderr, "Failed to copy staged files\n");
+ free(snapshot_dir);
+ return -1;
+ }
+
+ printf("Snapshot created at %s\n", snapshot_path);
+ free(snapshot_dir);
+ return 0;
+}
diff --git a/snapshot.h b/snapshot.h
new file mode 100644
index 0000000000..7542538f1f
--- /dev/null
+++ b/snapshot.h
@@ -0,0 +1,13 @@
+#ifndef SNAPSHOT_H
+#define SNAPSHOT_H
+
+#include "cache.h"
+#include "dir.h"
+
+// Check if snapshot feature is enabled
+int snapshot_enabled(void);
+
+// Create snapshot of staged files with commit message
+int create_snapshot(const char *message);
+
+#endif // SNAPSHOT_H
--
2.34.1
^ permalink raw reply related [flat|nested] 3+ messages in thread
* Re: changes for adding new features --snapshot,
2025-12-18 5:02 changes for adding new features --snapshot, Abdullah
@ 2025-12-19 10:41 ` Phillip Wood
2025-12-19 11:05 ` Chris Torek
0 siblings, 1 reply; 3+ messages in thread
From: Phillip Wood @ 2025-12-19 10:41 UTC (permalink / raw)
To: Abdullah, git
It would be helpful if there was a commit message explaining what this
new feature is and why it is needed. There are some comments in the code
explaining what it does but not why it is useful. Given that a commit is
a snapshot of the working copy I'm not sure why you'd want to save an
additional copy. As the code comments indicate you have not tested this
code I wonder how you know that what you're proposing is useful.
Thanks
Phillip
On 18/12/2025 05:02, Abdullah wrote:
> ---
> builtin/commit.c | 2 +
> snapshot.c | 108 +++++++++++++++++++++++++++++++++++++++++++++++
> snapshot.h | 13 ++++++
> 3 files changed, 123 insertions(+)
> create mode 100644 snapshot.c
> create mode 100644 snapshot.h
>
> diff --git a/builtin/commit.c b/builtin/commit.c
> index 0243f17d53..e880409be7 100644
> --- a/builtin/commit.c
> +++ b/builtin/commit.c
> @@ -43,6 +43,7 @@
> #include "commit-graph.h"
> #include "pretty.h"
> #include "trailer.h"
> +#include "snapshot.c"
>
> static const char * const builtin_commit_usage[] = {
> N_("git commit [-a | --interactive | --patch] [-s] [-v] [-u[<mode>]]
> [--amend]\n"
> @@ -1545,6 +1546,7 @@ struct repository *repo UNUSED)
> int fd;
> struct object_id oid;
> static struct option builtin_status_options[] = {
> + OPT_BOOLEAN(0, "snapshot", &opts.snapshot, "Create snapshot of staged files")
> OPT__VERBOSE(&verbose, N_("be verbose")),
> OPT_SET_INT('s', "short", &status_format,
> N_("show status concisely"), STATUS_FORMAT_SHORT),
> diff --git a/snapshot.c b/snapshot.c
> new file mode 100644
> index 0000000000..8c117a7314
> --- /dev/null
> +++ b/snapshot.c
> @@ -0,0 +1,108 @@
> +#include "snapshot.h"
> +#include "run-command.h"
> +#include "strbuf.h"
> +#include "utf8.h"
> +#include <time.h>
> +#include <stdio.h>
> +#include <stdlib.h>
> +#include <string.h>
> +#include <sys/stat.h>
> +#include <errno.h>
> +
> +#define DEFAULT_SNAPSHOT_DIR ".git/snapshots"
> +/*
> +## Idea
> +The `--snapshot` flag would create a copy of all staged files in a
> timestamped folder before the commit is made. Users could configure
> the snapshot folder, defaulting to `.git/snapshots`.
> +
> +## Example Usage
> +
> +# Enable snapshot feature
> +git config snapshot.enable true
> +
> +# Optional: configure snapshot folder
> +git config snapshot.folder "../snapshots"
> +
> +# Stage files and commit with snapshot
> +git add .
> +git commit -m "Initial commit" --snapshot
> +
> +# Example snapshot folder created:
> +# ../snapshots/2025-12-17_14-30-00_Initial_commit/
> +
> +# Notes:
> +these codes are never tested, I send it just to know if it fits to
> your development
> +*/
> +
> +// Get snapshot folder from config, or fallback to default
> +static char *get_snapshot_dir(void) {
> + const char *cfg_dir = git_config_get_string("snapshot.folder");
> + if (cfg_dir && *cfg_dir)
> + return strdup(cfg_dir);
> + return strdup(DEFAULT_SNAPSHOT_DIR);
> +}
> +
> +// Check if snapshot is enabled
> +int snapshot_enabled(void) {
> + const char *val = git_config_get_string("snapshot.enable");
> + return val && strcmp(val, "true") == 0;
> +}
> +
> +// Get current timestamp for folder name
> +static void get_timestamp(char *buffer, size_t size) {
> + time_t now = time(NULL);
> + struct tm *tm_info = localtime(&now);
> + strftime(buffer, size, "%Y-%m-%d_%H-%M-%S", tm_info);
> +}
> +
> +// Create snapshot of staged files
> +int create_snapshot(const char *message) {
> + char timestamp[64];
> + char snapshot_path[1024];
> +
> + char *snapshot_dir = get_snapshot_dir();
> + if (!snapshot_dir) {
> + fprintf(stderr, "Failed to get snapshot directory\n");
> + return -1;
> + }
> +
> + // Ensure base folder exists
> + if (mkdir(snapshot_dir, 0755) != 0 && errno != EEXIST) {
> + perror("mkdir snapshot_dir");
> + free(snapshot_dir);
> + return -1;
> + }
> +
> + get_timestamp(timestamp, sizeof(timestamp));
> +
> + // Sanitize commit message
> + char sanitized_msg[512];
> + snprintf(sanitized_msg, sizeof(sanitized_msg), "%s", message);
> + for (char *p = sanitized_msg; *p; p++)
> + if (*p == ' ') *p = '_';
> +
> + // Create snapshot folder path
> + snprintf(snapshot_path, sizeof(snapshot_path), "%s/%s_%s",
> snapshot_dir, timestamp, sanitized_msg);
> +
> + if (mkdir(snapshot_path, 0755) != 0) {
> + perror("mkdir snapshot_path");
> + free(snapshot_dir);
> + return -1;
> + }
> +
> + // Copy staged files
> + char cmd[2048];
> + snprintf(cmd, sizeof(cmd),
> + "git diff --name-only --cached | xargs -I{} cp --parents
> {} \"%s\"",
> + snapshot_path);
> +
> + int ret = system(cmd);
> + if (ret != 0) {
> + fprintf(stderr, "Failed to copy staged files\n");
> + free(snapshot_dir);
> + return -1;
> + }
> +
> + printf("Snapshot created at %s\n", snapshot_path);
> + free(snapshot_dir);
> + return 0;
> +}
> diff --git a/snapshot.h b/snapshot.h
> new file mode 100644
> index 0000000000..7542538f1f
> --- /dev/null
> +++ b/snapshot.h
> @@ -0,0 +1,13 @@
> +#ifndef SNAPSHOT_H
> +#define SNAPSHOT_H
> +
> +#include "cache.h"
> +#include "dir.h"
> +
> +// Check if snapshot feature is enabled
> +int snapshot_enabled(void);
> +
> +// Create snapshot of staged files with commit message
> +int create_snapshot(const char *message);
> +
> +#endif // SNAPSHOT_H
^ permalink raw reply [flat|nested] 3+ messages in thread
* Re: changes for adding new features --snapshot,
2025-12-19 10:41 ` Phillip Wood
@ 2025-12-19 11:05 ` Chris Torek
0 siblings, 0 replies; 3+ messages in thread
From: Chris Torek @ 2025-12-19 11:05 UTC (permalink / raw)
To: phillip.wood; +Cc: Abdullah, git
On Fri, Dec 19, 2025 at 2:41 AM Phillip Wood <phillip.wood123@gmail.com> wrote:
> It would be helpful if there was a commit message explaining what this
> new feature is and why it is needed. There are some comments in the code
> explaining what it does but not why it is useful. Given that a commit is
> a snapshot of the working copy I'm not sure why you'd want to save an
> additional copy.
To be a bit more precise, a commit is just a little bit more than
a snapshot of all staged files: it consists of a *tree object*, which
is this snapshot, plus a *commit object*, which contains metadata
to explain who made the commit, when, and why, and what
commit(s) come immediately before the new commit.
The sample code itself makes copies of some (but not all)
staged files, rather than a complete snapshot of all staged
files. Abdullah is probably under the impression that Git
saves only *changed* files, and that the staging area therefore
contains only these changed files, but that's not the case: every
snapshot contains *every* file.
This takes no extra disk space because of Git's clever method
of storing snapshots. The code in the diff does not make use of
this clever method; instead, it makes clumsy actual copies,
which generally do take up extra disk space. That's presumably
why there's a `--name-only` diff here:
>> git diff --name-only --cached | xargs -I{} cp --parents {} \"%s\"
To make a Git-style snapshot containing all files but using
no extra space, we could more simply run `git commit-tree`,
which produces the tree hash ID; we'd then save that somewhere
that Git can find it so that Git won't garbage collect the tree
later. The obvious place to save it is in a tag-like reference
(perhaps an actual tag, perhaps some new `refs/snaps/`
space or similar). But it's probably superior simply to create
an actual commit, without putting it on any branch, in the
same way that `git stash` makes commits but puts them on
no branch.
In any case, you (Phillip) are right that this doesn't explain
the use case for these extra "snapshot" commits or trees.
I rather suspect that the intended use is better-served simply
by making a branch (as often seems to me to be the cases
for which people use `git stash`...).
Chris
^ permalink raw reply [flat|nested] 3+ messages in thread
end of thread, other threads:[~2025-12-19 11:05 UTC | newest]
Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-12-18 5:02 changes for adding new features --snapshot, Abdullah
2025-12-19 10:41 ` Phillip Wood
2025-12-19 11:05 ` Chris Torek
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.