xipher

package module
v1.9.0 Latest Latest
Warning

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

Go to latest
Published: Jan 15, 2025 License: MIT Imports: 11 Imported by: 0

README

Xipher

Go Reference Go Report Card Test Status Release Status License

Xipher is a curated collection of cryptographic primitives put together to perform key/password based asymmetric encryption.

What does it do?

  • Allows sharing of data securely between two parties over an insecure channel using asymmetric encryption.
  • The sender encrypts the data using a public key (of the receiver - usually derived from a password) and shares the encrypted data with the receiver.
  • The receiver decrypts the data using the secret key (or password).

Key Aspects

  • Encrypts data with the public key generated based on a password.
  • Supports stream cipher along with stream compression, resulting in lower memory footprint and smaller ciphertext.
  • Supports post-quantum cryptography using the Kyber1024 algorithm.

CLI

Download the latest binary from the releases page and add it to your path.

Demo

Demo

Homebrew

Xipher can be installed with brew using the following command on macOS

brew install shibme/tap/xipher
Install Script
Install Latest Version

With Shell (MacOs/Linux):

curl -fsSL https://xipher.org/install/install.sh | sh

With PowerShell (Windows):

irm https://xipher.org/install/install.ps1 | iex
Install Specific Version

With Shell (MacOs/Linux):

curl -fsSL https://xipher.org/install/install.sh | sh -s v1.1.0

With PowerShell (Windows):

$v="1.1.0"; irm https://xipher.org/install/install.ps1 | iex
Docker

You can also run Xipher without installing using Docker:

docker run --rm -v $PWD:/data -it shibme/xipher help

Web Interface

A web interface interoperable with the CLI, implemented using web assembly is available here.

How does Xipher Web App work?
  • Receiver opens the Xipher web app on a browser.
  • Xipher generates a key pair and stores them in the browser local storage.
  • The Xiher web app returns the public key as a URL that can be shared.
  • Receiver shares the encryption URL (this contains the public key as a parameter) with the sender.
  • Sender opens the public encryption URL (opens Xipher encryption web page).
  • Sender inputs the data that needs to be encrypted.
  • Xipher encrypts the data using the public key from the URL.
  • Xipher returns ciphertext encrypted with the public key.
  • Sender sends the encrypted ciphertext to the receiver.
  • Receiver inputs the ciphertext in the decryption page.
  • Xipher decrypts the ciphertext using the secret key from local storage.
  • Xipher returns decrypted data.

The following sequence diagram illustrates the workflow of the web app.

sequenceDiagram
participant RX as Xipher<br>(on Browser)
actor Receiver
actor Sender
participant SX as Xipher<br>(on Browser)
    Receiver-->>+RX: Opens Xipher App on browser
    RX-->>RX: Generates a key pair and stores them in the browser local storage
    RX-->>-Receiver: Returns the Public Key<br>(as a URL that can be shared)
    Receiver->>+Sender: Shares the encryption URL<br>(this contains the public key as parameter)
    Sender-->>+SX: Opens the public encryption URL<br>(opens Xipher encryption web page)
    Sender-->>SX: Inputs the data that needs to be encrypted
    SX-->>SX: Encrypts the data using the public key from the URL
    SX-->>-Sender: Returns ciphertext encrypted with the Public Key
    Sender->>-Receiver: Sends the encrypted ciphertext to the Receiver
    Receiver-->>+RX: Inputs the ciphertext<br>(in the decyrption page)
    RX-->>RX: Decrypts the ciphertext<br>(using the secret key from local storage)
    RX-->>-Receiver: Returns decrypted data

Using as a Go package

Install the package

go get -u xipher.org/xipher

Use it in your code

package main

import (
	"encoding/base32"
	"fmt"

	"xipher.org/xipher"
)

func main() {
	// Creating a new secret key for password
	scrtKey, err := xipher.NewSecretKeyForPassword([]byte("Paws0meKittyKuwan!"))
	if err != nil {
		panic(err)
	}

	// Deriving  public key from secret key
	pubKey, err := scrtKey.PublicKey(false)
	if err != nil {
		panic(err)
	}
	publicKeyBytes, err := pubKey.Bytes()
	if err != nil {
		panic(err)
	}
	fmt.Println("PublicKey:", base32.StdEncoding.WithPadding(base32.NoPadding).EncodeToString(publicKeyBytes))

	platinText := []byte("Hello World!")

	// Encrypting plain text with public key
	cipherText, err := pubKey.Encrypt(platinText, true)
	if err != nil {
		panic(err)
	}
	fmt.Println("Encrypted:", base32.StdEncoding.WithPadding(base32.NoPadding).EncodeToString(cipherText))

	// Decrypting cipher text with secret key
	plainText, err := scrtKey.Decrypt(cipherText)
	if err != nil {
		panic(err)
	}
	fmt.Println("Decrypted:", string(plainText))
}

Web Assembly

To use xipher as a web assembly (wasm) module in a browser app, follow the example below.

<html>
	<head>
		<meta charset="utf-8"/>
		<script src="https://xipher.org/wasm/wasm_exec.js"></script>
		<script>
			const go = new Go();
			WebAssembly.instantiateStreaming(fetch("https://xipher.org/wasm/xipher.wasm"), go.importObject).then((result) => {
				go.run(result.instance);
			});
		</script>
	</head>
<body>
	Call wasm methods that begin with xipher. For example: xipherNewSecretKey()
</body>
</html>

Under the hood

Xipher uses the following algorithms and libraries to achieve its functionality:

Workflow

The following sequence diagram illustrates the workflow of Xipher in encrypting data using a password based public key.

sequenceDiagram
participant RX as Xipher
actor Receiver
actor Sender
participant SX as Xipher
    Receiver-->>+RX: Derive public (inputs password)
    RX-->>-Receiver: Returns Public Key
    Receiver->>Sender: Shares Public Key
    Sender-->>+SX: Encrypt data with public key
    SX-->>-Sender: Returns ciphertext encrypted with Public Key
    Sender->>Receiver: Sends the encrypted ciphertext to the Receiver
    Receiver-->>+RX: Decrypt data (inputs ciphertext and password)
    RX-->>-Receiver: Returns decrypted data

Disclaimer

This tool/library is provided without any warranties, and there is no guarantee of its stability. Due to the experimental nature of some of its components, it is anticipated that modifications to the code, repository, and API will be made in the future. Caution is advised before incorporating this into a production application. Please report any identified security issues promptly. Your cooperation in notifying us of such concerns is highly appreciated.

Documentation

Index

Constants

This section is empty.

Variables

This section is empty.

Functions

func VersionInfo

func VersionInfo() string

Types

type PublicKey

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

func ParsePublicKey

func ParsePublicKey(pubKeyBytes []byte) (*PublicKey, error)

ParsePublicKey parses the given bytes and returns a corresponding public key.

func (*PublicKey) Bytes

func (publicKey *PublicKey) Bytes() ([]byte, error)

Bytes returns the public key as bytes.

func (*PublicKey) Encrypt

func (publicKey *PublicKey) Encrypt(data []byte, compress bool) (ciphertext []byte, err error)

Encrypt encrypts data with the public key.

func (*PublicKey) EncryptStream

func (publicKey *PublicKey) EncryptStream(dst io.Writer, src io.Reader, compress bool) (err error)

EncryptStream encrypts src with the public key and writes to dst.

func (*PublicKey) NewEncryptingWriter

func (publicKey *PublicKey) NewEncryptingWriter(dst io.Writer, compress bool) (writer io.WriteCloser, err error)

type SecretKey

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

func NewSecretKey

func NewSecretKey() (*SecretKey, error)

NewSecretKey creates a new random private key.

func NewSecretKeyForPassword

func NewSecretKeyForPassword(password []byte) (*SecretKey, error)

NewSecretKeyForPassword creates a new private key for the given password.

func NewSecretKeyForPasswordAndSpec

func NewSecretKeyForPasswordAndSpec(password []byte, iterations, memory, threads uint8) (*SecretKey, error)

NewSecretKeyForPasswordAndSpec creates a new private key for the given password and kdf spec.

func ParseSecretKey

func ParseSecretKey(key []byte) (*SecretKey, error)

ParseSecretKey parses the given bytes and returns a corresponding private key.

func (*SecretKey) Bytes

func (secretKey *SecretKey) Bytes() ([]byte, error)

Bytes returns the private key as bytes only if it is not password based.

func (*SecretKey) Decrypt

func (secretKey *SecretKey) Decrypt(ciphertext []byte) (data []byte, err error)

Decrypt decrypts the given ciphertext and returns the decrypted data.

func (*SecretKey) DecryptStream

func (secretKey *SecretKey) DecryptStream(dst io.Writer, src io.Reader) (err error)

DecryptStream decrypts src and writes to dst.

func (*SecretKey) Encrypt

func (secretKey *SecretKey) Encrypt(data []byte, compress bool) (ciphertext []byte, err error)

Encrypt encrypts data with the secret key treating it as a symmetric key.

func (*SecretKey) EncryptStream

func (secretKey *SecretKey) EncryptStream(dst io.Writer, src io.Reader, compress bool) (err error)

EncryptStream encrypts src with the secret key treating it as a symmetric key and writes to dst.

func (*SecretKey) NewDecryptingReader

func (secretKey *SecretKey) NewDecryptingReader(src io.Reader) (io.Reader, error)

func (*SecretKey) NewEncryptingWriter

func (secretKey *SecretKey) NewEncryptingWriter(dst io.Writer, compress bool) (writer io.WriteCloser, err error)

func (*SecretKey) PublicKey

func (secretKey *SecretKey) PublicKey(pq bool) (*PublicKey, error)

PublicKey returns the public key corresponding to the private key.

Directories

Path Synopsis
crypto
asx
ecc
kyb
xcp
internal
cli

Jump to

Keyboard shortcuts

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