mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-31 10:52:28 -05:00 
			
		
		
		
	- codeberg.org/gruf/go-ffmpreg: v0.6.9 -> v0.6.10
- github.com/ncruces/go-sqlite3: v0.27.1 -> v0.28.0
- github.com/stretchr/testify: v1.10.0 -> v1.11.1
- github.com/tdewolff/minify/v2 v2.23.11 -> v2.24.2
- go.opentelemetry.io/otel{,/*}: v1.37.0 -> v1.38.0
- go.opentelemetry.io/contrib/*: v0.62.0 -> v0.63.0
Reviewed-on: https://codeberg.org/superseriousbusiness/gotosocial/pulls/4406
Co-authored-by: kim <grufwub@gmail.com>
Co-committed-by: kim <grufwub@gmail.com>
		
	
			
		
			
				
	
	
		
			135 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			135 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package vfs
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	_ "embed"
 | |
| 	"encoding/binary"
 | |
| 
 | |
| 	"github.com/tetratelabs/wazero/api"
 | |
| 
 | |
| 	"github.com/ncruces/go-sqlite3/internal/util"
 | |
| 	"github.com/ncruces/go-sqlite3/util/sql3util"
 | |
| )
 | |
| 
 | |
| func cksmWrapFile(file File, flags OpenFlag) File {
 | |
| 	// Checksum only main databases.
 | |
| 	if flags&OPEN_MAIN_DB == 0 {
 | |
| 		return file
 | |
| 	}
 | |
| 	return &cksmFile{File: file}
 | |
| }
 | |
| 
 | |
| type cksmFile struct {
 | |
| 	File
 | |
| 	verifyCksm  bool
 | |
| 	computeCksm bool
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) ReadAt(p []byte, off int64) (n int, err error) {
 | |
| 	n, err = c.File.ReadAt(p, off)
 | |
| 	p = p[:n]
 | |
| 
 | |
| 	if isHeader(p, off) {
 | |
| 		c.init((*[100]byte)(p))
 | |
| 	}
 | |
| 
 | |
| 	// Verify checksums.
 | |
| 	if c.verifyCksm && sql3util.ValidPageSize(len(p)) {
 | |
| 		cksm1 := cksmCompute(p[:len(p)-8])
 | |
| 		cksm2 := *(*[8]byte)(p[len(p)-8:])
 | |
| 		if cksm1 != cksm2 {
 | |
| 			return 0, _IOERR_DATA
 | |
| 		}
 | |
| 	}
 | |
| 	return n, err
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) WriteAt(p []byte, off int64) (n int, err error) {
 | |
| 	if isHeader(p, off) {
 | |
| 		c.init((*[100]byte)(p))
 | |
| 	}
 | |
| 
 | |
| 	// Compute checksums.
 | |
| 	if c.computeCksm && sql3util.ValidPageSize(len(p)) {
 | |
| 		*(*[8]byte)(p[len(p)-8:]) = cksmCompute(p[:len(p)-8])
 | |
| 	}
 | |
| 
 | |
| 	return c.File.WriteAt(p, off)
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) Pragma(name string, value string) (string, error) {
 | |
| 	switch name {
 | |
| 	case "checksum_verification":
 | |
| 		b, ok := sql3util.ParseBool(value)
 | |
| 		if ok {
 | |
| 			c.verifyCksm = b && c.computeCksm
 | |
| 		}
 | |
| 		if !c.verifyCksm {
 | |
| 			return "0", nil
 | |
| 		}
 | |
| 		return "1", nil
 | |
| 
 | |
| 	case "page_size":
 | |
| 		if c.computeCksm && value != "" {
 | |
| 			// Do not allow page size changes on a checksum database.
 | |
| 			return "", nil
 | |
| 		}
 | |
| 	}
 | |
| 	return "", _NOTFOUND
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) DeviceCharacteristics() DeviceCharacteristic {
 | |
| 	ret := c.File.DeviceCharacteristics()
 | |
| 	if c.verifyCksm {
 | |
| 		ret &^= IOCAP_SUBPAGE_READ
 | |
| 	}
 | |
| 	return ret
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) fileControl(ctx context.Context, mod api.Module, op _FcntlOpcode, pArg ptr_t) _ErrorCode {
 | |
| 	if op == _FCNTL_PRAGMA {
 | |
| 		rc := vfsFileControlImpl(ctx, mod, c, op, pArg)
 | |
| 		if rc != _NOTFOUND {
 | |
| 			return rc
 | |
| 		}
 | |
| 	}
 | |
| 	return vfsFileControlImpl(ctx, mod, c.File, op, pArg)
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) init(header *[100]byte) {
 | |
| 	if r := header[20] == 8; r != c.computeCksm {
 | |
| 		c.computeCksm = r
 | |
| 		c.verifyCksm = r
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) SharedMemory() SharedMemory {
 | |
| 	if f, ok := c.File.(FileSharedMemory); ok {
 | |
| 		return f.SharedMemory()
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c *cksmFile) Unwrap() File {
 | |
| 	return c.File
 | |
| }
 | |
| 
 | |
| func isHeader(p []byte, off int64) bool {
 | |
| 	return off == 0 && len(p) >= 100 && bytes.HasPrefix(p, []byte("SQLite format 3\000"))
 | |
| }
 | |
| 
 | |
| func cksmCompute(a []byte) (cksm [8]byte) {
 | |
| 	var s1, s2 uint32
 | |
| 	for len(a) >= 8 {
 | |
| 		s1 += binary.LittleEndian.Uint32(a[0:4]) + s2
 | |
| 		s2 += binary.LittleEndian.Uint32(a[4:8]) + s1
 | |
| 		a = a[8:]
 | |
| 	}
 | |
| 	if len(a) != 0 {
 | |
| 		panic(util.AssertErr())
 | |
| 	}
 | |
| 	binary.LittleEndian.PutUint32(cksm[0:4], s1)
 | |
| 	binary.LittleEndian.PutUint32(cksm[4:8], s2)
 | |
| 	return
 | |
| }
 |