GHSA-649x-hxfx-57j2

Suggest an improvement
Source
https://github.com/advisories/GHSA-649x-hxfx-57j2
Import Source
https://github.com/github/advisory-database/blob/main/advisories/github-reviewed/2024/05/GHSA-649x-hxfx-57j2/GHSA-649x-hxfx-57j2.json
JSON Data
https://api.osv.dev/v1/vulns/GHSA-649x-hxfx-57j2
Aliases
Published
2024-05-08T14:32:32Z
Modified
2024-05-10T21:33:14Z
Severity
  • 4.9 (Medium) CVSS_V3 - CVSS:3.1/AV:N/AC:L/PR:H/UI:N/S:U/C:N/I:N/A:H CVSS Calculator
Summary
Vitess vulnerable to infinite memory consumption and vtgate crash
Details

Summary

When executing the following simple query, the vtgate will go into an endless loop that also keeps consuming memory and eventually will OOM.

Details

When running the following query, the evalengine will try evaluate it and runs forever.

select _utf16 0xFF

The source of the bug lies in the collation logic that we have. The bug applies to all utf16, utf32 and ucs2 encodings. In general, the bug is there for any encoding where the minimal byte length for a single character is more than 1 byte.

The decoding functions for these collations all implement logic like the following to enforce the minimal character length:

https://github.com/vitessio/vitess/blob/8f6cfaaa643a08dc111395a75a2d250ee746cfa8/go/mysql/collations/charset/unicode/utf16.go#L69-L71

The problem is that all the callers of DecodeRune expect progress by returning the number of bytes consumed. This means that if there's only 1 byte left in an input, it will here return still 0 and the caller(s) don't consume the character.

One example of such a caller is the following:

https://github.com/vitessio/vitess/blob/8f6cfaaa643a08dc111395a75a2d250ee746cfa8/go/mysql/collations/charset/convert.go#L73-L79

The logic here moves forward the pointer in the input []byte but if DecodeRune returns 0 in case of error, it will keep running forever. The OOM happens since it keeps adding the ? as the invalid character to the destination buffer infinitely, growing forever until it runs out of memory.

The fix here would be to always return forward progress also on invalid strings.

There's also a separate bug here that even if progress is guaranteed, select _utf16 0xFF will return the wrong result currently. MySQL will pad here the input when the _utf16 introducer is used with leading 0x00 bytes and then decode to UTF-16, resulting in the output of ÿ here.

PoC

select _utf16 0xFF

Impact

Denial of service attack by triggering unbounded memory usage.

Database specific
{
    "nvd_published_at": "2024-05-08T14:15:08Z",
    "cwe_ids": [
        "CWE-835"
    ],
    "severity": "MODERATE",
    "github_reviewed": true,
    "github_reviewed_at": "2024-05-08T14:32:32Z"
}
References

Affected packages

Go / github.com/vitessio/vitess

Package

Name
github.com/vitessio/vitess
View open source insights on deps.dev
Purl
pkg:golang/github.com/vitessio/vitess

Affected ranges

Type
SEMVER
Events
Introduced
19.0.0
Fixed
19.0.4

Go / github.com/vitessio/vitess

Package

Name
github.com/vitessio/vitess
View open source insights on deps.dev
Purl
pkg:golang/github.com/vitessio/vitess

Affected ranges

Type
SEMVER
Events
Introduced
18.0.0
Fixed
18.0.5

Go / github.com/vitessio/vitess

Package

Name
github.com/vitessio/vitess
View open source insights on deps.dev
Purl
pkg:golang/github.com/vitessio/vitess

Affected ranges

Type
SEMVER
Events
Introduced
0Unknown introduced version / All previous versions are affected
Fixed
17.0.7

Go / vitess.io/vitess

Package

Name
vitess.io/vitess
View open source insights on deps.dev
Purl
pkg:golang/vitess.io/vitess

Affected ranges

Type
SEMVER
Events
Introduced
0Unknown introduced version / All previous versions are affected
Fixed
0.17.7

Go / vitess.io/vitess

Package

Name
vitess.io/vitess
View open source insights on deps.dev
Purl
pkg:golang/vitess.io/vitess

Affected ranges

Type
SEMVER
Events
Introduced
0.18.0
Fixed
0.18.5

Go / vitess.io/vitess

Package

Name
vitess.io/vitess
View open source insights on deps.dev
Purl
pkg:golang/vitess.io/vitess

Affected ranges

Type
SEMVER
Events
Introduced
0.19.0
Fixed
0.19.4