horan.kse3-eio_3.0.2.0.source-code.Adaptors.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kse3-eio_3 Show documentation
Show all versions of kse3-eio_3 Show documentation
Kerr Scala Extensions 3, module kse3-eio
// This file is distributed under the BSD 3-clause license. See file LICENSE.
// Copyright (c) 2014-15, 2020-23 Rex Kerr, UCSF, and Calico Life Sciences LLC
package kse.eio
import java.io._
import java.nio._
import java.nio.file._
import java.nio.channels.{ReadableByteChannel, WritableByteChannel, SeekableByteChannel}
import java.nio.charset.StandardCharsets._
import java.util.Base64
import scala.language.experimental.relaxedExtensionImports
import kse.basics.{given, _}
import kse.flow.{given, _}
import kse.maths.{given, _}
import kse.maths.packed.{given, _}
object EioBase64 {
private val encode64urlTable: Array[Byte] = Array(
'A'.toByte, 'B'.toByte, 'C'.toByte, 'D'.toByte, 'E'.toByte, 'F'.toByte, 'G'.toByte, 'H'.toByte,
'I'.toByte, 'J'.toByte, 'K'.toByte, 'L'.toByte, 'M'.toByte, 'N'.toByte, 'O'.toByte, 'P'.toByte,
'Q'.toByte, 'R'.toByte, 'S'.toByte, 'T'.toByte, 'U'.toByte, 'V'.toByte, 'W'.toByte, 'X'.toByte,
'Y'.toByte, 'Z'.toByte, 'a'.toByte, 'b'.toByte, 'c'.toByte, 'd'.toByte, 'e'.toByte, 'f'.toByte,
'g'.toByte, 'h'.toByte, 'i'.toByte, 'j'.toByte, 'k'.toByte, 'l'.toByte, 'm'.toByte, 'n'.toByte,
'o'.toByte, 'p'.toByte, 'q'.toByte, 'r'.toByte, 's'.toByte, 't'.toByte, 'u'.toByte, 'v'.toByte,
'w'.toByte, 'x'.toByte, 'y'.toByte, 'z'.toByte, '0'.toByte, '1'.toByte, '2'.toByte, '3'.toByte,
'4'.toByte, '5'.toByte, '6'.toByte, '7'.toByte, '8'.toByte, '9'.toByte, '-'.toByte, '_'.toByte
)
private val decode64Table: Array[Byte] = Array(
// Starts at index 9, given in groups of 9
64, 64, -1, -1, 64, -1, -1, -1, -1, // 9-17
-1, -1, -1, -1, -1, -1, -1, -1, -1, // 18-26
-1, -1, -1, -1, -1, 64, -1, -1, -1, // 27-35
-1, -1, -1, -1, -1, -1, -1, 62, -1, // 36-44
62, -1, 63, 52, 53, 54, 55, 56, 57, // 45-53
58, 59, 60, 61, -1, -1, -1, 64, -1, // 54-62
-1, -1, 0, 1, 2, 3, 4, 5, 6, // 63-71
7, 8, 9, 10, 11, 12, 13, 14, 15, // 72-80
16, 17, 18, 19, 20, 21, 22, 23, 24, // 81-89
25, -1, -1, -1, -1, 63, -1, 26, 27, // 90-98
28, 29, 30, 31, 32, 33, 34, 35, 36, // 99-107
37, 38, 39, 40, 41, 42, 43, 44, 45, //108-115
46, 47, 48, 49, 50, 51 //117-122
)
def encodeUrlRange(raw: Array[Byte], i0: Int, iN: Int, lineLength: Int = Int.MaxValue): Array[Byte] =
if i0 < 0 || iN > raw.length then throw new RuntimeException(s"Invalid input range: $i0 until $iN in input of length ${raw.length}")
var l = ((iN - i0)*4 + 2)/3
val n = if lineLength < 4 then 1 else lineLength/4
if l > 4*n then l += (l-1)/(4*n)
if l > Int.MaxValue - 7 then throw new RuntimeException(s"Overly long Base64 encode: would require $l bytes")
val a = new Array[Byte](l.toInt)
var i = i0
var j = 0
var k = 0
while i+2 < iN do
k += 1
if k > n then
k = 1
a(j) = '\n'.toByte
j += 1
val chunk = ((raw(i) & 0xFF) << 16) | ((raw(i+1) & 0xFF) << 8) | (raw(i+2) & 0xFF)
a(j ) = encode64urlTable( chunk >>> 18 )
a(j+1) = encode64urlTable((chunk >>> 12) & 0x3F)
a(j+2) = encode64urlTable((chunk >>> 6) & 0x3F)
a(j+3) = encode64urlTable( chunk & 0x3F)
i += 3
j += 4
if i < iN then
if k >= n then
a(j) = '\n'.toByte
j += 1
if i + 1 == iN then
val chunk = (raw(i) & 0xFF) << 16
a(j ) = encode64urlTable( chunk >>> 18)
a(j+1) = encode64urlTable((chunk >>> 12) & 0x3F)
else
val chunk = ((raw(i) & 0xFF) << 16) | ((raw(i+1) & 0xFF) << 8)
a(j ) = encode64urlTable( chunk >>> 18 )
a(j+1) = encode64urlTable((chunk >>> 12) & 0x3F)
a(j+2) = encode64urlTable((chunk >>> 6) & 0x3F)
a
def encodeUrl(raw: Array[Byte], lineLength: Int = Int.MaxValue): Array[Byte] =
encodeUrlRange(raw, 0, raw.length, lineLength)
def stringEncodeUrl(raw: Array[Byte], lineLength: Int = Int.MaxValue): String =
new String(encodeUrlRange(raw, 0, raw.length, lineLength), ISO_8859_1)
def decodeRangeInto(encoded: Array[Byte], i0: Int, iN: Int)(target: Array[Byte], index: Int): Int Or Err = Or.Ret:
var bits = 0
var nb = 0
var i = i0
if i0 < 0 || iN > encoded.length then Err.break(s"Input $i0 until $iN not within input of length ${encoded.length}")
if index < 0 then Err.break(s"Output index is negative: $index")
var j = index
while i < iN do
val b = encoded(i)
val y =
if b >= 9 && b <= 122 then decode64Table(b-9)
else -1
if y == -1 then Err.break(s"Invalid Base64 character ${b.toChar} at index $i")
if y != 64 then
bits = (bits << 6) | y
nb += 6
if nb >= 24 then
if j > target.length-2 then Err.break(s"Insufficient space in decoding array with ${4+(iN-i)} input bytes remaining")
target(j) = bits.byte(2); j += 1
target(j) = bits.byte(1); j += 1
target(j) = bits.byte(0); j += 1
bits = 0
nb = 0
i += 1
if nb >= 8 then
if j >= target.length then Err.break(s"Insufficient space in decoding array with $nb input bits remaining")
bits = bits << (24 - nb)
target(j) = bits.byte(2)
bits = bits & 0xFFFF
j += 1
if nb >= 16 then
if j >= target.length then Err.break(s"Insufficient space in decoding array with ${nb - 8} input bits remaining")
target(j) = bits.byte(1)
bits = bits & 0xFF
j += 1
if bits != 0 then Err.break(s"Trailing bits: ${if bits.byte(1) != 0 then bits.byte(1).bitString else bits.byte(0).bitString}; Base64 data must be incomplete")
j - index
def decodeInto(encoded: Array[Byte])(target: Array[Byte], index: Int): Int Or Err =
decodeRangeInto(encoded, 0, encoded.length)(target, index)
def decodeRange(encoded: Array[Byte], i0: Int, iN: Int): Array[Byte] Or Err =
if i0 < 0 || iN > encoded.length then Err.or(s"Base64 decode range $i0 until $iN not within array of length ${encoded.length}")
else
val a = new Array[Byte]((((iN - i0).toLong*3 + 2) / 4).toInt)
decodeRangeInto(encoded, i0, iN)(a, 0).map(i => a.shrinkCopy(i))
def decode(encoded: Array[Byte]): Array[Byte] Or Err =
val a = new Array[Byte](((encoded.length.toLong*3 + 2)/4).toInt)
decodeRangeInto(encoded, 0, encoded.length)(a, 0).map(i => a.shrinkCopy(i))
def decodeRangeInto(encoded: String, i0: Int, iN: Int)(target: Array[Byte], index: Int): Int Or Err = Or.Ret:
var bits = 0
var nb = 0
var i = i0
if iN > encoded.length then Err.break(s"Input length exceeded: $i0 until $iN not within ${encoded.length}")
if index < 0 then Err.break(s"Output index is negative: $index")
var j = index
while i < iN do
val c = encoded.charAt(i)
val y =
if c >= 9 && c <= 122 then decode64Table(c-9)
else -1
if y == -1 then Err.break(s"Invalid Base64 character $c at index $i")
if y != 64 then
bits = (bits << 6) | y
nb += 6
if nb >= 24 then
if j > target.length-2 then Err.break(s"Insufficient space in decoding array with ${4+(iN-i)} input bytes remaining")
target(j) = bits.byte(2); j += 1
target(j) = bits.byte(1); j += 1
target(j) = bits.byte(0); j += 1
bits = 0
nb = 0
i += 1
if nb >= 8 then
if j >= target.length then Err.break(s"Insufficient space in decoding array with $nb input bits remaining")
bits = bits << (24 - nb)
target(j) = bits.byte(2)
bits = bits & 0xFFFF
j += 1
if nb >= 16 then
if j >= target.length then Err.break(s"Insufficient space in decoding array with ${nb - 8} input bits remaining")
target(j) = bits.byte(1)
bits = bits & 0xFF
j += 1
if bits != 0 then Err.break(s"Trailing bits: ${if bits.byte(1) != 0 then bits.byte(1).bitString else bits.byte(0).bitString}; Base64 data must be incomplete")
j - index
def decodeInto(encoded: String)(target: Array[Byte], index: Int): Int Or Err =
decodeRangeInto(encoded, 0, encoded.length)(target, index)
def decodeRange(encoded: String, i0: Int, iN: Int): Array[Byte] Or Err =
if i0 < 0 || iN > encoded.length then Err.or(s"Base64 decode range $i0 until $iN not within string of length ${encoded.length}")
else
val a = new Array[Byte]((((iN - i0).toLong*3) / 4).toInt)
decodeRangeInto(encoded, i0, iN)(a, 0).map(i => a.shrinkCopy(i))
def decode(encoded: String): Array[Byte] Or Err =
val a = new Array[Byte](((encoded.length.toLong*3)/4).toInt)
decodeRangeInto(encoded, 0, encoded.length)(a, 0).map(i => a.shrinkCopy(i))
}
object EioBase85 {
private def encodeTable_to_decodeTable(encoder: Array[Byte]): Array[Byte] =
if encoder.length == 0 then encoder
else
var bmin = '\t'.toByte
var bmax = ' '.toByte
encoder.peek(){ b =>
if b < bmin then bmin = b
if b > bmax then bmax = b
}
if bmin < '\t' then throw new Exception(s"Tried to decode a table with indices smaller than tab: $bmin")
val decoder: Array[Byte] = Array.fill[Byte](1 + bmax - 9)(-1)
encoder.visit(){ (b, i) =>
if decoder(b - 9) != -1 then throw new Exception(s"Encoder irreversible because of double mapping to $b")
decoder(b - 9) = i.toByte
}
if decoder( 0) == -1 then decoder( 0) = (encoder.length).toByte
if decoder( 1) == -1 then decoder( 1) = (encoder.length).toByte
if decoder( 3) == -1 then decoder( 3) = (encoder.length).toByte
if decoder(23) == -1 then decoder(23) = (encoder.length).toByte
decoder
private val encodeAsciiTable: Array[Byte] = Array.tabulate(85)(i => (i+33).toByte)
private val encodeZmqTable: Array[Byte] = Array(
'0'.toByte, '1'.toByte, '2'.toByte, '3'.toByte, '4'.toByte, '5'.toByte, '6'.toByte, '7'.toByte, '8'.toByte, '9'.toByte,
'a'.toByte, 'b'.toByte, 'c'.toByte, 'd'.toByte, 'e'.toByte, 'f'.toByte, 'g'.toByte, 'h'.toByte, 'i'.toByte, 'j'.toByte,
'k'.toByte, 'l'.toByte, 'm'.toByte, 'n'.toByte, 'o'.toByte, 'p'.toByte, 'q'.toByte, 'r'.toByte, 's'.toByte, 't'.toByte,
'u'.toByte, 'v'.toByte, 'w'.toByte, 'x'.toByte, 'y'.toByte, 'z'.toByte, 'A'.toByte, 'B'.toByte, 'C'.toByte, 'D'.toByte,
'E'.toByte, 'F'.toByte, 'G'.toByte, 'H'.toByte, 'I'.toByte, 'J'.toByte, 'K'.toByte, 'L'.toByte, 'M'.toByte, 'N'.toByte,
'O'.toByte, 'P'.toByte, 'Q'.toByte, 'R'.toByte, 'S'.toByte, 'T'.toByte, 'U'.toByte, 'V'.toByte, 'W'.toByte, 'X'.toByte,
'Y'.toByte, 'Z'.toByte, '.'.toByte, '-'.toByte, ':'.toByte, '+'.toByte, '='.toByte, '^'.toByte, '!'.toByte, '/'.toByte,
'*'.toByte, '?'.toByte, '&'.toByte, '<'.toByte, '>'.toByte, '('.toByte, ')'.toByte, '['.toByte, ']'.toByte, '{'.toByte,
'}'.toByte, '@'.toByte, '%'.toByte, '$'.toByte, '#'.toByte
)
private val decodeZmqTable = encodeTable_to_decodeTable(encodeZmqTable)
private val decodeAsciiTable = encodeTable_to_decodeTable(encodeAsciiTable)
private def encodeRangeWithTable(raw: Array[Byte], i0: Int, iN: Int, table: Array[Byte]): Array[Byte] =
if i0 < 0 || iN > raw.length then throw new RuntimeException(s"Range $i0 until $iN out of bounds of array of length ${raw.length}")
val l = ((iN - i0)*5 + 3)/4
if l > Int.MaxValue - 7 then throw new RuntimeException(s"Overly long Base85 encode: would require $l bytes")
val result = new Array[Byte](l.toInt)
var code = 0
var i = i0
var j = 0
while i+3 < iN do
code = Pack.I(raw(i+3), raw(i+2), raw(i+1), raw(i))
i += 4
var c85 = (code.u / 85.u).s
result(j+4) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+3) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+2) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+1) = table(code - 85*c85)
result(j) = table(c85)
j += 5
if i < iN then
val overflow = iN - i
code = 0
while i < iN do
code = (code << 8) | (raw(i) & 0xFF)
i += 1
code = code << (8 * (4 - overflow))
var c = (code.u / 52200625.u).s
result(j) = table(c)
code = code - 52200625*c
c = code / 614125
result(j+1) = table(c)
if overflow > 1 then
code = code - 614125*c
c = code / 7225
result(j+2) = table(c)
if overflow > 2 then
code = code - 7225*c
c = code / 85
result(j+3) = table(c)
result
def encodeZmqRange(raw: Array[Byte], i0: Int, iN: Int): Array[Byte] = encodeRangeWithTable(raw, i0, iN, encodeZmqTable)
def encodeZmq(raw: Array[Byte]): Array[Byte] = encodeRangeWithTable(raw, 0, raw.length, encodeZmqTable)
def encodeZmq(raw: Array[Int]): Array[Byte] = encodeRangeWithTable(raw.unpackBytes, 0, raw.length*4, encodeZmqTable)
def stringEncodeZmq(raw: Array[Byte]): String = new String(encodeZmq(raw), ISO_8859_1)
def encodeAsciiRange(raw: Array[Byte], i0: Int, iN: Int): Array[Byte] = encodeRangeWithTable(raw, i0, iN, encodeAsciiTable)
def encodeAscii(raw: Array[Byte]): Array[Byte] =
inline def table(i: Int) = (i + 33).toByte
val l = (raw.length.toLong*5 + 3)/4
if l > Int.MaxValue - 7 then throw new RuntimeException(s"Overly long Base85 encode: would require $l bytes")
val result = new Array[Byte](l.toInt)
var code = 0
var i = 0
var j = 0
while i+3 < raw.length do
code = Pack.I(raw(i+3), raw(i+2), raw(i+1), raw(i))
i += 4
var c85 = (code.u / 85.u).s
result(j+4) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+3) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+2) = table(code - 85*c85)
code = c85
c85 = c85 / 85
result(j+1) = table(code - 85*c85)
result(j) = table(c85)
j += 5
if i < raw.length then
val overflow = raw.length - i
code = 0
while i < raw.length do
code = (code << 8) | (raw(i) & 0xFF)
i += 1
code = code << (8 * (4 - overflow))
var c = (code.u / 52200625.u).s
result(j) = table(c)
code = code - 52200625*c
c = code / 614125
result(j+1) = table(c)
if overflow > 1 then
code = code - 614125*c
c = code / 7225
result(j+2) = table(c)
if overflow > 2 then
code = code - 7225*c
c = code / 85
result(j+3) = table(c)
result
def stringEncodeAscii(raw: Array[Byte]): String = new String(encodeAscii(raw), ISO_8859_1)
private def decodeRangeWithTable(encoded: Array[Byte], i0: Int, iN: Int, table: Array[Byte]): Array[Byte] Or Err = Or.Ret:
if i0 < 0 || iN > encoded.length then Err.break(s"Invalid decoding range: $i0 until $iN in array of length ${encoded.length}")
val l = ((iN - i0).toLong*4)/5
var a = new Array[Byte](l.toInt)
var lead: Int = 0
var rest: Int = 0
var stored = 0
var i = i0
var j = 0
val bmax = 9 + table.length - 1
while i < iN do
val b = encoded(i)
val y =
if b >= 9 && b <= bmax then table(b - 9)
else -1
if y < 0 then Err.break(s"Invalid Base85 character for this encoding: '${b.toChar}' at $i")
else if y < 85 then
if stored == 0 then
lead = y
stored = 1
else
rest = 85*rest + y
stored += 1
if stored == 5 then
stored = 0
if lead > 82 || (lead == 82 && rest > 14516045) then Err.break(s"Over-full Base85 number $lead*85^4 + $rest ending at $i")
rest = lead*52200625 + rest
a(j) = ( rest >>> 24 ).toByte
a(j+1) = ((rest >>> 16) & 0xFF).toByte
a(j+2) = ((rest >>> 8) & 0xFF).toByte
a(j+3) = ( rest & 0xFF).toByte
rest = 0
j += 4
i += 1
if stored > 0 then
val n = stored
while stored < 5 do
rest = rest*85 + 84
stored += 1
if lead > 82 || (lead == 82 && rest > 14516045) then Err.break(s"Over-full Base85 number $lead*85^4 + $rest at end of input")
rest = lead*52200625 + rest
a(j) = (rest >>> 24).toByte
j += 1
if n > 2 then
a(j) = ((rest >>> 16) & 0xFF).toByte
j += 1
if n > 3 then
a(j) = ((rest >>> 8) & 0xFF).toByte
j += 1
a.shrinkCopy(j)
def decodeZmqRange(encoded: Array[Byte], i0: Int, iN: Int): Array[Byte] Or Err = decodeRangeWithTable(encoded, i0, iN, decodeZmqTable)
def decodeAsciiRange(encoded: Array[Byte], i0: Int, iN: Int): Array[Byte] Or Err = decodeRangeWithTable(encoded, i0, iN, decodeAsciiTable)
def decodeZmq(encoded: Array[Byte]): Array[Byte] Or Err = decodeRangeWithTable(encoded, 0, encoded.length, decodeZmqTable)
def decodeAscii(encoded: Array[Byte]): Array[Byte] Or Err = decodeRangeWithTable(encoded, 0, encoded.length, decodeAsciiTable)
private def decodeRangeWithTable(encoded: String, i0: Int, iN: Int, table: Array[Byte]): Array[Byte] Or Err = Or.Ret:
if i0 < 0 || iN > encoded.length then Err.break(s"Invalid decoding range: $i0 until $iN in string of length ${encoded.length}")
val l = ((iN - i0).toLong*4)/5
var a = new Array[Byte](l.toInt)
var lead: Int = 0
var rest: Int = 0
var stored = 0
var i = i0
var j = 0
val bmax = 9 + table.length - 1
while i < iN do
val b = encoded.charAt(i)
val y =
if b >= 9 && b <= bmax then table(b - 9)
else -1
if y < 0 then Err.break(s"Invalid Base85 character for this encoding: '${b.toChar}' at $i")
else if y < 85 then
if stored == 0 then
lead = y
stored = 1
else
rest = 85*rest + y
stored += 1
if stored == 5 then
stored = 0
if lead > 82 || (lead == 82 && rest > 14516045) then Err.break(s"Over-full Base85 number $lead*85^4 + $rest ending at $i")
rest = lead*52200625 + rest
a(j) = ( rest >>> 24 ).toByte
a(j+1) = ((rest >>> 16) & 0xFF).toByte
a(j+2) = ((rest >>> 8) & 0xFF).toByte
a(j+3) = ( rest & 0xFF).toByte
rest = 0
j += 4
i += 1
if stored > 0 then
val n = stored
while stored < 5 do
rest = rest*85 + 84
stored += 1
if lead > 82 || (lead == 82 && rest > 14516045) then Err.break(s"Over-full Base85 number $lead*85^4 + $rest at end of input")
rest = lead*52200625 + rest
a(j) = (rest >>> 24).toByte
j += 1
if n > 2 then
a(j) = ((rest >>> 16) & 0xFF).toByte
j += 1
if n > 3 then
a(j) = ((rest >>> 8) & 0xFF).toByte
j += 1
a.shrinkCopy(j)
def decodeZmqRange(encoded: String, i0: Int, iN: Int): Array[Byte] Or Err = decodeRangeWithTable(encoded, i0, iN, decodeZmqTable)
def decodeAsciiRange(encoded: String, i0: Int, iN: Int): Array[Byte] Or Err = decodeRangeWithTable(encoded, i0, iN, decodeAsciiTable)
def decodeZmq(encoded: String): Array[Byte] Or Err = decodeRangeWithTable(encoded, 0, encoded.length, decodeZmqTable)
def decodeAscii(encoded: String): Array[Byte] Or Err = decodeRangeWithTable(encoded, 0, encoded.length, decodeAsciiTable)
}
extension (underlying: Array[Byte]) {
inline def utf8 = new String(underlying, UTF_8)
inline def ascii = new String(underlying, US_ASCII)
inline def rawString = new String(underlying, ISO_8859_1)
inline def iso8859_1 = new String(underlying, ISO_8859_1)
inline def stringEncode64 = Base64.getUrlEncoder.encodeToString(underlying)
inline def stringEncode64basic = Base64.getEncoder.encodeToString(underlying)
inline def stringEncode64url = Base64.getUrlEncoder.encodeToString(underlying)
inline def stringEncode64mime = Base64.getMimeEncoder.encodeToString(underlying)
inline def stringEncode64lines = EioBase64.stringEncodeUrl(underlying, 76)
inline def stringEncode85 = EioBase85.stringEncodeZmq(underlying)
inline def stringEncode85zmq = EioBase85.stringEncodeZmq(underlying)
inline def stringEncode85ascii = EioBase85.stringEncodeAscii(underlying)
inline def encode64 = Base64.getUrlEncoder.encode(underlying)
inline def encode64basic = Base64.getEncoder().encode(underlying)
inline def encode64url = Base64.getUrlEncoder.encode(underlying)
inline def encode64mime = Base64.getMimeEncoder().encode(underlying)
inline def encode64lines = EioBase64.encodeUrl(underlying, 76)
inline def decode64 = EioBase64.decode(underlying)
inline def decode64basic = safe{ Base64.getDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def decode64url = safe{ Base64.getUrlDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def decode64mime = safe{ Base64.getMimeDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def encode85 = EioBase85.encodeZmq(underlying)
inline def encode85zmq = EioBase85.encodeZmq(underlying)
inline def encode85ascii = EioBase85.encodeAscii(underlying)
inline def decode85 = EioBase85.decodeZmq(underlying)
inline def decode85zmq = EioBase85.decodeZmq(underlying)
inline def decode85ascii = EioBase85.decodeAscii(underlying)
}
extension (underlying: String) {
inline def bytes: Array[Byte] = underlying.getBytes(UTF_8)
inline def decode64 = EioBase64.decode(underlying)
inline def decode64basic = safe{ Base64.getDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def decode64url = safe{ Base64.getUrlDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def decode64mime = safe{ Base64.getMimeDecoder().decode(underlying) }.mapAlt(t => Err(t.getMessage))
inline def decode85 = EioBase85.decodeZmq(underlying)
inline def decode85zmq = EioBase85.decodeZmq(underlying)
inline def decode85ascii = EioBase85.decodeAscii(underlying)
}
extension (underlying: java.util.zip.ZipEntry) {
def cleanName =
val n = underlying.getName
val i = n.indexOf('/')
val j = n.indexOf('\\')
if i < 0 && j > 0 then n.replace('\\', '/') else n
def cleanPath =
val n = underlying.cleanName
val fs = FileSystems.getDefault
fs.getPath(if fs.getSeparator == "/" then n else n.replace("/", fs.getSeparator))
}
//////////////////////////////////////////////////////////////
// Adaptors between different ways to hold/view binary data //
//////////////////////////////////////////////////////////////
trait CountedTransfer[S, T] {
def apply(source: S, offset: Int)(target: T, index: Int)(count: Int): Int
}
object CountedTransfer {
val array2buffer: CountedTransfer[Array[Byte], ByteBuffer] = new CountedTransfer[Array[Byte], ByteBuffer] {
def apply(source: Array[Byte], offset: Int)(target: ByteBuffer, index: Int)(count: Int): Int =
target.put(source, offset, count)
count
}
val array2array: CountedTransfer[Array[Byte], Array[Byte]] = new CountedTransfer[Array[Byte], Array[Byte]] {
def apply(source: Array[Byte], offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
System.arraycopy(source, offset, target, index, count)
count
}
val buffer2array: CountedTransfer[ByteBuffer, Array[Byte]] = new CountedTransfer[ByteBuffer, Array[Byte]] {
def apply(source: ByteBuffer, offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
source.get(target, index, count)
count
}
val array2stream: CountedTransfer[Array[Byte], OutputStream] = new CountedTransfer[Array[Byte], OutputStream] {
def apply(source: Array[Byte], offset: Int)(target: OutputStream, index: Int)(count: Int): Int =
target.write(source, offset, count)
count
}
val stream2array: CountedTransfer[InputStream, Array[Byte]] = new CountedTransfer[InputStream, Array[Byte]] {
def apply(source: InputStream, offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
source.read(target, index, count)
}
val array2channel: CountedTransfer[Array[Byte], WritableByteChannel] = new CountedTransfer[Array[Byte], WritableByteChannel] {
def apply(source: Array[Byte], offset: Int)(target: WritableByteChannel, index: Int)(count: Int): Int =
target.write(ByteBuffer.wrap(source, offset, count))
}
val channel2array: CountedTransfer[ReadableByteChannel, Array[Byte]] = new CountedTransfer[ReadableByteChannel, Array[Byte]] {
def apply(source: ReadableByteChannel, offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
source.read(ByteBuffer.wrap(target, index, count))
}
val array2seeker: CountedTransfer[Array[Byte], SeekableByteChannel] = new CountedTransfer[Array[Byte], SeekableByteChannel] {
def apply(source: Array[Byte], offset: Int)(target: SeekableByteChannel, index: Int)(count: Int): Int =
target.write(ByteBuffer.wrap(source, offset, count))
}
val seeker2array: CountedTransfer[SeekableByteChannel, Array[Byte]] = new CountedTransfer[SeekableByteChannel, Array[Byte]] {
def apply(source: SeekableByteChannel, offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
source.read(ByteBuffer.wrap(target, index, count))
}
val array2multi: CountedTransfer[Array[Byte], MultiArrayChannel] = new CountedTransfer[Array[Byte], MultiArrayChannel] {
def apply(source: Array[Byte], offset: Int)(target: MultiArrayChannel, index: Int)(count: Int): Int =
target.write(source, offset)(count)
}
val multi2array: CountedTransfer[MultiArrayChannel, Array[Byte]] = new CountedTransfer[MultiArrayChannel, Array[Byte]] {
def apply(source: MultiArrayChannel, offset: Int)(target: Array[Byte], index: Int)(count: Int): Int =
source.read(target, index)(count)
}
}
extension (underlying: Array[Byte])
inline def buffer = ByteBuffer.wrap(underlying).order(ByteOrder.LITTLE_ENDIAN)
inline def input = new ByteArrayInputStream(underlying)
inline def output = new FixedArrayOutputStream(underlying, 0, underlying.length)
inline def readChannel = MultiArrayChannel.fixedBuffer(underlying, underlying.length)
inline def writeChannel = MultiArrayChannel.fixedBuffer(underlying, 0)
inline def growCopy = MultiArrayChannel.copyOf(underlying)
inline def growCopyBy(chunk: Int) = MultiArrayChannel.chunked(chunk).copyOf(underlying)
inline def growCopyMax(maxSize: Long) = MultiArrayChannel.copyOf(underlying, maxSize)
inline def growCopyMaxBy(maxSize: Long, chunk: Int) = MultiArrayChannel.chunked(chunk).copyOf(underlying, maxSize)
final class FixedArrayOutputStream(val buffer: Array[Byte], val i0: Int, val iN: Int) extends OutputStream {
private var isNowOpen = true
private var index = i0
private def checkOpen(): Unit = if !isNowOpen then throw new IOException("OutputStream is closed")
override def close: Unit =
isNowOpen = false
def write(b: Int): Unit =
checkOpen()
if index >= iN then throw new IOException("FixedArrayOutputStream is full")
buffer(index) = (b & 0xFF).toByte
index += 1
override def write(ab: Array[Byte], offset: Int, length: Int): Unit =
if length > 0 then
checkOpen()
if index >= iN then throw new IOException("FixedArrayOutputStream is full")
if iN - index < 0 || iN - index < length then throw new IOException("Buffer underflow for FixedArrayOutputStream")
System.arraycopy(ab, offset, buffer, index, length)
index += length
override def write(ab: Array[Byte]) = write(ab, 0, ab.length)
def count: Int = index - i0
}
final class MultiArrayChannel private (val growthLimit: Long, val maxChunkSize: Int, existing: Array[Array[Byte]])
extends SeekableByteChannel {
private val initialLimit =
var n = 0L
existing.peek(){ ai => n += ai.length }
n
private var limit = initialLimit
private var storage: Array[Array[Byte]] =
if initialLimit > 0 then
var emptyCount = 0
existing.peek(){ ai => if ai.length == 0 then emptyCount += 1 }
val a = new Array[Array[Byte]](existing.length - emptyCount)
var j = 0
existing.peek(){ ai =>
if ai.length > 0 then
a(j) = ai
j += 1
}
a
else Array(new Array[Byte](256))
private var active: Array[Byte] = storage(0)
private var iactive: Int = 0
private var zero: Long = 0L
private var allocated: Long = if initialLimit > 0 then initialLimit else 256L
private var index: Long = 0L
private var isNowOpen = true
private def checkOpen(): Unit =
if !isNowOpen then throw new java.nio.channels.ClosedChannelException()
private def zeroToIndexOrEnd(): Unit =
if (active ne null) && limit >= zero then active.fillRange((limit - zero).toInt, (index - zero).toInt)(0)
else
if (active ne null) && index > zero then active.fillRange(0, (index - zero).toInt)(0)
var z = zero
var i = iactive
while z > limit && i > 0 do
i -= 1
val a = storage(i)
z -= a.length
if limit > z then a.fillRange((limit - z).toInt, a.length)(0)
else a.fill(0)
private def ensureExtraCapacity(m: Int): Unit =
var needed = (m + index) - allocated
if storage(End).length < maxChunkSize then
val h = storage(End).length
val most = initialLimit + growthLimit - allocated
val expand: Long = if h < 128 then 256 else if storage.length > 3 then maxChunkSize else 2L*h
val l = ((most min expand) min maxChunkSize).toInt
val extra = l - h
if extra > 0 then
needed -= extra
storage(End) = storage(End).copyToSize(l)
allocated += extra
if iactive >= storage.length then
if index >= allocated then zero = allocated
else
active = storage(End)
zero = allocated - active.length
else if iactive == storage.length - 1 then
active = storage(End)
while needed > 0 && allocated < initialLimit + growthLimit do
val most = initialLimit + growthLimit - allocated
val expand: Long = if storage.length > 3 then maxChunkSize else if needed < 128 then 256 else 2L*needed
val l = ((most min expand) min maxChunkSize).toInt
storage = storage.copyToSize(storage.length + 1)
storage(End) = new Array[Byte](l)
allocated += l
needed -= l
if iactive >= storage.length - 1 then
if index >= allocated then
iactive = storage.length
zero = allocated
else
iactive = storage.length - 1
active = storage(iactive)
zero = allocated - active.length
def isOpen: Boolean = isNowOpen
def close(): Unit = isNowOpen = false
def position: Long = index
def position(p: Long): this.type =
checkOpen()
if p < 0 then throw new IllegalArgumentException(s"Negative channel position $p")
else if p > growthLimit + initialLimit then throw new IllegalArgumentException(s"Position $p exceeds maximum possible channel capacity")
else
if p >= allocated then
zero = allocated
active = null
iactive = storage.length
else if p < zero then
while iactive > 0 && p < zero do
iactive -= 1
zero -= storage(iactive).length
active = storage(iactive)
else
while iactive < storage.length && p >= zero + active.length do
zero += active.length
iactive += 1
if iactive < storage.length then active = storage(iactive)
index = p
this
def size: Long = limit
def truncate(p: Long): this.type =
checkOpen()
if p < 0 then throw new IllegalArgumentException(s"Cannot truncate channel to negative size $p")
if p < limit then
if p < index then
position(p)
limit = p
this
def readTo[T](target: T, start: Int)(count: Int)(using transfer: CountedTransfer[Array[Byte], T]): Int =
if count <= 0 then 0
else
checkOpen()
if index >= limit then -1
else
var m = count
var n = 0
while index < limit && m > 0 do
val i = (index - zero).toInt
val r = (limit - index).toInt min (active.length - i)
var h = transfer(active, i)(target, n + start)(if m < r then m else r)
if h < 0 then
h = 0
if m == count then n = -1
index += h
if i + h == active.length then
zero += active.length
iactive += 1
active = if iactive < storage.length then storage(iactive) else null
n += h
m -= h
if h == 0 then m = 0
n
private def readAsMuchAsPossible[T](target: T)(using transfer: CountedTransfer[Array[Byte], T]): Long =
var n: Long = 0
var m = 0
var cycles = 0L
while m >= 0 && cycles <= n do
val count =
if index >= limit || iactive >= storage.length then 1
else availableToRead.clampToInt min active.length - (index - zero).toInt
m = readTo(target, 0)(count)
if m > 0 then n += m
if m >= 0 then cycles += 1
if cycles == 0 then -1L else n
def read(buffer: Array[Byte]): Int =
readTo(buffer, 0)(buffer.length)(using CountedTransfer.array2array)
def read(buffer: Array[Byte], offset: Int)(count: Int): Int =
readTo(buffer, offset)(count)(using CountedTransfer.array2array)
def read(buffer: ByteBuffer): Int =
readTo[ByteBuffer](buffer, 0)(buffer.remaining)(using CountedTransfer.array2buffer)
def read(target: OutputStream): Long =
readAsMuchAsPossible(target)(using CountedTransfer.array2stream)
def read(target: OutputStream)(count: Int): Int =
readTo(target, 0)(count)(using CountedTransfer.array2stream)
def read(target: WritableByteChannel): Long = target match
case that: MultiArrayChannel => readAsMuchAsPossible(that )(using CountedTransfer.array2multi)
case _ => readAsMuchAsPossible(target)(using CountedTransfer.array2channel)
def read(target: WritableByteChannel)(count: Int): Int = target match
case that: MultiArrayChannel => readTo(that, 0)(count)(using CountedTransfer.array2multi)
case _ => readTo(target, 0)(count)(using CountedTransfer.array2channel)
def writeFrom[S](source: S, start: Int)(count: Int)(using transfer: CountedTransfer[S, Array[Byte]]): Int =
if count <= 0 then 0
else
checkOpen()
if index >= initialLimit + growthLimit then return -1
var n = 0
var m = count
if limit < index then zeroToIndexOrEnd()
if allocated - index < m then ensureExtraCapacity(m)
while index < allocated && m > 0 do
val i = (index - zero).toInt
val r = active.length - i
var h = transfer(source, start + n)(active, i)(if m < r then m else r)
if h < 0 then
h = 0
if m == count then n = -1
index += h
if index > limit then limit = index
if h == r then
zero += active.length
iactive += 1
active = if iactive < storage.length then storage(iactive) else null
n += h
m -= h
if h == 0 then m = 0
n
def writeAsMuchAsPossible[S](source: S)(available: S => Long)(using transfer: CountedTransfer[S, Array[Byte]]): Long =
var n: Long = 0
var attempt = 4096 min maxChunkSize
var m = 0
var cycles = 0L
while m >= 0 do
val a = available(source)
val count =
if a > 0 && !(cycles > 16 && (n >> 4) < cycles) then a.clampToInt
else if index >= allocated then attempt
else attempt min active.length - (index - zero).toInt
m = writeFrom(source, 0)(count)
if m > 0 then
n += m
if n >= attempt then
attempt = if attempt > (maxChunkSize >> 1) then maxChunkSize else attempt * 2
if index >= initialLimit + growthLimit then m = -1
cycles += 1
else if m == 0 then cycles += 1
if cycles == 0 then -1L else n
def write(buffer: Array[Byte]): Int =
writeFrom[Array[Byte]](buffer, 0)(buffer.length)(using CountedTransfer.array2array)
def write(buffer: Array[Byte], offset: Int)(count: Int): Int =
writeFrom[Array[Byte]](buffer, offset)(count)(using CountedTransfer.array2array)
def write(buffer: ByteBuffer): Int =
writeFrom[ByteBuffer](buffer, 0)(buffer.remaining)(using CountedTransfer.buffer2array)
def write(source: InputStream): Long =
writeAsMuchAsPossible(source)(_.available)(using CountedTransfer.stream2array)
def write(source: InputStream)(count: Int): Int =
writeFrom(source, 0)(count)(using CountedTransfer.stream2array)
def write(source: ReadableByteChannel): Long = source match
case that: MultiArrayChannel => writeAsMuchAsPossible(that )(_.availableToRead )(using CountedTransfer.multi2array)
case sb: SeekableByteChannel => writeAsMuchAsPossible(sb )(c => c.size - c.position)(using CountedTransfer.seeker2array)
case _ => writeAsMuchAsPossible(source)(_ => 0 )(using CountedTransfer.channel2array)
private def compactImpl(before: Boolean, after: Boolean): Unit =
val high = index max limit
val low = index min limit
var li = iactive
var lz = zero
while low < lz && li > 0 do
li -= 1
lz -= storage(li).length
var hi = iactive
var hz = zero
while hi < storage.length && high >= hz + storage(hi).length do
hz += storage(hi).length
hi += 1
if hi > li && before && li > 0 then
storage = storage.copyOfRange(li, storage.length)
iactive -= li
zero -= lz
index -= lz
limit -= lz
allocated -= lz
hi -= li
hz -= lz
li = 0
lz = 0
if hi > li && after && hi+1 < storage.length then
val dropped = allocated - (hz + storage(hi).length)
storage = storage.copyOfRange(0, hi+1)
allocated -= dropped
if storage.length == 1 && before then
if limit == low then
index -= limit
limit = 0
else if index >= limit-index then
System.arraycopy(active, index.toInt, active, 0, (limit - index).toInt)
limit -= index
index = 0
def releasePassed(): Long =
val i = index
compactImpl(true, false)
i - index
def releaseUnused(): Unit = compactImpl(false, true)
def compact(): this.type =
compactImpl(true, true)
this
def availableToRead: Long =
if !isNowOpen then -1L
else if index > limit then 0L else limit - index
def bufferAvailableToWrite: Long =
if !isNowOpen then -1L
else if index > allocated then 0L else allocated - index
def maxAvailableToWrite: Long =
if !isNowOpen then -1L
else initialLimit + growthLimit - index
def canWrite(buffer: ByteBuffer): Boolean =
buffer.remaining == 0 || (maxAvailableToWrite >= buffer.remaining)
def getBytes: Array[Byte] Or Array[Byte] =
if isNowOpen then throw new IllegalArgumentException("Cannot get bytes from open channel")
if storage eq null then throw new IllegalArgumentException("Buffers already detatched")
if limit == 0 then Is(MultiArrayChannel.noBytes)
else
if storage.length == 1 || limit <= storage(0).length then Is(storage(0).copyToSize(limit.toInt))
else
val a = new Array[Byte](if limit > Int.MaxValue - 7 then Int.MaxValue - 7 else limit.toInt)
var m = 0
var i = 0
while m < a.length do
val ai = storage(i)
val h = (a.length - m) min ai.length
System.arraycopy(ai, 0, a, m, h)
m += h
i += 1
a.isIf(_.length == limit)
def detatchBuffers(): Array[Array[Byte]] =
if isNowOpen then throw new IllegalArgumentException("Cannot detatch buffers from open channel")
if storage eq null then throw new IllegalArgumentException("Buffers already detatched")
if limit <= 0 then
storage = null
Array.empty[Array[Byte]]
else if limit == allocated then
val ans = storage
storage = null
ans
else if index == limit then
val a = new Array[Array[Byte]](iactive + 1)
nFor(iactive){ i => a(i) = storage(i) }
a(iactive) = active.shrinkCopy((index - zero).toInt)
storage = null
a
else
var nb = limit
var na = 0
while na < storage.length && nb > 0 do
nb -= storage(na).length
na += 1
val a = new Array[Array[Byte]](na)
nb = limit
nFor(na - 1){ i => val si = storage(i); nb -= si.length; a(i) = si }
a(na - 1) = storage(na - 1).shrinkCopy(nb.toInt)
storage = null
a
}
object MultiArrayChannel {
val noInitialArrays = Array.empty[Array[Byte]]
val noBytes = Array.empty[Byte]
inline val defaultMaxSize = 0x200000000L
inline val defaultChunkSize = 0x20000000
private def make(chunk: Int, xs: Array[Byte]) =
new MultiArrayChannel(defaultMaxSize - xs.length, chunk, if xs == null || xs.length == 0 then noInitialArrays else Array(xs))
private def make(limit: Long, chunk: Int, xs: Array[Byte]) =
new MultiArrayChannel(0L max limit, chunk, if xs == null || xs.length == 0 then noInitialArrays else Array(xs))
def empty() = make(defaultChunkSize, null)
def empty(growthLimit: Long) = make(growthLimit, defaultChunkSize, null)
def fixedBuffer(b: Array[Byte], readableLength: Int) =
val sgbc = new MultiArrayChannel(0L, b.length, Array(b))
if readableLength < 0 then sgbc.truncate(0L)
else if readableLength < b.length then sgbc.truncate(readableLength)
sgbc
def of(b: Array[Byte]) = make(defaultChunkSize, b)
def of(b: Array[Byte], growthLimit: Long) = make(growthLimit, defaultChunkSize, b)
def copyOf(b: Array[Byte]) = make(defaultChunkSize, b.copy)
def copyOf(b: Array[Byte], growthLimit: Long) = make(growthLimit, defaultChunkSize, b.copy)
def copyOf(bb: ByteBuffer) = make(defaultChunkSize, bb.getBytes)
def copyOf(bb: ByteBuffer, growthLimit: Long) = make(growthLimit, defaultChunkSize, bb.getBytes)
def chunked(size: Int): kse.eio.MultiArrayChannel.ChannelChunkSize = ChannelChunkSize(size)
opaque type ChannelChunkSize = Int
object ChannelChunkSize {
inline def apply(size: Int): ChannelChunkSize = size
extension (chunkSize: ChannelChunkSize)
def value: Int = (chunkSize: Int) max 0x8
extension (chunkSize: kse.eio.MultiArrayChannel.ChannelChunkSize) {
def empty() = make(chunkSize.value, null)
def empty(growthLimit: Long) = make(growthLimit, chunkSize.value, null)
def of(b: Array[Byte]) = make(chunkSize.value, b)
def of(b: Array[Byte], growthLimit: Long) = make(growthLimit, chunkSize.value, b)
def copyOf(b: Array[Byte]) = make(chunkSize.value, b.copy)
def copyOf(b: Array[Byte], growthLimit: Long) = make(growthLimit, chunkSize.value, b)
def copyOf(bb: ByteBuffer) = make(chunkSize.value, bb.getBytes)
def copyOf(bb: ByteBuffer, growthLimit: Long) = make(growthLimit, chunkSize.value, bb.getBytes)
}
}
}
final class ByteBufferOutputStream(val buffer: ByteBuffer) extends OutputStream {
private var isNowOpen = true
private def checkOpen(): Unit = if !isNowOpen then throw new IOException("OutputStream is closed")
override def close(): Unit =
isNowOpen = false
def write(b: Int): Unit =
checkOpen()
buffer put (b & 0xFF).toByte
override def write(b: Array[Byte]): Unit =
checkOpen()
buffer put b
override def write(b: Array[Byte], off: Int, len: Int): Unit =
checkOpen()
buffer.put(b, off, len)
}
final class ByteBufferInputStream(val buffer: ByteBuffer) extends InputStream {
buffer.flip
private var isNowOpen = true
private def checkOpen(): Unit = if !isNowOpen then throw new IOException("InputStream is closed")
override def available(): Int =
checkOpen()
buffer.remaining
override def close(): Unit =
isNowOpen = false
override def mark(readlimit: Int): Unit =
checkOpen()
buffer.mark()
override def markSupported = true
def read(): Int =
checkOpen()
buffer.get & 0xFF
override def read(b: Array[Byte]): Int = read(b, 0, b.length)
override def read(b: Array[Byte], offset: Int, len: Int): Int =
if len <= 0 then 0
else
checkOpen()
if buffer.remaining < len then
val n = buffer.remaining
if n == 0 then throw new BufferUnderflowException()
buffer.get(b, offset, n)
n
else
buffer.get(b, offset, len)
len
override def reset: Unit =
checkOpen()
buffer.reset
override def skip(n: Long): Long =
if n < 0 then 0L
else
checkOpen()
if n > buffer.remaining then
val i = buffer.remaining
buffer.position(buffer.limit)
i
else
buffer.position((n + buffer.position).toInt)
n
}
object ByteBufferCompanion {
val emptyBytes = Array.empty[Byte]
}
extension (buffer: ByteBuffer)
inline def output = new ByteBufferOutputStream(buffer)
inline def input = new ByteBufferInputStream(buffer)
def getBytes: Array[Byte] =
if buffer.remaining == 0 then ByteBufferCompanion.emptyBytes
else
val a = new Array[Byte](buffer.remaining)
buffer.get(a)
a
final class RandomAccessFileOutputStream(val raf: RandomAccessFile) extends OutputStream {
override def close(): Unit = raf.close()
override def write(b: Array[Byte]): Unit = raf.write(b)
override def write(b: Array[Byte], off: Int, len: Int): Unit = raf.write(b, off, len)
def write(b: Int): Unit = raf.writeByte(b)
}
final class RandomAccessFileInputStream(val raf: RandomAccessFile) extends InputStream {
private var markedPosition: Long = -1L
override def available(): Int =
(raf.length() - raf.getFilePointer()).clamp(0, Int.MaxValue).toInt
override def close(): Unit = raf.close()
override def mark(readlimit: Int): Unit =
markedPosition = raf.getFilePointer()
override def markSupported = true
def read(): Int = raf.read()
override def read(b: Array[Byte]): Int = raf.read(b)
override def read(b: Array[Byte], offset: Int, len: Int): Int = raf.read(b, offset, len)
override def reset: Unit =
if markedPosition >= 0 then
val l = raf.length()
if markedPosition > l then throw new IOException(s"Reset to $markedPosition in file shortened to $l")
else raf.seek(markedPosition)
else throw new IOException("Reset on unmarked stream")
override def skip(n: Long): Long =
if n < 0 then 0L
else
val l = raf.length()
val p = raf.getFilePointer()
if n > l - p then
raf.seek(l)
l - p
else
raf.seek(p+n)
n
}
extension (raf: RandomAccessFile)
inline def output = new RandomAccessFileOutputStream(raf)
inline def input = new RandomAccessFileInputStream(raf)
final class SeekableByteChannelOutputStream(val sbc: SeekableByteChannel) extends OutputStream {
private val oneByte = ByteBuffer.wrap(new Array[Byte](1))
override def close(): Unit = sbc.close()
override def write(b: Array[Byte]): Unit =
val bb = ByteBuffer wrap b
val n = sbc.write(bb)
if (n < b.length) throw new IOException("Tried to write ${b.length} bytes but only could write $n")
override def write(b: Array[Byte], off: Int, len: Int): Unit =
if (len > 0) {
val bb = ByteBuffer wrap b
bb.position(off)
bb.limit(off +# len)
val n = sbc.write(bb)
if (n < len) throw new IOException("Tried to write ${b.length} bytes but could only write $n")
}
override def write(b: Int): Unit = synchronized {
oneByte.clear
oneByte put b.toByte
oneByte.flip
val n = sbc.write(oneByte)
if (n != 1) throw new IOException("Tried to write a byte but couldn't")
}
}
final class SeekableByteChannelInputStream(val sbc: SeekableByteChannel) extends InputStream {
private var markedPosition: Long = -1L
private val oneByte = ByteBuffer.wrap(new Array[Byte](1))
private def atEndException(): Nothing = throw new IOException("Read at end of SeekableByteChannel")
private def noProgressException(): Nothing = throw new IOException("Read failed")
private def checkReadSize(i: Int): Int =
if i < 0 then atEndException()
else if i == 0 then noProgressException()
else i
override def available(): Int =
(sbc.size - sbc.position).clamp(0, Int.MaxValue).toInt
override def close(): Unit = sbc.close()
override def mark(readlimit: Int): Unit =
markedPosition = sbc.position
override def markSupported = true
def read(): Int =
oneByte.clear
checkReadSize( sbc.read(oneByte) )
oneByte.flip
oneByte.get & 0xFF
override def read(b: Array[Byte]): Int =
if b.length == 0 then 0
else
val bb = ByteBuffer wrap b
checkReadSize( sbc.read(bb) )
override def read(b: Array[Byte], offset: Int, len: Int): Int =
if len <= 0 then 0
else
val bb = ByteBuffer wrap b
bb.position(offset)
bb.limit(offset +# len)
checkReadSize( sbc.read(bb) )
override def reset: Unit =
if markedPosition >= 0 then
val l = sbc.size
if sbc.position > l then throw new IOException(s"Reset to $markedPosition in SeekableByteChannel shortened to $l")
else sbc.position(markedPosition)
else throw new IOException("Reset on unmarked stream")
override def skip(n: Long): Long =
if n < 0 then 0L
else
val l = sbc.size - sbc.position
if n > l then
sbc.position(sbc.size)
l
else
sbc.position(sbc.position + n)
n
}
extension (sbc: SeekableByteChannel)
inline def output = new SeekableByteChannelOutputStream(sbc)
inline def input = new SeekableByteChannelInputStream(sbc)
final class InputStreamByteChannel(input: InputStream) extends ReadableByteChannel {
private val certainlyClosed = java.util.concurrent.atomic.AtomicBoolean(false)
private val backingBuffer: java.util.concurrent.atomic.AtomicReference[Array[Byte]] = java.util.concurrent.atomic.AtomicReference(null)
def close =
input.close
certainlyClosed.set(true)
def isOpen = !certainlyClosed.get
def read(bb: ByteBuffer): Int =
if certainlyClosed.get then -1
else
if bb.hasArray then
val a = bb.array
val pos = bb.position
val avail = bb.remaining
val n = input.read(a, pos, avail)
if n > 0 then bb.position(pos + n)
if n < 0 then certainlyClosed.set(true)
n
else
var a = backingBuffer.get
if (a eq null) || (a.length < bb.remaining) then
a = new Array[Byte](bb.remaining max 16)
backingBuffer.set(a)
val n = input.read(a, 0, bb.remaining)
if n > 0 then bb.put(a, 0, n)
if n < 0 then certainlyClosed.set(true)
n
}
extension (is: InputStream)
inline def channel = new InputStreamByteChannel(is)
© 2015 - 2025 Weber Informatics LLC | Privacy Policy