2023-03-01 09:18:00 +00:00
|
|
|
# Templates
|
|
|
|
|
|
|
|
Jujutsu supports a functional language to customize output of commands.
|
|
|
|
The language consists of literals, keywords, operators, functions, and
|
|
|
|
methods.
|
|
|
|
|
|
|
|
A couple of `jj` commands accept a template via `-T`/`--template` option.
|
|
|
|
|
|
|
|
## Keywords
|
|
|
|
|
|
|
|
Keywords represent objects of different types; the types are described in
|
|
|
|
a follow-up section.
|
|
|
|
|
|
|
|
### Commit keywords
|
|
|
|
|
|
|
|
The following keywords can be used in `jj log`/`jj obslog` templates.
|
|
|
|
|
|
|
|
* `description: String`
|
|
|
|
* `change_id: ChangeId`
|
|
|
|
* `commit_id: CommitId`
|
2023-03-19 12:13:47 +00:00
|
|
|
* `parents: List<Commit>`
|
2023-03-01 09:18:00 +00:00
|
|
|
* `author: Signature`
|
|
|
|
* `committer: Signature`
|
|
|
|
* `working_copies: String`: For multi-workspace repository, indicate
|
|
|
|
working-copy commit as `<workspace name>@`.
|
|
|
|
* `current_working_copy: Boolean`: True for the working-copy commit of the
|
|
|
|
current workspace.
|
2023-10-25 07:31:18 +00:00
|
|
|
* `branches: List<RefName>`
|
|
|
|
* `tags: List<RefName>`
|
|
|
|
* `git_refs: List<RefName>`
|
|
|
|
* `git_head: List<RefName>`
|
2023-04-09 20:06:59 +00:00
|
|
|
* `divergent: Boolean`: True if the commit's change id corresponds to multiple
|
|
|
|
visible commits.
|
2023-04-08 01:28:16 +00:00
|
|
|
* `hidden: Boolean`: True if the commit is not visible (a.k.a. abandoned).
|
2023-03-01 09:18:00 +00:00
|
|
|
* `conflict: Boolean`: True if the commit contains merge conflicts.
|
|
|
|
* `empty: Boolean`: True if the commit modifies no files.
|
2023-08-15 13:18:56 +00:00
|
|
|
* `root: Boolean`: True if the commit is the root commit.
|
2023-03-01 09:18:00 +00:00
|
|
|
|
|
|
|
### Operation keywords
|
|
|
|
|
|
|
|
The following keywords can be used in `jj op log` templates.
|
|
|
|
|
|
|
|
* `current_operation: Boolean`
|
|
|
|
* `description: String`
|
|
|
|
* `id: OperationId`
|
|
|
|
* `tags: String`
|
|
|
|
* `time: TimestampRange`
|
|
|
|
* `user: String`
|
|
|
|
|
|
|
|
## Operators
|
|
|
|
|
|
|
|
The following operators are supported.
|
|
|
|
|
|
|
|
* `x.f()`: Method call.
|
|
|
|
* `x ++ y`: Concatenate `x` and `y` templates.
|
|
|
|
|
|
|
|
## Global functions
|
|
|
|
|
|
|
|
The following functions are defined.
|
|
|
|
|
templater: add fill(width, content) function
The parameter order follows indent()/label() functions, but this might be
a bad idea because fill() is more likely to have optional parameters. We can
instead add template.fill(width) method as well as .indent(prefix). If we take
this approach, we'll probably need to add string.fill()/indent() methods,
and/or implicit cast at method resolution. The good thing about the method
syntax is that we can add string.refill(), etc. for free, without inventing
generic labeled template functions.
For #1043, I think it's better to add a config like ui.log-word-wrap = true.
We could add term_width/graph_width keywords to the templater, but the
implementation would be more complicated, and is difficult to use for the
basic use case. Unlike Mercurial, our templater doesn't have a context map
to override the graph_width stub.
2023-03-04 14:07:14 +00:00
|
|
|
* `fill(width: Integer, content: Template) -> Template`: Fill lines at
|
|
|
|
the given `width`.
|
2023-03-02 08:16:22 +00:00
|
|
|
* `indent(prefix: Template, content: Template) -> Template`: Indent
|
|
|
|
non-empty lines by the given `prefix`.
|
2023-03-01 09:18:00 +00:00
|
|
|
* `label(label: Template, content: Template) -> Template`: Apply label to
|
|
|
|
the content. The `label` is evaluated as a space-separated string.
|
|
|
|
* `if(condition: Boolean, then: Template[, else: Template]) -> Template`:
|
|
|
|
Conditionally evaluate `then`/`else` template content.
|
|
|
|
* `concat(content: Template...) -> Template`:
|
|
|
|
Same as `content_1 ++ ... ++ content_n`.
|
|
|
|
* `separate(separator: Template, content: Template...) -> Template`:
|
|
|
|
Insert separator between **non-empty** contents.
|
|
|
|
|
|
|
|
## Types
|
|
|
|
|
|
|
|
### Boolean type
|
|
|
|
|
2023-09-02 08:49:21 +00:00
|
|
|
No methods are defined. Can be constructed with `false` or `true` literal.
|
2023-03-01 09:18:00 +00:00
|
|
|
|
2023-03-19 12:13:47 +00:00
|
|
|
### Commit type
|
|
|
|
|
|
|
|
This type cannot be printed. All commit keywords are accessible as 0-argument
|
|
|
|
methods.
|
|
|
|
|
2023-03-01 09:18:00 +00:00
|
|
|
### CommitId / ChangeId type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.short([len: Integer]) -> String`
|
|
|
|
* `.shortest([min_len: Integer]) -> ShortestIdPrefix`: Shortest unique prefix.
|
|
|
|
|
|
|
|
### Integer type
|
|
|
|
|
|
|
|
No methods are defined.
|
|
|
|
|
2023-03-06 12:02:57 +00:00
|
|
|
### List type
|
|
|
|
|
2023-10-27 00:50:32 +00:00
|
|
|
A list can be implicitly converted to `Boolean`. The following methods are
|
|
|
|
defined.
|
2023-03-06 13:02:39 +00:00
|
|
|
|
|
|
|
* `.join(separator: Template) -> Template`: Concatenate elements with
|
|
|
|
the given `separator`.
|
2023-03-14 10:57:03 +00:00
|
|
|
* `.map(|item| expression) -> ListTemplate`: Apply template `expression`
|
2023-03-19 12:13:47 +00:00
|
|
|
to each element. Example: `parents.map(|c| c.commit_id().short())`
|
2023-03-06 12:02:57 +00:00
|
|
|
|
2023-03-14 10:57:03 +00:00
|
|
|
### ListTemplate type
|
|
|
|
|
|
|
|
The following methods are defined. See also the `List` type.
|
|
|
|
|
|
|
|
* `.join(separator: Template) -> Template`
|
|
|
|
|
2023-03-01 09:18:00 +00:00
|
|
|
### OperationId type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.short([len: Integer]) -> String`
|
|
|
|
|
2023-10-25 06:36:22 +00:00
|
|
|
### RefName type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.name() -> String`: Local branch or tag name.
|
|
|
|
* `.remote() -> String`: Remote name or empty if this is a local ref.
|
|
|
|
|
2023-03-01 09:18:00 +00:00
|
|
|
### ShortestIdPrefix type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.prefix() -> String`
|
|
|
|
* `.rest() -> String`
|
|
|
|
* `.upper() -> ShortestIdPrefix`
|
|
|
|
* `.lower() -> ShortestIdPrefix`
|
|
|
|
|
|
|
|
### Signature type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.name() -> String`
|
|
|
|
* `.email() -> String`
|
|
|
|
* `.username() -> String`
|
|
|
|
* `.timestamp() -> Timestamp`
|
|
|
|
|
|
|
|
### String type
|
|
|
|
|
|
|
|
A string can be implicitly converted to `Boolean`. The following methods are
|
|
|
|
defined.
|
|
|
|
|
|
|
|
* `.contains(needle: Template) -> Boolean`
|
|
|
|
* `.first_line() -> String`
|
2023-03-07 09:19:50 +00:00
|
|
|
* `.lines() -> List<String>`: Split into lines excluding newline characters.
|
2023-03-01 09:18:00 +00:00
|
|
|
* `.upper() -> String`
|
|
|
|
* `.lower() -> String`
|
2023-08-22 18:40:51 +00:00
|
|
|
* `.starts_with(needle: Template) -> Boolean`
|
|
|
|
* `.ends_with(needle: Template) -> Boolean`
|
|
|
|
* `.remove_prefix(needle: Template) -> String`: Removes the passed prefix, if present
|
|
|
|
* `.remove_suffix(needle: Template) -> String`: Removes the passed suffix, if present
|
|
|
|
* `.substr(start: Integer, end: Integer) -> String`: Extract substring. Negative values count from the end.
|
2023-03-01 09:18:00 +00:00
|
|
|
|
2023-09-14 23:43:47 +00:00
|
|
|
#### String literals
|
|
|
|
|
|
|
|
String literals must be surrounded by double quotes (`"`). The following escape
|
|
|
|
sequences starting with a backslash have their usual meaning: `\"`, `\\`, `\n`,
|
|
|
|
`\r`, `\t`, `\0`. Other escape sequences are not supported. Any UTF-8 characters
|
|
|
|
are allowed inside a string literal, with two exceptions: unescaped `"`-s and
|
|
|
|
uses of `\` that don't form a valid escape sequence.
|
|
|
|
|
2023-03-01 09:18:00 +00:00
|
|
|
### Template type
|
|
|
|
|
2023-03-19 11:43:51 +00:00
|
|
|
Most types can be implicitly converted to `Template`. No methods are defined.
|
2023-03-01 09:18:00 +00:00
|
|
|
|
|
|
|
### Timestamp type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.ago() -> String`: Format as relative timestamp.
|
2023-03-14 12:40:36 +00:00
|
|
|
* `.format(format: String) -> String`: Format with [the specified strftime-like
|
|
|
|
format string](https://docs.rs/chrono/latest/chrono/format/strftime/).
|
2023-08-11 20:31:47 +00:00
|
|
|
* `.utc() -> Timestamp`: Convert timestamp into UTC timezone.
|
2023-03-01 09:18:00 +00:00
|
|
|
|
|
|
|
### TimestampRange type
|
|
|
|
|
|
|
|
The following methods are defined.
|
|
|
|
|
|
|
|
* `.start() -> Timestamp`
|
|
|
|
* `.end() -> Timestamp`
|
|
|
|
* `.duration() -> String`
|
|
|
|
|
|
|
|
## Configuration
|
|
|
|
|
2023-08-13 01:38:57 +00:00
|
|
|
The default templates and aliases() are defined in the `[templates]` and
|
|
|
|
`[template-aliases]` sections of the config respectively. The exact definitions
|
|
|
|
can be seen in the `cli/src/config/templates.toml` file in jj's source tree.
|
|
|
|
|
|
|
|
<!--- TODO: Find a way to embed the default config files in the docs -->
|
2023-03-01 09:18:00 +00:00
|
|
|
|
|
|
|
New keywords and functions can be defined as aliases, by using any
|
|
|
|
combination of the predefined keywords/functions and other aliases.
|
|
|
|
|
|
|
|
For example:
|
|
|
|
|
|
|
|
```toml
|
|
|
|
[template-aliases]
|
|
|
|
'commit_change_ids' = '''
|
|
|
|
concat(
|
|
|
|
format_field("Commit ID", commit_id),
|
|
|
|
format_field("Change ID", commit_id),
|
|
|
|
)
|
|
|
|
'''
|
|
|
|
'format_field(key, value)' = 'key ++ ": " ++ value ++ "\n"'
|
|
|
|
```
|