114 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			114 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C++
		
	
	
	
//===- unittests/Rewrite/RewriteBufferTest.cpp - RewriteBuffer tests ------===//
 | 
						|
//
 | 
						|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | 
						|
// See https://llvm.org/LICENSE.txt for license information.
 | 
						|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "clang/Rewrite/Core/RewriteBuffer.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
using namespace clang;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
#define EXPECT_OUTPUT(Buf, Output) EXPECT_EQ(Output, writeOutput(Buf))
 | 
						|
 | 
						|
static std::string writeOutput(const RewriteBuffer &Buf) {
 | 
						|
  std::string Result;
 | 
						|
  raw_string_ostream OS(Result);
 | 
						|
  Buf.write(OS);
 | 
						|
  OS.flush();
 | 
						|
  return Result;
 | 
						|
}
 | 
						|
 | 
						|
static void tagRange(unsigned Offset, unsigned Len, StringRef tagName,
 | 
						|
                     RewriteBuffer &Buf) {
 | 
						|
  std::string BeginTag;
 | 
						|
  raw_string_ostream(BeginTag) << '<' << tagName << '>';
 | 
						|
  std::string EndTag;
 | 
						|
  raw_string_ostream(EndTag) << "</" << tagName << '>';
 | 
						|
 | 
						|
  Buf.InsertTextAfter(Offset, BeginTag);
 | 
						|
  Buf.InsertTextBefore(Offset+Len, EndTag);
 | 
						|
}
 | 
						|
 | 
						|
TEST(RewriteBuffer, TagRanges) {
 | 
						|
  StringRef Input = "hello world";
 | 
						|
  const char *Output = "<outer><inner>hello</inner></outer> ";
 | 
						|
 | 
						|
  RewriteBuffer Buf;
 | 
						|
  Buf.Initialize(Input);
 | 
						|
  StringRef RemoveStr = "world";
 | 
						|
  size_t Pos = Input.find(RemoveStr);
 | 
						|
  Buf.RemoveText(Pos, RemoveStr.size());
 | 
						|
 | 
						|
  StringRef TagStr = "hello";
 | 
						|
  Pos = Input.find(TagStr);
 | 
						|
  tagRange(Pos, TagStr.size(), "outer", Buf);
 | 
						|
  tagRange(Pos, TagStr.size(), "inner", Buf);
 | 
						|
 | 
						|
  EXPECT_OUTPUT(Buf, Output);
 | 
						|
}
 | 
						|
 | 
						|
TEST(RewriteBuffer, DISABLED_RemoveLineIfEmpty_XFAIL) {
 | 
						|
  StringRef Input = "def\n"
 | 
						|
                    "ghi\n"
 | 
						|
                    "jkl\n";
 | 
						|
  RewriteBuffer Buf;
 | 
						|
  Buf.Initialize(Input);
 | 
						|
 | 
						|
  // Insert "abc\n" at the start.
 | 
						|
  Buf.InsertText(0, "abc\n");
 | 
						|
  EXPECT_OUTPUT(Buf, "abc\n"
 | 
						|
                     "def\n"
 | 
						|
                     "ghi\n"
 | 
						|
                     "jkl\n");
 | 
						|
 | 
						|
  // Remove "def\n".
 | 
						|
  //
 | 
						|
  // After the removal of "def", we have:
 | 
						|
  //
 | 
						|
  //   "abc\n"
 | 
						|
  //   "\n"
 | 
						|
  //   "ghi\n"
 | 
						|
  //   "jkl\n"
 | 
						|
  //
 | 
						|
  // Because removeLineIfEmpty=true, RemoveText has to remove the "\n" left on
 | 
						|
  // the line.  This happens correctly for the rewrite buffer itself, so the
 | 
						|
  // next check below passes.
 | 
						|
  //
 | 
						|
  // However, RemoveText's implementation incorrectly records the delta for
 | 
						|
  // removing the "\n" using the rewrite buffer offset, 4, where it was
 | 
						|
  // supposed to use the original input offset, 3.  Interpreted as an original
 | 
						|
  // input offset, 4 points to "g" not to "\n".  Thus, any future modifications
 | 
						|
  // at the original input's "g" will incorrectly see "g" as having become an
 | 
						|
  // empty string and so will map to the next character, "h", in the rewrite
 | 
						|
  // buffer.
 | 
						|
  StringRef RemoveStr0 = "def";
 | 
						|
  Buf.RemoveText(Input.find(RemoveStr0), RemoveStr0.size(),
 | 
						|
                 /*removeLineIfEmpty*/ true);
 | 
						|
  EXPECT_OUTPUT(Buf, "abc\n"
 | 
						|
                     "ghi\n"
 | 
						|
                     "jkl\n");
 | 
						|
 | 
						|
  // Try to remove "ghi\n".
 | 
						|
  //
 | 
						|
  // As discussed above, the original input offset for "ghi\n" incorrectly
 | 
						|
  // maps to the rewrite buffer offset for "hi\nj", so we end up with:
 | 
						|
  //
 | 
						|
  //   "abc\n"
 | 
						|
  //   "gkl\n"
 | 
						|
  //
 | 
						|
  // To show that removeLineIfEmpty=true is the culprit, change true to false
 | 
						|
  // and append a newline to RemoveStr0 above.  The test then passes.
 | 
						|
  StringRef RemoveStr1 = "ghi\n";
 | 
						|
  Buf.RemoveText(Input.find(RemoveStr1), RemoveStr1.size());
 | 
						|
  EXPECT_OUTPUT(Buf, "abc\n"
 | 
						|
                     "jkl\n");
 | 
						|
}
 | 
						|
 | 
						|
} // anonymous namespace
 |