mz_ore/codegen.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
// Copyright Materialize, Inc. and contributors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License in the LICENSE file at the
// root of this repository, or online at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//! Code generation utilities.
/// A code generation buffer.
///
/// A `CodegenBuf` provides a string-based API for generating Rust code. Its
/// value is in the various function it provides to automatically manage
/// indentation.
#[derive(Clone, Debug, Default)]
pub struct CodegenBuf {
inner: String,
level: usize,
}
impl CodegenBuf {
/// Creates a new code generation buffer.
pub fn new() -> CodegenBuf {
CodegenBuf::default()
}
/// Consumes the buffer, returning its contents.
pub fn into_string(self) -> String {
self.inner
}
/// Writes a string into the buffer directly.
pub fn write<S>(&mut self, s: S)
where
S: AsRef<str>,
{
self.inner.push_str(s.as_ref());
}
/// Writes a line into the buffer at the current indentation level.
///
/// Specifically, the method writes (4 * indentation level) spaces into the
/// buffer, followed by `s`, followed by a newline character.
pub fn writeln<S>(&mut self, s: S)
where
S: AsRef<str>,
{
self.start_line();
self.write(s);
self.end_line();
}
/// Starts a new line.
///
/// Specifically, the method writes (4 * indentation level) spaces into
/// the buffer.
pub fn start_line(&mut self) {
for _ in 0..self.level {
self.write(" ");
}
}
/// Ends the current line.
///
/// Specifically, the method writes a newline character into the buffer.
pub fn end_line(&mut self) {
self.write("\n");
}
/// Writes a new indented block.
///
/// Specifically, if `s` is empty, the method writes the line `{` into the
/// buffer; otherwise writes the line `s {` into the buffer at the current
/// indentation level. Then it increments the buffer's indentation level,
/// runs the provided function, then decrements the indentation level and writes
/// a closing `}`.
pub fn write_block<S, F>(&mut self, s: S, f: F)
where
S: AsRef<str>,
F: FnOnce(&mut Self),
{
self.start_line();
self.write(s.as_ref());
if !s.as_ref().is_empty() {
self.inner.push(' ');
}
self.write("{\n");
self.level += 1;
f(self);
self.level -= 1;
self.writeln("}");
}
/// Closes the current indented block and starts a new one at the same
/// indentation level.
///
/// Specifically, the method writes the line `} s {` into the buffer at one
/// less than the buffer's indentation level.
///
/// # Panics
///
/// Panics if the current indentation level is zero.
pub fn restart_block<S>(&mut self, s: S)
where
S: AsRef<str>,
{
self.level -= 1;
self.start_line();
self.write("} ");
self.write(s.as_ref());
self.write(" {\n");
self.level += 1;
}
}