Skip to content

Commit

Permalink
WIP: git restore
Browse files Browse the repository at this point in the history
Currently restoring with --staged and --source is broken in an odd way.
  • Loading branch information
AtkinsSJ committed Jun 21, 2024
1 parent 7ad9934 commit f64f482
Show file tree
Hide file tree
Showing 2 changed files with 182 additions and 0 deletions.
2 changes: 2 additions & 0 deletions packages/git/src/subcommands/__exports__.js
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import module_log from './log.js'
import module_pull from './pull.js'
import module_push from './push.js'
import module_remote from './remote.js'
import module_restore from './restore.js'
import module_show from './show.js'
import module_status from './status.js'
import module_version from './version.js'
Expand All @@ -50,6 +51,7 @@ export default {
"pull": module_pull,
"push": module_push,
"remote": module_remote,
"restore": module_restore,
"show": module_show,
"status": module_status,
"version": module_version,
Expand Down
180 changes: 180 additions & 0 deletions packages/git/src/subcommands/restore.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,180 @@
/*
* Copyright (C) 2024 Puter Technologies Inc.
*
* This file is part of Puter's Git client.
*
* Puter's Git client is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import git, { STAGE, TREE, WORKDIR } from 'isomorphic-git';
import { find_repo_root } from '../git-helpers.js';
import path from 'path-browserify';

export default {
name: 'restore',
usage: 'git restore [--source=<tree>] [--staged] [--worktree] [--] [<pathspec>...]',
description: 'Add file contents to the index.',
args: {
allowPositionals: true,
options: {
'source': {
description: 'Source to restore from.',
type: 'string',
short: 's',
},
'staged': {
description: 'Restore the file in the index.',
type: 'boolean',
short: 'S',
},
'worktree': {
description: 'Restore the file in the working tree.',
type: 'boolean',
short: 'W',
},
'overlay': {
description: 'Enable overlay mode. In overlay mode, files that do not exist in the source are not deleted.',
type: 'boolean',
value: false,
},
'no-overlay': {
description: 'Disable overlay mode. Any files not in the source will be deleted.',
type: 'boolean',
},
},
},
execute: async (ctx) => {
const { io, fs, env, args } = ctx;
const { stdout, stderr } = io;
const { options, positionals } = args;
const cache = {};

if (!options.staged && !options.worktree)
options.worktree = true;

if (options['no-overlay'])
options.overlay = false;

const FROM_INDEX = Symbol('FROM_INDEX');

if (!options.source) {
if (options.staged) {
options.source = 'HEAD';
} else {
options.source = FROM_INDEX;
}
}
const source_ref = options.source;

const pathspecs = positionals.map(it => path.resolve(env.PWD, it));
if (pathspecs.length === 0)
throw new Error(`you must specify path(s) to restore`);

const { dir, gitdir } = await find_repo_root(fs, env.PWD);

const operations = await git.walk({
fs, dir, gitdir, cache,
trees: [
source_ref === FROM_INDEX ? STAGE() : TREE({ ref: source_ref }),
TREE({ ref: 'HEAD' }), // Only required to check if a file is tracked.
STAGE(),
WORKDIR(),
],
map: async (filepath, [ source, head, staged, workdir]) => {
// Reject paths that don't match pathspecs.
const abs_filepath = path.resolve(env.PWD, filepath);
if (!pathspecs.some(abs_path =>
(filepath === '.') || (abs_filepath.startsWith(abs_path)) || (path.dirname(abs_filepath) === abs_path),
)) {
return null;
}

if (await git.isIgnored({ fs, dir, gitdir, filepath }))
return null;

const [
source_type, staged_type, workdir_type
] = await Promise.all([
source?.type(), staged?.type(), workdir?.type()
]);

// Exclude directories from results, but still iterate them.
if ((!source_type || source_type === 'tree')
&& (!staged_type || staged_type === 'tree')
&& (!workdir_type || workdir_type === 'tree')) {
return;
}

// We need to modify the index or working tree if their oid doesn't match the source's.
const [
source_oid, staged_oid, workdir_oid
] = await Promise.all([
source_type === 'blob' ? source.oid() : undefined,
staged_type === 'blob' ? staged.oid() : undefined,
workdir_type === 'blob' ? workdir.oid() : undefined,
]);
const something_changed = (options.staged && staged_oid !== source_oid) || (options.worktree && workdir_oid !== source_oid);
if (!something_changed)
return null;

return Promise.all([
// Update the index
(async () => {
if (!options.staged || staged_oid === source_oid)
return;

// FIXME: `git restore FILE --staged --source FOO` doesn't do anything!

if (source_oid) {
// Put the file in the index
await git.updateIndex({
fs, dir, gitdir, cache,
filepath,
add: true,
oid: source_oid,
mode: await source.mode(),
});
} else if (!options.overlay) {
// Remove the file from the index
await git.remove({ fs, dir, gitdir, cache, filepath });
}
})(),
// Update the working tree
(async () => {
if (!options.worktree || workdir_oid === source_oid)
return;

// If the file isn't in source, it needs to be deleted if it is tracked by git.
// For now, I'll consider a file tracked if it exists in HEAD. This may not be correct though.
// TODO: Add an isTracked(file) method to isomorphic-git
if (!source && !head)
return null;

if (source_oid) {
// Write the file
// Unfortunately, reading the source's file data is done differently depending on if it's the index or not.
const source_content = source_ref === FROM_INDEX
? (await git.readBlob({ fs, dir, gitdir, cache, oid: source_oid })).blob
: await source.content();
await fs.promises.writeFile(abs_filepath, source_content);
} else if (!options.overlay) {
// Delete the file
await fs.promises.unlink(abs_filepath);
}
})(),
]);
},
});
await Promise.all(operations);
}
}

0 comments on commit f64f482

Please sign in to comment.