124 lines
2.7 KiB
Go
124 lines
2.7 KiB
Go
// Copyright (c) 2024-2026 Celestino Amoroso (celestino.amoroso@gmail.com).
|
|
// All rights reserved.
|
|
|
|
// builtin-os-file.go
|
|
package expr
|
|
|
|
import (
|
|
"fmt"
|
|
"slices"
|
|
|
|
"git.portale-stac.it/go-pkg/expr/file"
|
|
"git.portale-stac.it/go-pkg/expr/kern"
|
|
)
|
|
|
|
const fileByteIteratorType = "fileByteIterator"
|
|
|
|
type fileFileByteIterator struct {
|
|
reader *file.Reader
|
|
index int64
|
|
count int64
|
|
b byte
|
|
autoClose bool
|
|
}
|
|
|
|
func newFileByteIterator(r *file.Reader, autoClose bool) *fileFileByteIterator {
|
|
return &fileFileByteIterator{reader: r, index: -1, autoClose: autoClose}
|
|
}
|
|
|
|
func (it *fileFileByteIterator) TypeName() string {
|
|
return fileByteIteratorType
|
|
}
|
|
|
|
func (it *fileFileByteIterator) String() string {
|
|
if it.reader.Valid() {
|
|
return fmt.Sprintf("$(%s@%q)", fileByteIteratorType, it.reader.GetName())
|
|
}
|
|
return fmt.Sprintf("$(%s@<nil>)", fileByteIteratorType)
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Count() int64 {
|
|
return it.count
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Next() (item any, err error) { // must return io.EOF after the last item
|
|
if it.b, err = it.reader.ReadByte(); err == nil {
|
|
it.index++
|
|
it.count++
|
|
item = it.b
|
|
} else if it.autoClose {
|
|
it.Clean()
|
|
}
|
|
return
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Current() (item any, err error) {
|
|
item = it.b
|
|
return
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Index() int64 {
|
|
return it.index
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Reset() (err error) {
|
|
if err = it.reader.Reset(); err == nil {
|
|
it.index = -1
|
|
it.count = 0
|
|
it.b = 0
|
|
}
|
|
return
|
|
}
|
|
|
|
func (it *fileFileByteIterator) HasOperation(name string) bool {
|
|
return slices.Contains([]string{kern.NextName, kern.ResetName, kern.IndexName, kern.CountName, kern.CurrentName, kern.CleanName}, name)
|
|
}
|
|
|
|
func (it *fileFileByteIterator) Clean() (err error) {
|
|
if it.reader.Valid() {
|
|
if err = it.reader.GetFile().Close(); err == nil {
|
|
it.reader = nil
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (it *fileFileByteIterator) CallOperation(name string, args map[string]any) (v any, err error) {
|
|
switch name {
|
|
case kern.NextName:
|
|
v, err = it.Next()
|
|
case kern.ResetName:
|
|
err = it.Reset()
|
|
case kern.CleanName:
|
|
err = it.Clean()
|
|
case kern.IndexName:
|
|
v = int64(it.Index())
|
|
case kern.CurrentName:
|
|
v, err = it.Current()
|
|
case kern.CountName:
|
|
v = it.count
|
|
default:
|
|
err = kern.ErrNoOperation(name)
|
|
}
|
|
return
|
|
}
|
|
|
|
func fileByteIteratorFunc(ctx kern.ExprContext, name string, args map[string]any) (result any, err error) {
|
|
var handle *file.Reader
|
|
var invalidFileHandle any
|
|
var autoClose bool
|
|
|
|
result = nil
|
|
|
|
if handle, invalidFileHandle, autoClose, err = initFileHandle(ctx, name, args); err == nil {
|
|
if handle != nil {
|
|
result = newFileByteIterator(handle, autoClose)
|
|
}
|
|
}
|
|
|
|
if err == nil && (handle == nil || invalidFileHandle != nil) {
|
|
err = errInvalidFileHandle(name, invalidFileHandle)
|
|
}
|
|
return
|
|
}
|