mirror of
https://codeberg.org/forgejo/forgejo
synced 2024-09-17 20:31:10 +02:00
792b4dba2c
* update github.com/blevesearch/bleve v2.0.2 -> v2.0.3 * github.com/denisenkom/go-mssqldb v0.9.0 -> v0.10.0 * github.com/editorconfig/editorconfig-core-go v2.4.1 -> v2.4.2 * github.com/go-chi/cors v1.1.1 -> v1.2.0 * github.com/go-git/go-billy v5.0.0 -> v5.1.0 * github.com/go-git/go-git v5.2.0 -> v5.3.0 * github.com/go-ldap/ldap v3.2.4 -> v3.3.0 * github.com/go-redis/redis v8.6.0 -> v8.8.2 * github.com/go-sql-driver/mysql v1.5.0 -> v1.6.0 * github.com/go-swagger/go-swagger v0.26.1 -> v0.27.0 * github.com/lib/pq v1.9.0 -> v1.10.1 * github.com/mattn/go-sqlite3 v1.14.6 -> v1.14.7 * github.com/go-testfixtures/testfixtures v3.5.0 -> v3.6.0 * github.com/issue9/identicon v1.0.1 -> v1.2.0 * github.com/klauspost/compress v1.11.8 -> v1.12.1 * github.com/mgechev/revive v1.0.3 -> v1.0.6 * github.com/microcosm-cc/bluemonday v1.0.7 -> v1.0.8 * github.com/niklasfasching/go-org v1.4.0 -> v1.5.0 * github.com/olivere/elastic v7.0.22 -> v7.0.24 * github.com/pelletier/go-toml v1.8.1 -> v1.9.0 * github.com/prometheus/client_golang v1.9.0 -> v1.10.0 * github.com/xanzy/go-gitlab v0.44.0 -> v0.48.0 * github.com/yuin/goldmark v1.3.3 -> v1.3.5 * github.com/6543/go-version v1.2.4 -> v1.3.1 * do github.com/lib/pq v1.10.0 -> v1.10.1 again ...
165 lines
3.9 KiB
Go
Vendored
165 lines
3.9 KiB
Go
Vendored
// Copyright (C) MongoDB, Inc. 2017-present.
|
|
//
|
|
// 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 at http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
package bsoncore
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
)
|
|
|
|
// NewArrayLengthError creates and returns an error for when the length of an array exceeds the
|
|
// bytes available.
|
|
func NewArrayLengthError(length, rem int) error {
|
|
return lengthError("array", length, rem)
|
|
}
|
|
|
|
// Array is a raw bytes representation of a BSON array.
|
|
type Array []byte
|
|
|
|
// NewArrayFromReader reads an array from r. This function will only validate the length is
|
|
// correct and that the array ends with a null byte.
|
|
func NewArrayFromReader(r io.Reader) (Array, error) {
|
|
return newBufferFromReader(r)
|
|
}
|
|
|
|
// Index searches for and retrieves the value at the given index. This method will panic if
|
|
// the array is invalid or if the index is out of bounds.
|
|
func (a Array) Index(index uint) Value {
|
|
value, err := a.IndexErr(index)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return value
|
|
}
|
|
|
|
// IndexErr searches for and retrieves the value at the given index.
|
|
func (a Array) IndexErr(index uint) (Value, error) {
|
|
elem, err := indexErr(a, index)
|
|
if err != nil {
|
|
return Value{}, err
|
|
}
|
|
return elem.Value(), err
|
|
}
|
|
|
|
// DebugString outputs a human readable version of Array. It will attempt to stringify the
|
|
// valid components of the array even if the entire array is not valid.
|
|
func (a Array) DebugString() string {
|
|
if len(a) < 5 {
|
|
return "<malformed>"
|
|
}
|
|
var buf bytes.Buffer
|
|
buf.WriteString("Array")
|
|
length, rem, _ := ReadLength(a) // We know we have enough bytes to read the length
|
|
buf.WriteByte('(')
|
|
buf.WriteString(strconv.Itoa(int(length)))
|
|
length -= 4
|
|
buf.WriteString(")[")
|
|
var elem Element
|
|
var ok bool
|
|
for length > 1 {
|
|
elem, rem, ok = ReadElement(rem)
|
|
length -= int32(len(elem))
|
|
if !ok {
|
|
buf.WriteString(fmt.Sprintf("<malformed (%d)>", length))
|
|
break
|
|
}
|
|
fmt.Fprintf(&buf, "%s", elem.Value().DebugString())
|
|
if length != 1 {
|
|
buf.WriteByte(',')
|
|
}
|
|
}
|
|
buf.WriteByte(']')
|
|
|
|
return buf.String()
|
|
}
|
|
|
|
// String outputs an ExtendedJSON version of Array. If the Array is not valid, this method
|
|
// returns an empty string.
|
|
func (a Array) String() string {
|
|
if len(a) < 5 {
|
|
return ""
|
|
}
|
|
var buf bytes.Buffer
|
|
buf.WriteByte('[')
|
|
|
|
length, rem, _ := ReadLength(a) // We know we have enough bytes to read the length
|
|
|
|
length -= 4
|
|
|
|
var elem Element
|
|
var ok bool
|
|
for length > 1 {
|
|
elem, rem, ok = ReadElement(rem)
|
|
length -= int32(len(elem))
|
|
if !ok {
|
|
return ""
|
|
}
|
|
fmt.Fprintf(&buf, "%s", elem.Value().String())
|
|
if length > 1 {
|
|
buf.WriteByte(',')
|
|
}
|
|
}
|
|
if length != 1 { // Missing final null byte or inaccurate length
|
|
return ""
|
|
}
|
|
|
|
buf.WriteByte(']')
|
|
return buf.String()
|
|
}
|
|
|
|
// Values returns this array as a slice of values. The returned slice will contain valid values.
|
|
// If the array is not valid, the values up to the invalid point will be returned along with an
|
|
// error.
|
|
func (a Array) Values() ([]Value, error) {
|
|
return values(a)
|
|
}
|
|
|
|
// Validate validates the array and ensures the elements contained within are valid.
|
|
func (a Array) Validate() error {
|
|
length, rem, ok := ReadLength(a)
|
|
if !ok {
|
|
return NewInsufficientBytesError(a, rem)
|
|
}
|
|
if int(length) > len(a) {
|
|
return NewArrayLengthError(int(length), len(a))
|
|
}
|
|
if a[length-1] != 0x00 {
|
|
return ErrMissingNull
|
|
}
|
|
|
|
length -= 4
|
|
var elem Element
|
|
|
|
var keyNum int64
|
|
for length > 1 {
|
|
elem, rem, ok = ReadElement(rem)
|
|
length -= int32(len(elem))
|
|
if !ok {
|
|
return NewInsufficientBytesError(a, rem)
|
|
}
|
|
|
|
// validate element
|
|
err := elem.Validate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// validate keys increase numerically
|
|
if fmt.Sprint(keyNum) != elem.Key() {
|
|
return fmt.Errorf("array key %q is out of order or invalid", elem.Key())
|
|
}
|
|
keyNum++
|
|
}
|
|
|
|
if len(rem) < 1 || rem[0] != 0x00 {
|
|
return ErrMissingNull
|
|
}
|
|
return nil
|
|
}
|