stream

package module
v0.0.0-...-578ce65 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: May 7, 2024 License: MIT Imports: 4 Imported by: 1

README

go-aead-iostream

IO stream for go std AEAD

Conn example

https://github.com/c0mm4nd/go-aead-conn

Example

package main

import (
	"crypto/aes"
	"crypto/cipher"
	"io"
	"os"
	"log"

	stream "github.com/c0mm4nd/go-aead-iostream"
)

func main() {
	// replace with your key
	seed := []byte("an arbitrary key")
	pathToNewFile := "/tmp/initial_file"
	pathToEncryptedFile := "/tmp/encrypted_file"
	pathToDecryptedFile := "/tmp/decrypted_file"

	err := encryptNewFile(seed, pathToNewFile, pathToEncryptedFile)
	if err != nil {
		panic(err)
	}

	err = decryptExistingFile(seed, pathToDecryptedFile, pathToEncryptedFile)
	if err != nil {
		panic(err)
	}
}

func encryptNewFile(seed []byte, pathToNewFile string, pathToEncryptedFile string) error {
	// Prepare ciphers.
	aesCipher, err := aes.NewCipher(seed)
	if err != nil {
		log.Print("Couldn't create AES cipher.")
		log.Printf("Error: %s", err)
		return err
	}
	aeadCipher, err := cipher.NewGCM(aesCipher)
	if err != nil {
		log.Print("Couldn't create AEAD cipher.")
		log.Printf("Error: %s", err)
		return err
	}

	// Create a new file and write some content into it.
	initialFile, err := os.OpenFile(pathToNewFile, os.O_CREATE|os.O_WRONLY, 0644)
	if err != nil {
		log.Printf("Couldn't create or open file. Path: %s", pathToNewFile)
		log.Printf("Error: %s", err)
		return err
	}
	defer initialFile.Close()
	rawMessage := []byte("Package cipher implements standard block cipher modes that can be wrapped around low-level block cipher implementations. See https://csrc.nist.gov/groups/ST/toolkit/BCM/current_modes.html and NIST Special Publication 800-38A.")
	initialFile.Write(rawMessage)
	initialFile.Close()

	// Reopen the new file for reading.
	initialFile, err = os.OpenFile(pathToNewFile, os.O_RDONLY, 0644)
	if err != nil {
		log.Printf("Couldn't open file for reading. Path: %s", pathToNewFile)
		log.Printf("Error: %s", err)
		return err
	}
	defer initialFile.Close()

	// Create an empty file, to which can be written later.
	encryptedFile, err := os.OpenFile(pathToEncryptedFile, os.O_CREATE|os.O_WRONLY, 0644)
	if err != nil {
		log.Printf("Couldn't create or open file. Path: %s", pathToEncryptedFile)
		log.Printf("Error: %s", err)
		return err
	}
	defer encryptedFile.Close()

	// Create the StreamWriteCloser, which can be piped into any output.
	chunkSize := 64
	encryptedWriter := stream.NewStreamWriteCloser(seed, chunkSize, encryptedFile, aeadCipher)
	defer encryptedWriter.Close()

	// Create a buffer to hold the data in the specified chunk size.
	buf := make([]byte, chunkSize)

	// Use io.CopyBuffer to read from the unencrypted file stream and write to the encrypted stream.
	if _, err := io.CopyBuffer(encryptedWriter, initialFile, buf); err != nil {
		log.Printf("Couldn't write encrypted file. Path: %s", pathToEncryptedFile)
		log.Printf("Error: %s", err)
		return err
	}

	return nil
}

func decryptExistingFile(seed []byte, pathToDecryptedFile string, pathToEncryptedFile string) error {
	// Prepare ciphers.
	block, err := aes.NewCipher(seed)
	if err != nil {
		log.Print("Couldn't create AES cipher.")
		log.Printf("Error: %s", err)
		return err
	}
	aead, err := cipher.NewGCM(block)
	if err != nil {
		log.Print("Couldn't create AEAD cipher.")
		log.Printf("Error: %s", err)
		return err
	}

	// Open the encrypted file for reading.
	encryptedFile, err := os.Open(pathToEncryptedFile)
	if err != nil {
		log.Printf("Couldn't create or open file. Path: %s", pathToEncryptedFile)
		log.Printf("Error: %s", err)
		return err
	}
	defer encryptedFile.Close()

	// Create a new file to write the decrypted data to.
	decryptedFile, err := os.Create(pathToDecryptedFile)
	if err != nil {
		log.Printf("Couldn't create or open file. Path: %s", pathToDecryptedFile)
		log.Printf("Error: %s", err)
		return err
	}
	defer decryptedFile.Close()

	// Create the StreamReader, which can be piped into any output.
	chunkSize := 64
	r := stream.NewStreamReader(seed, chunkSize, encryptedFile, aead)

	// Create a buffer to hold the data in the specified chunk size.
	buf := make([]byte, 64)

	// Use io.CopyBuffer to read from the encrypted file stream and write to the decrypted stream.
	if _, err := io.CopyBuffer(decryptedFile, r, buf); err != nil {
		log.Printf("Couldn't write encrypted file. Path: %s", pathToDecryptedFile)
		log.Printf("Error: %s", err)
		return err
	}

	return nil
}

Documentation

Index

Constants

This section is empty.

Variables

This section is empty.

Functions

This section is empty.

Types

type StreamReader

type StreamReader struct {
	// contains filtered or unexported fields
}

func NewStreamReader

func NewStreamReader(seed []byte, chunkSize int, backend io.Reader, aead cipher.AEAD) *StreamReader

func (*StreamReader) Read

func (r *StreamReader) Read(dst []byte) (n int, err error)

func (*StreamReader) ReadByte

func (r *StreamReader) ReadByte() (b byte, err error)

type StreamWriteCloser

type StreamWriteCloser struct {
	// contains filtered or unexported fields
}

func NewStreamWriteCloser

func NewStreamWriteCloser(seed []byte, chunkSize int, backend io.WriteCloser, aead cipher.AEAD) *StreamWriteCloser

func (*StreamWriteCloser) Close

func (w *StreamWriteCloser) Close() (err error)

func (*StreamWriteCloser) Write

func (w *StreamWriteCloser) Write(src []byte) (n int, err error)

func (*StreamWriteCloser) WriteByte

func (w *StreamWriteCloser) WriteByte(c byte) (err error)

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL