mirror of
https://github.com/git/git.git
synced 2024-11-30 21:44:02 +08:00
notes.c: introduce "--[no-]stripspace" option
This commit introduces a new option "--[no-]stripspace" to git notes append, git notes edit, and git notes add. This option allows users to control whether the note message need to stripped out. For the consideration of backward compatibility, let's look at the behavior about "stripspace" in "git notes" command: 1. "Edit Message" case: using the default editor to edit the note message. In "edit" case, the edited message will always be stripped out, the implementation which can be found in the "prepare_note_data()". In addition, the "-c" option supports to reuse an existing blob as a note message, then open the editor to make a further edition on it, the edited message will be stripped. This commit doesn't change the default behavior of "edit" case by using an enum "notes_stripspace", only when "--no-stripspace" option is specified, the note message will not be stripped out. If you do not specify the option or you specify "--stripspace", clearly, the note message will be stripped out. 2. "Assign Message" case: using the "-m"/"-F"/"-C" option to specify the note message. In "assign" case, when specify message by "-m" or "-F", the message will be stripped out by default, but when specify message by "-C", the message will be copied verbatim, in other word, the message will not be stripped out. One more thing need to note is "the order of the options matter", that is, if you specify "-C" before "-m" or "-F", the reused message by "-C" will be stripped out together, because everytime concat "-m" or "-F" message, the concated message will be stripped together. Oppositely, if you specify "-m" or "-F" before "-C", the reused message by "-C" will not be stripped out. This commit doesn't change the default behavior of "assign" case by extending the "stripspace" field in "struct note_msg", so we can distinguish the different behavior of "-m"/"-F" and "-C" options when we need to parse and concat the message. Signed-off-by: Teng Long <dyroneteng@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
b7d87ad537
commit
c4e2aa7d45
@ -9,10 +9,10 @@ SYNOPSIS
|
||||
--------
|
||||
[verse]
|
||||
'git notes' [list [<object>]]
|
||||
'git notes' add [-f] [--allow-empty] [--separator=<paragraph-break>] [-F <file> | -m <msg> | (-c | -C) <object>] [<object>]
|
||||
'git notes' add [-f] [--allow-empty] [--separator=<paragraph-break>] [--[no-]stripspace] [-F <file> | -m <msg> | (-c | -C) <object>] [<object>]
|
||||
'git notes' copy [-f] ( --stdin | <from-object> [<to-object>] )
|
||||
'git notes' append [--allow-empty] [--separator=<paragraph-break>] [-F <file> | -m <msg> | (-c | -C) <object>] [<object>]
|
||||
'git notes' edit [--allow-empty] [<object>]
|
||||
'git notes' append [--allow-empty] [--separator=<paragraph-break>] [--[no-]stripspace] [-F <file> | -m <msg> | (-c | -C) <object>] [<object>]
|
||||
'git notes' edit [--allow-empty] [<object>] [--[no-]stripspace]
|
||||
'git notes' show [<object>]
|
||||
'git notes' merge [-v | -q] [-s <strategy> ] <notes-ref>
|
||||
'git notes' merge --commit [-v | -q]
|
||||
@ -141,20 +141,26 @@ OPTIONS
|
||||
If multiple `-m` options are given, their values
|
||||
are concatenated as separate paragraphs.
|
||||
Lines starting with `#` and empty lines other than a
|
||||
single line between paragraphs will be stripped out.
|
||||
single line between paragraphs will be stripped out,
|
||||
if you wish to keep them verbatim, use `--no-stripspace`.
|
||||
|
||||
-F <file>::
|
||||
--file=<file>::
|
||||
Take the note message from the given file. Use '-' to
|
||||
read the note message from the standard input.
|
||||
Lines starting with `#` and empty lines other than a
|
||||
single line between paragraphs will be stripped out.
|
||||
single line between paragraphs will be stripped out,
|
||||
if you wish to keep them verbatim, use with
|
||||
`--no-stripspace` option.
|
||||
|
||||
-C <object>::
|
||||
--reuse-message=<object>::
|
||||
Take the given blob object (for example, another note) as the
|
||||
note message. (Use `git notes copy <object>` instead to
|
||||
copy notes between objects.)
|
||||
copy notes between objects.). By default, message will be
|
||||
copied verbatim, but if you wish to strip out the lines
|
||||
starting with `#` and empty lines other than a single line
|
||||
between paragraphs, use with`--stripspace` option.
|
||||
|
||||
-c <object>::
|
||||
--reedit-message=<object>::
|
||||
@ -170,6 +176,13 @@ OPTIONS
|
||||
(a newline is added at the end as needed). Defaults to a
|
||||
blank line.
|
||||
|
||||
--[no-]stripspace::
|
||||
Strip leading and trailing whitespace from the note message.
|
||||
Also strip out empty lines other than a single line between
|
||||
paragraphs. For lines starting with `#` will be stripped out
|
||||
in non-editor cases like "-m", "-F" and "-C", but not in
|
||||
editor case like "git notes edit", "-c", etc.
|
||||
|
||||
--ref <ref>::
|
||||
Manipulate the notes tree in <ref>. This overrides
|
||||
`GIT_NOTES_REF` and the "core.notesRef" configuration. The ref
|
||||
|
@ -101,14 +101,21 @@ static const char * const git_notes_get_ref_usage[] = {
|
||||
static const char note_template[] =
|
||||
N_("Write/edit the notes for the following object:");
|
||||
|
||||
enum notes_stripspace {
|
||||
UNSPECIFIED = -1,
|
||||
NO_STRIPSPACE = 0,
|
||||
STRIPSPACE = 1,
|
||||
};
|
||||
|
||||
struct note_msg {
|
||||
int stripspace;
|
||||
enum notes_stripspace stripspace;
|
||||
struct strbuf buf;
|
||||
};
|
||||
|
||||
struct note_data {
|
||||
int given;
|
||||
int use_editor;
|
||||
int stripspace;
|
||||
char *edit_path;
|
||||
struct strbuf buf;
|
||||
struct note_msg **messages;
|
||||
@ -213,7 +220,8 @@ static void prepare_note_data(const struct object_id *object, struct note_data *
|
||||
if (launch_editor(d->edit_path, &d->buf, NULL)) {
|
||||
die(_("please supply the note contents using either -m or -F option"));
|
||||
}
|
||||
strbuf_stripspace(&d->buf, 1);
|
||||
if (d->stripspace)
|
||||
strbuf_stripspace(&d->buf, 1);
|
||||
}
|
||||
}
|
||||
|
||||
@ -248,7 +256,9 @@ static void concat_messages(struct note_data *d)
|
||||
append_separator(&d->buf);
|
||||
strbuf_add(&msg, d->messages[i]->buf.buf, d->messages[i]->buf.len);
|
||||
strbuf_addbuf(&d->buf, &msg);
|
||||
if (d->messages[i]->stripspace)
|
||||
if ((d->stripspace == UNSPECIFIED &&
|
||||
d->messages[i]->stripspace == STRIPSPACE) ||
|
||||
d->stripspace == STRIPSPACE)
|
||||
strbuf_stripspace(&d->buf, 0);
|
||||
strbuf_reset(&msg);
|
||||
}
|
||||
@ -266,7 +276,7 @@ static int parse_msg_arg(const struct option *opt, const char *arg, int unset)
|
||||
strbuf_addstr(&msg->buf, arg);
|
||||
ALLOC_GROW_BY(d->messages, d->msg_nr, 1, d->msg_alloc);
|
||||
d->messages[d->msg_nr - 1] = msg;
|
||||
msg->stripspace = 1;
|
||||
msg->stripspace = STRIPSPACE;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -286,7 +296,7 @@ static int parse_file_arg(const struct option *opt, const char *arg, int unset)
|
||||
|
||||
ALLOC_GROW_BY(d->messages, d->msg_nr, 1, d->msg_alloc);
|
||||
d->messages[d->msg_nr - 1] = msg;
|
||||
msg->stripspace = 1;
|
||||
msg->stripspace = STRIPSPACE;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -319,7 +329,7 @@ static int parse_reuse_arg(const struct option *opt, const char *arg, int unset)
|
||||
msg->buf.len = len;
|
||||
ALLOC_GROW_BY(d->messages, d->msg_nr, 1, d->msg_alloc);
|
||||
d->messages[d->msg_nr - 1] = msg;
|
||||
msg->stripspace = 0;
|
||||
msg->stripspace = NO_STRIPSPACE;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -453,7 +463,7 @@ static int add(int argc, const char **argv, const char *prefix)
|
||||
struct notes_tree *t;
|
||||
struct object_id object, new_note;
|
||||
const struct object_id *note;
|
||||
struct note_data d = { .buf = STRBUF_INIT };
|
||||
struct note_data d = { .buf = STRBUF_INIT, .stripspace = UNSPECIFIED };
|
||||
|
||||
struct option options[] = {
|
||||
OPT_CALLBACK_F('m', "message", &d, N_("message"),
|
||||
@ -473,6 +483,8 @@ static int add(int argc, const char **argv, const char *prefix)
|
||||
OPT__FORCE(&force, N_("replace existing notes"), PARSE_OPT_NOCOMPLETE),
|
||||
OPT_STRING(0, "separator", &separator, N_("separator"),
|
||||
N_("insert <paragraph-break> between paragraphs")),
|
||||
OPT_BOOL(0, "stripspace", &d.stripspace,
|
||||
N_("remove unnecessary whitespace")),
|
||||
OPT_END()
|
||||
};
|
||||
|
||||
@ -626,7 +638,7 @@ static int append_edit(int argc, const char **argv, const char *prefix)
|
||||
const struct object_id *note;
|
||||
char *logmsg;
|
||||
const char * const *usage;
|
||||
struct note_data d = { .buf = STRBUF_INIT };
|
||||
struct note_data d = { .buf = STRBUF_INIT, .stripspace = UNSPECIFIED };
|
||||
struct option options[] = {
|
||||
OPT_CALLBACK_F('m', "message", &d, N_("message"),
|
||||
N_("note contents as a string"), PARSE_OPT_NONEG,
|
||||
@ -644,6 +656,8 @@ static int append_edit(int argc, const char **argv, const char *prefix)
|
||||
N_("allow storing empty note")),
|
||||
OPT_STRING(0, "separator", &separator, N_("separator"),
|
||||
N_("insert <paragraph-break> between paragraphs")),
|
||||
OPT_BOOL(0, "stripspace", &d.stripspace,
|
||||
N_("remove unnecessary whitespace")),
|
||||
OPT_END()
|
||||
};
|
||||
int edit = !strcmp(argv[0], "edit");
|
||||
|
@ -32,7 +32,7 @@ test_expect_success 'add note by editor' '
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying single "-m"' '
|
||||
test_expect_success 'add note by specifying single "-m", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
@ -42,10 +42,26 @@ test_expect_success 'add note by specifying single "-m"' '
|
||||
|
||||
git notes add -m "${LF}first-line${MULTI_LF}second-line${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
git notes add --stripspace -m "${LF}first-line${MULTI_LF}second-line${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying multiple "-m"' '
|
||||
test_expect_success 'add note by specifying single "-m" and "--no-stripspace" ' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}first-line${MULTI_LF}second-line
|
||||
EOF
|
||||
|
||||
git notes add --no-stripspace \
|
||||
-m "${LF}first-line${MULTI_LF}second-line${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying multiple "-m", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
@ -59,9 +75,156 @@ test_expect_success 'add note by specifying multiple "-m"' '
|
||||
-m "second-line" \
|
||||
-m "${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
git notes add --stripspace -m "${LF}" \
|
||||
-m "first-line" \
|
||||
-m "${MULTI_LF}" \
|
||||
-m "second-line" \
|
||||
-m "${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add notes by specifying multiple "-m" and "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}
|
||||
first-line
|
||||
${MULTI_LF}
|
||||
second-line${LF}
|
||||
EOF
|
||||
|
||||
git notes add --no-stripspace \
|
||||
-m "${LF}" \
|
||||
-m "first-line" \
|
||||
-m "${MULTI_LF}" \
|
||||
-m "second-line" \
|
||||
-m "${LF}" &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying single "-F", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
|
||||
second-line
|
||||
EOF
|
||||
|
||||
cat >note-file <<-EOF &&
|
||||
${LF}
|
||||
first-line
|
||||
${MULTI_LF}
|
||||
second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
git notes add -F note-file &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
git notes add --stripspace -F note-file &&
|
||||
git notes show >actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying single "-F" and "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}
|
||||
first-line
|
||||
${MULTI_LF}
|
||||
second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file <<-EOF &&
|
||||
${LF}
|
||||
first-line
|
||||
${MULTI_LF}
|
||||
second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
git notes add --no-stripspace -F note-file &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying multiple "-F", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
file-1-first-line
|
||||
|
||||
file-1-second-line
|
||||
|
||||
file-2-first-line
|
||||
|
||||
file-2-second-line
|
||||
EOF
|
||||
|
||||
cat >note-file-1 <<-EOF &&
|
||||
${LF}
|
||||
file-1-first-line
|
||||
${MULTI_LF}
|
||||
file-1-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file-2 <<-EOF &&
|
||||
${LF}
|
||||
file-2-first-line
|
||||
${MULTI_LF}
|
||||
file-2-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
git notes add -F note-file-1 -F note-file-2 &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
git notes add --stripspace -F note-file-1 -F note-file-2 &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying multiple "-F" with "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}
|
||||
file-1-first-line
|
||||
${MULTI_LF}
|
||||
file-1-second-line
|
||||
${LF}
|
||||
|
||||
${LF}
|
||||
file-2-first-line
|
||||
${MULTI_LF}
|
||||
file-2-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file-1 <<-EOF &&
|
||||
${LF}
|
||||
file-1-first-line
|
||||
${MULTI_LF}
|
||||
file-1-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file-2 <<-EOF &&
|
||||
${LF}
|
||||
file-2-first-line
|
||||
${MULTI_LF}
|
||||
file-2-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
git notes add --no-stripspace -F note-file-1 -F note-file-2 &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'append note by editor' '
|
||||
test_when_finished "git notes remove" &&
|
||||
@ -221,6 +384,45 @@ test_expect_success 'append notes by specifying multiple "-F"' '
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'append note by specifying multiple "-F" with "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
initial-line
|
||||
${LF}${LF}
|
||||
file-1-first-line
|
||||
${MULTI_LF}
|
||||
file-1-second-line
|
||||
${LF}
|
||||
|
||||
${LF}
|
||||
file-2-first-line
|
||||
${MULTI_LF}
|
||||
file-2-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file-1 <<-EOF &&
|
||||
${LF}
|
||||
file-1-first-line
|
||||
${MULTI_LF}
|
||||
file-1-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >note-file-2 <<-EOF &&
|
||||
${LF}
|
||||
file-2-first-line
|
||||
${MULTI_LF}
|
||||
file-2-second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
git notes add -m "initial-line" &&
|
||||
git notes append --no-stripspace -F note-file-1 -F note-file-2 &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add notes with empty messages' '
|
||||
rev=$(git rev-parse HEAD) &&
|
||||
git notes add -m "${LF}" \
|
||||
@ -229,7 +431,7 @@ test_expect_success 'add notes with empty messages' '
|
||||
test_i18ngrep "Removing note for object" actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying "-C" , do not stripspace is the default behavior' '
|
||||
test_expect_success 'add note by specifying "-C", "--no-stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}
|
||||
@ -242,10 +444,36 @@ test_expect_success 'add note by specifying "-C" , do not stripspace is the defa
|
||||
cat expect | git hash-object -w --stdin >blob &&
|
||||
git notes add -C $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
git notes add --no-stripspace -C $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add notes with "-C" and "-m", "-m" will stripspace all together' '
|
||||
test_expect_success 'reuse note by specifying "-C" and "--stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >data <<-EOF &&
|
||||
${LF}
|
||||
first-line
|
||||
${MULTI_LF}
|
||||
second-line
|
||||
${LF}
|
||||
EOF
|
||||
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
|
||||
second-line
|
||||
EOF
|
||||
|
||||
cat data | git hash-object -w --stdin >blob &&
|
||||
git notes add --stripspace -C $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'reuse with "-C" and add note with "-m", "-m" will stripspace all together' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >data <<-EOF &&
|
||||
${LF}
|
||||
@ -269,7 +497,7 @@ test_expect_success 'add notes with "-C" and "-m", "-m" will stripspace all toge
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add notes with "-m" and "-C", "-C" will not stripspace all together' '
|
||||
test_expect_success 'add note with "-m" and reuse note with "-C", "-C" will not stripspace all together' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >data <<-EOF &&
|
||||
|
||||
@ -288,4 +516,62 @@ test_expect_success 'add notes with "-m" and "-C", "-C" will not stripspace all
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying "-c", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
|
||||
second-line
|
||||
EOF
|
||||
|
||||
echo "initial-line" | git hash-object -w --stdin >blob &&
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes add -c $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes add --stripspace -c $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'add note by specifying "-c" with "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}first-line${MULTI_LF}second-line${LF}
|
||||
EOF
|
||||
|
||||
echo "initial-line" | git hash-object -w --stdin >blob &&
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes add --no-stripspace -c $(cat blob) &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'edit note by specifying "-c", "--stripspace" is the default behavior' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
first-line
|
||||
|
||||
second-line
|
||||
EOF
|
||||
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes edit &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual &&
|
||||
git notes remove &&
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes edit --stripspace &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_expect_success 'edit note by specifying "-c" with "--no-stripspace"' '
|
||||
test_when_finished "git notes remove" &&
|
||||
cat >expect <<-EOF &&
|
||||
${LF}first-line${MULTI_LF}second-line${LF}
|
||||
EOF
|
||||
|
||||
MSG="${LF}first-line${MULTI_LF}second-line${LF}" git notes add --no-stripspace &&
|
||||
git notes show >actual &&
|
||||
test_cmp expect actual
|
||||
'
|
||||
|
||||
test_done
|
||||
|
Loading…
Reference in New Issue
Block a user