eemeli/yaml

CST.setScalarValue() does not preserve non-semantic indentation for block scalars

pjeby opened this issue · 3 comments

pjeby commented

Describe the bug
CST.setScalarValue() does not preserve non-semantic indentation for block scalars

To Reproduce
Apply setScalarValue() to change the content of a 4-space indented block scalar value in an unindented map. The indentation will be reduced to 2 spaces if afterKey is set, zero otherwise.

Expected behaviour
The original indentation should be kept.

Versions (please complete the following information):

  • Environment: node 14.16 / electron 13.5.2 / chrome 91.0.4472.164
  • yaml: 2.0.0-10

Additional context
I'm not sure if this is technically a problem with setScalarValue, or with parsing not keeping track of the original indentation relative to the key.

Yeah, this is a bug with CST.setScalarValue(). What should happen is that it looks at the previous raw content of the node, and figures out a suitable content indent from that, somewhere around here:

let indent = 'indent' in token ? token.indent : null
if (afterKey && typeof indent === 'number') indent += 2

A PR for this would be very welcome, as I'm not really sure when I'll get to this myself.

pjeby commented

Hm. Would it perhaps be better to just capture the indent at parse time, though? I mean, the issue is that the node's .indent is 0 in my use case, rather than the first-line indent of the block scalar.

Determining the scalar content indent is normally done during composition, i.e. when the actual node value is parsed from the CST contents; here. When working with CST nodes directly, that is called internally by CST.resolveAsScalar().

So the issue here is that for this use, part of that work needs to also be done in CST.setScalarValue(), and that the logic for that needs to get extracted from its current wrappings.