Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

Support truncating from the left, width padding/truncation incorrect with some unicode input #283

Open
moh-eulith opened this issue Oct 13, 2022 · 12 comments · May be fixed by #285
Open

Support truncating from the left, width padding/truncation incorrect with some unicode input #283

moh-eulith opened this issue Oct 13, 2022 · 12 comments · May be fixed by #285
Labels

Comments

@moh-eulith
Copy link

Left truncation is the default in the inspiration for this crate (Log4j). It usually produces more relevant results, e.g. for filenames, logger names, etc.

Both styles of truncation should be supported, with right truncation (current implementation) as the default for backward compatibility.

Log4j uses a minus sign on the length (negative length) to flip the truncation. So it might look like this for log4rs: {f:>-15.15} (left truncate or left pad the filename field to exactly 15 chars)

@estk
Copy link
Owner

estk commented Oct 13, 2022

Sounds like a worthy enhancement, happy to mentor if you'd like to take a stab at it.

@moh-eulith
Copy link
Author

I had a look at this. unicode encoding makes this diabolically difficult. The existing code is already running into this issue, so maybe fixing the unicode issues is the first step. Here is a failing test case inspired by this stackoverflow question

    #[test]
    #[cfg(feature = "simple_writer")]
    fn right_align_formatter_hard_unicode() {
        let pw = PatternEncoder::new("{({l} {m}):>15}");

        let mut buf = vec![];
        pw.encode(
            &mut SimpleWriter(&mut buf),
            &Record::builder()
                .level(Level::Info)
                .args(format_args!("\u{01f5}\u{0067}\u{0301}"))
                .build(),
        )
        .unwrap();
        assert_eq!(buf.len(), 18);
    }

a simple println!("\u{01f5}\u{0067}\u{0301}"); produces ǵǵ (two visible characters, not sure the pasting is working right), but the padding thinks it's 3 characters.

@estk
Copy link
Owner

estk commented Oct 17, 2022

@moh-eulith can you open a Draft pr with the above failing test?

@moh-eulith moh-eulith linked a pull request Oct 17, 2022 that will close this issue
@estk estk changed the title Support truncating from the left Support truncating from the left, width padding/truncation incorrect with some unicode input Oct 17, 2022
@estk estk added the bug label Oct 17, 2022
@estk
Copy link
Owner

estk commented Oct 17, 2022

Ok so I've kinda gotten to the bottom of this. There are in fact 3 unicode "characters" ie codepoints here but they result in only 2 graphemes and therefore the message has a unicode width of 2. I have a fix for the test thanks to the unicode_segmentation crate. But I think I need to think on this for a minute to decide how to proceed.

What do you think?

fn char_starts(buf: &[u8]) -> usize {
    if let Ok(s) = std::str::from_utf8(buf) {
        s.graphemes(true).count()
    } else {
        buf.iter().filter(|&&b| is_char_boundary(b)).count()
    }
}
....
    #[test]
    fn char_starts_gs() {
        let buf = "\u{01f5}\u{0067}\u{0301}";
        let starts = char_starts(buf.as_bytes());
        assert_eq!(2, starts);
    }

@moh-eulith
Copy link
Author

I'm not clear on what the buf contents are that are passed to the write function. Is the content guaranteed to be on byte, unicode code point, or grapheme boundary?

@estk
Copy link
Owner

estk commented Oct 17, 2022

Negative, an io::Writer just takes bytes and writes them, there is no guarantee of a particular segmentation, which is why this is really only a best-effort check.

@moh-eulith
Copy link
Author

So a full fix would require some manner of buffering. Max encoding length for utf8 is 4 bytes, another 2 for combining, so that's a 6 byte buffer. I'm by no means a unicode expert, but it appears combining is not limited to a single one per grapheme, so theoretically, we have to look ahead past all combining codepoints.

Additionally, there are "Codepoints U+035C–0362 are double diacritics, diacritic signs placed across two letters."

A test case for this insanity might look like: println!("a\u{0308}\u{032C}\u{0362}e\u{0301}\u{0331}") which prints: ͏ä̬͢é̱

Honestly, I don't know how far this should go. I think enough buffering for 1 codepoint + 1 combining is practical enough for real text.

@estk
Copy link
Owner

estk commented Oct 17, 2022

Counting graphemes seems to be enough.

    #[test]
    fn char_starts_gs() {
        let buf = "\u{01f5}\u{0067}\u{0301}";
        let starts = char_starts(buf.as_bytes());
        assert_eq!(2, starts);
        let buf = "a\u{0308}\u{032C}\u{0362}e\u{0301}\u{0331}";
        let starts = char_starts(buf.as_bytes());
        assert_eq!(2, starts);
    }

@moh-eulith
Copy link
Author

Agreed that grapheme counting is the right way to go. However, because we don't have a guarantee on the buf segmentation, it leads to failures:

    #[test]
    fn char_starts_gs() {
        let buf = "\u{01f5}\u{0067}\u{0301}";
        test_all_splits(buf, 2);
        let buf = "a\u{0308}\u{032C}\u{0362}e\u{0301}\u{0331}";
        test_all_splits(buf, 2);
    }

    fn test_all_splits(buf: &str, len: usize) {
        let b = buf.as_bytes();
        for i in 1..b.len() {
            assert_eq!(len, char_starts(&b[..i]) + char_starts(&b[i..]));
        }
    }

@moh-eulith
Copy link
Author

Sorry, should've made the example simpler. I think the minimal test case that fails test_all_splits is:

        let buf = "e\u{0331}";
        test_all_splits(buf, 1);

@estk
Copy link
Owner

estk commented Oct 17, 2022

No worries, yep I agree there will be miscalculations when we span a Unicode sequence, we just don't have enough information to do the right thing.

@estk estk added open pr An open PR exists and removed help wanted labels Oct 30, 2022
@DavidZemon
Copy link

Is it rude to just ignore unicode and put a note in the docs about "unicode support is undefined and may result in unexpected trimmings" 😅

@bconn98 bconn98 linked a pull request Apr 4, 2024 that will close this issue
# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants