UNPKG

tap

Version:

A Test-Anything-Protocol library for JavaScript

297 lines (240 loc) 9.56 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.grabCollectionEndComments = grabCollectionEndComments; exports.default = void 0; var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn")); var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf")); var _get2 = _interopRequireDefault(require("@babel/runtime/helpers/get")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits")); var _constants = require("../constants"); var _BlankLine = _interopRequireDefault(require("./BlankLine")); var _CollectionItem = _interopRequireDefault(require("./CollectionItem")); var _Comment = _interopRequireDefault(require("./Comment")); var _Node2 = _interopRequireDefault(require("./Node")); var _Range = _interopRequireDefault(require("./Range")); function grabCollectionEndComments(node) { var cnode = node; while (cnode instanceof _CollectionItem.default) { cnode = cnode.node; } if (!(cnode instanceof Collection)) return null; var len = cnode.items.length; var ci = -1; for (var i = len - 1; i >= 0; --i) { var n = cnode.items[i]; if (n.type === _constants.Type.COMMENT) { // Keep sufficiently indented comments with preceding node var _n$context = n.context, indent = _n$context.indent, lineStart = _n$context.lineStart; if (indent > 0 && n.range.start >= lineStart + indent) break; ci = i; } else if (n.type === _constants.Type.BLANK_LINE) ci = i;else break; } if (ci === -1) return null; var ca = cnode.items.splice(ci, len - ci); var prevEnd = ca[0].range.start; while (true) { cnode.range.end = prevEnd; if (cnode.valueRange && cnode.valueRange.end > prevEnd) cnode.valueRange.end = prevEnd; if (cnode === node) break; cnode = cnode.context.parent; } return ca; } var Collection = /*#__PURE__*/ function (_Node) { (0, _inherits2.default)(Collection, _Node); (0, _createClass2.default)(Collection, null, [{ key: "nextContentHasIndent", value: function nextContentHasIndent(src, offset, indent) { var lineStart = _Node2.default.endOfLine(src, offset) + 1; offset = _Node2.default.endOfWhiteSpace(src, lineStart); var ch = src[offset]; if (!ch) return false; if (offset >= lineStart + indent) return true; if (ch !== '#' && ch !== '\n') return false; return Collection.nextContentHasIndent(src, offset, indent); } }]); function Collection(firstItem) { var _this; (0, _classCallCheck2.default)(this, Collection); _this = (0, _possibleConstructorReturn2.default)(this, (0, _getPrototypeOf2.default)(Collection).call(this, firstItem.type === _constants.Type.SEQ_ITEM ? _constants.Type.SEQ : _constants.Type.MAP)); for (var i = firstItem.props.length - 1; i >= 0; --i) { if (firstItem.props[i].start < firstItem.context.lineStart) { // props on previous line are assumed by the collection _this.props = firstItem.props.slice(0, i + 1); firstItem.props = firstItem.props.slice(i + 1); var itemRange = firstItem.props[0] || firstItem.valueRange; firstItem.range.start = itemRange.start; break; } } _this.items = [firstItem]; var ec = grabCollectionEndComments(firstItem); if (ec) Array.prototype.push.apply(_this.items, ec); return _this; } (0, _createClass2.default)(Collection, [{ key: "parse", /** * @param {ParseContext} context * @param {number} start - Index of first character * @returns {number} - Index of the character after this */ value: function parse(context, start) { this.context = context; var parseNode = context.parseNode, src = context.src; // It's easier to recalculate lineStart here rather than tracking down the // last context from which to read it -- eemeli/yaml#2 var lineStart = _Node2.default.startOfLine(src, start); var firstItem = this.items[0]; // First-item context needs to be correct for later comment handling // -- eemeli/yaml#17 firstItem.context.parent = this; this.valueRange = _Range.default.copy(firstItem.valueRange); var indent = firstItem.range.start - firstItem.context.lineStart; var offset = start; offset = _Node2.default.normalizeOffset(src, offset); var ch = src[offset]; var atLineStart = _Node2.default.endOfWhiteSpace(src, lineStart) === offset; var prevIncludesTrailingLines = false; while (ch) { while (ch === '\n' || ch === '#') { if (atLineStart && ch === '\n' && !prevIncludesTrailingLines) { var blankLine = new _BlankLine.default(); offset = blankLine.parse({ src: src }, offset); this.valueRange.end = offset; if (offset >= src.length) { ch = null; break; } this.items.push(blankLine); offset -= 1; // blankLine.parse() consumes terminal newline } else if (ch === '#') { if (offset < lineStart + indent && !Collection.nextContentHasIndent(src, offset, indent)) { return offset; } var comment = new _Comment.default(); offset = comment.parse({ indent: indent, lineStart: lineStart, src: src }, offset); this.items.push(comment); this.valueRange.end = offset; if (offset >= src.length) { ch = null; break; } } lineStart = offset + 1; offset = _Node2.default.endOfIndent(src, lineStart); if (_Node2.default.atBlank(src, offset)) { var wsEnd = _Node2.default.endOfWhiteSpace(src, offset); var next = src[wsEnd]; if (!next || next === '\n' || next === '#') { offset = wsEnd; } } ch = src[offset]; atLineStart = true; } if (!ch) { break; } if (offset !== lineStart + indent && (atLineStart || ch !== ':')) { if (lineStart > start) offset = lineStart; break; } if (firstItem.type === _constants.Type.SEQ_ITEM !== (ch === '-')) { var typeswitch = true; if (ch === '-') { // map key may start with -, as long as it's followed by a non-whitespace char var _next = src[offset + 1]; typeswitch = !_next || _next === '\n' || _next === '\t' || _next === ' '; } if (typeswitch) { if (lineStart > start) offset = lineStart; break; } } var node = parseNode({ atLineStart: atLineStart, inCollection: true, indent: indent, lineStart: lineStart, parent: this }, offset); if (!node) return offset; // at next document start this.items.push(node); this.valueRange.end = node.valueRange.end; offset = _Node2.default.normalizeOffset(src, node.range.end); ch = src[offset]; atLineStart = false; prevIncludesTrailingLines = node.includesTrailingLines; // Need to reset lineStart and atLineStart here if preceding node's range // has advanced to check the current line's indentation level // -- eemeli/yaml#10 & eemeli/yaml#38 if (ch) { var ls = offset - 1; var prev = src[ls]; while (prev === ' ' || prev === '\t') { prev = src[--ls]; } if (prev === '\n') { lineStart = ls + 1; atLineStart = true; } } var ec = grabCollectionEndComments(node); if (ec) Array.prototype.push.apply(this.items, ec); } return offset; } }, { key: "setOrigRanges", value: function setOrigRanges(cr, offset) { offset = (0, _get2.default)((0, _getPrototypeOf2.default)(Collection.prototype), "setOrigRanges", this).call(this, cr, offset); this.items.forEach(function (node) { offset = node.setOrigRanges(cr, offset); }); return offset; } }, { key: "toString", value: function toString() { var src = this.context.src, items = this.items, range = this.range, value = this.value; if (value != null) return value; var str = src.slice(range.start, items[0].range.start) + String(items[0]); for (var i = 1; i < items.length; ++i) { var item = items[i]; var _item$context = item.context, atLineStart = _item$context.atLineStart, indent = _item$context.indent; if (atLineStart) for (var _i = 0; _i < indent; ++_i) { str += ' '; } str += String(item); } return _Node2.default.addStringTerminator(src, range.end, str); } }, { key: "includesTrailingLines", get: function get() { return this.items.length > 0; } }]); return Collection; }(_Node2.default); exports.default = Collection;