Skip to content

The Apocrypha cipher takes a message and a key given by the user and proceeds to encrypt the message with the key by iteratively changing the key as it encrypts.

Notifications You must be signed in to change notification settings

sean-porras/Apocrypha

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

33 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Apocrypha

1. from ecclesiastical Latin "Apocrypha (scripta)" meaning "hidden (writings)".
2. the plane of Oblivion belonging to Hermaeus Mora, the Daedric Prince of Fate, Knowledge and Memory


Description

version stableversion documentation

The Apocrypha cipher takes a message and a key given by the user and proceeds to encrypt the message with the key by iteratively changing the key as it encrypts.

NOTE: For the best reading experience and to correctly view this document as intended, please view in an IntelliJ IDE or other Markdown viewer which displays similarly. There should be no random #s at the beginning of lines as appears on GitHub.



Requirements

  • Python 3.9+ (Haven't tested versions before 3.9)
  • blake3 Python Library (pip install blake3)


Usage

This guide will walkthrough a set of basic scenarios for encryption, decryption, and using the config modification subsystem. NOTE: This guide won't walk through EVERY scenario, but will explain inputs in each example run.

The | character will indicate where an input is requested from the user.


###How to encrypt with a custom message.

A_version: 1.9.4
A_func<E;D;C>: |

From here you can enter anything that starts with the letters e, d, or c as valid input options, in our case, we want to enter anything that begins with the letter e. Inputs here are NOT case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: e
eA_gen.type[<python{INT};apocrypha;custom{full;param};file;msg>]: |

Here is the encryption prompt for the key format to either be generated or used. In this case, we want to use a custom message, so we input msg. NOTE: Inputs here are NOT case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: e
eA_gen.type[<python{INT};apocrypha;custom{full;param};file;msg>]: msg
eA_key = |

Here you input what key you want to use for encrypting your message. This IS case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: e
eA_gen.type[<python{INT};apocrypha;custom{full;param};file;msg>]: msg
eA_key = This is a custom key
Message: |

Here is where you input the message you want to encrypt with your entered custom key. All characters on a standard en_US keyboard have been confirmed to work, and in theory, all Unicode characters should work as well, but this is not tested.


A_version: 1.9.4
A_func<E;D;C>: e
eA_gen.type[<python{INT};apocrypha;custom{full;param};file;msg>]: msg
eA_key = This is a custom key
Message: Super Seeecret Message!

Time: 0.005131959915161133

Encrypted Message:

[[685, 2910, 2985, 3999, 4195, 3460, 3522, 3629, 3724, 4595, 4916, 736, 687, 1001, 3472, 1758, 1618, 4823, 4127, 4337, 4770, 1697, -933, 3399], '22c8c32bd4ef6a7490f11607b9dd4e1f97a01089339d0eacdf27669da8450a72']

Press enter when done: |

Now you have the encrypted message as well as how long it took to encrypt the message printed out for you for copying and sending. The string of numbers is the encrypted message itself, so in a pinch, you can send just that (NOTE: MUST include the square brackets, eg [1, 2, 3]). The string of numbers and letters after the list of numbers is the final hash of the key after encryption used to guarantee during decryption that the message matches with the key.


###How to decrypt with a custom message

A_version: 1.9.4
A_func<E;D;C>: |

From here you can enter anything that starts with the letters e, d, or c as valid input options, in our case, we want to enter anything that begins with the letter d. Inputs here are NOT case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: d
dA_k.format[<link{full;param};file;msg>]: |

Here is the decryption prompt for the key format to either be generated or used. In this case, we want to use a custom message, so we input msg. NOTE: Inputs here are NOT case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: d
dA_k.format[<link{full;param};file;msg>]: msg
dA_key = |

Here you input what key you want to use for decrypting your message. This IS case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: d
dA_k.format[<link{full;param};file;msg>]: msg
dA_key = This is a custom key
dA_eM: |

Here is where you enter the encrypted message. It should, in a basic form, look something similar to [[1, 2, 3], '2uh75vh32ilu3v25bo873eljt325h1'], although much longer.


A_version: 1.9.4
A_func<E;D;C>: d
dA_k.format[<link{full;param};file;msg>]: msg
dA_key = This is a custom key
dA_eM: [[685, 2910, 2985, 3999, 4195, 3460, 3522, 3629, 3724, 4595, 4916, 736, 687, 1001, 3472, 1758, 1618, 4823, 4127, 4337, 4770, 1697, -933, 3399], '22c8c32bd4ef6a7490f11607b9dd4e1f97a01089339d0eacdf27669da8450a72']
Decrypted Message:

Super Seeecret Message!

Key Hash Match: True

Message Final Key Hash: 22c8c32bd4ef6a7490f11607b9dd4e1f97a01089339d0eacdf27669da8450a72
Current Final Key Hash: 22c8c32bd4ef6a7490f11607b9dd4e1f97a01089339d0eacdf27669da8450a72
Start Initial Key Hash: 6a1760f236570743b0f06a790bc93ee972972d8f22bbbbe36e6af94babadb9a2

Press enter to close the program. |

Here you have the results of your decryption, complete with what the decrypted message is, if the final key hashes matched, what they were, and what the initial key hash was.


####Using the config subsystem

A_version: 1.9.4
A_func<E;D;C>: |

From here you can enter anything that starts with the letters e, d, or c as valid input options, in our case, we want to enter anything that begins with the letter c. Inputs here are NOT case-sensitive.


A_version: 1.9.4
A_func<E;D;C>: c
NOTE: Only a 'config.json' file in the same directory as Apocrypha.py will be accepted currently
Config Handler Subsystem. Type 'help' for commands.
> |

The help command in the config handler subsystem acts as its own usage guide, as it's not recommended modifying config settings unless you really know what you're doing. (Note: Currently, as of {AD} 1.9.3, the config currently has no real functionality, this is planned to change, but all modifiable functionality previously has been integrated already.)


###Limited Command Line Functionality

You can bypass certain prompts using the command line to streamline the process of encryption and decryption, but currently, a prompted input is still required even given command line arguments.

Full command line functionality and support given any number of arguments is planned for the future along with a rewrite of the program as a whole to allow this functionality.


To get the usage message for the command line

python3 Apocrypha.py --help OR python3 Apocrypha.py --usage

Usage: python3 Apocrypha.py [--globals] [<E;D;C>] [<key gen/type>] [<key/filepath>] [<message>] [-f <filepath>]

NOTE: Non-global flags shown in the usage message (eg. [-f <filepath>]) haven't been implemented yet.

The program then runs the program according to the rest of the arguments given, otherwise runs the main program.


To begin the encryption process from eA_gen.type prompt

python3 Apocrypha.py e


To begin the decryption process from dA_k.format prompt

python3 Apocrypha.py d


To use a custom message in encryption, key pre-entered. If the key is a single word, quotations are not needed.

python3 Apocrypha.py e msg "This is a custom key"

You will be prompted for the Message: to encrypt.


To use a custom message in decryption, key pre-entered. If the key is a single word, quotations are not needed.

python3 Apocrypha.py d msg "This is a custom key"

You will be prompted with dA_eM: for the encrypted message to decrypt.


To use a custom file in encryption

python3 Apocrypha.py e file

You will be prompted for eA_filepath and continue from there.


To use a custom file in decryption, filepath given

python3 Apocrypha.py d file <filepath>

You will be prompted for the Encrypted Message: to decrypt.



Documentation

Class Objects

####Config

The Config class object is used to create an easy library of attributes. These dictate the functions of the Apocrypha program.


######CURRENTLY UNIMPLEMENTED: multi_in: default False. Dictates whether or not multiple runs of Apoc will be run on a single file containing commands for each separate result.


output: default 'print'. Dictates what form the output of Apoc will be given as. Apoc currently only supports printing the output. Future support for .txt, .json, and .apoc formats is planned.



Functions

####Aencode1(config: Config) -> str or None

Generates and returns a key to be used for encryption in Aencode2.

Parameter config: Config object which allows for config options to be utilized.

Returns: str (valid link) or None (indicative of txt, json, or apoc file)

The first step in the Apocrypha encryption process. Works to determine the key that'll be used, and create one if necessary. Takes user input to create either a link using the secrets module as the user has specified, the Library of Babel's built-in random function, or given specific parameters of the key to be used so it can be constructed and fetched. The file option is available which returns None and proceeds to the second part of encryption via the Aencode2 function. Otherwise the custom key option is available which returns the key entered by the user with a post-fixed ".msg" signifier to be identified by the Aencode2 function.


####Aencode2(config: Config, key: str or None) -> None

Given a key of either a valid link or a NoneType, processes to encrypt and print the output.

Parameter config: Config object which allows for config options to be utilized

Parameter key: str is indicative of a valid link, None is indicative of a file being used

Returns: None; Works to print out the result of encryption via printing to the console.

The second and final step in the Apocrypha encryption process. Gets the file location of the downloaded valid file to read and prepare to use as a key for the encryption. If the key format that gets passed is a link, it'll attempt to open it via the default system web browser, so it can be downloaded. It then moves on to get the user to input the filepath or the custom key for the program to utilize.

The program then prompts for the message the user wants to encrypt after validating the key. It then encrypts the message using the Apocrypha method: Find and compile a list of all indexes of characters matching the current character being encrypted of the message in the key. It then selects a random one of those characters and deletes it from the key file, which then changes the file to then repeat the process until the message is complete. If the character is not present in the key, then a random character is selected, and that key is used for its location. That character selected from the key's ord is then multiplied by the character the user is encrypting's ord value. It finally gets the hash of the final result of the key after encryption as a unique identifier of that encryption.

This method allows for the same key and the same message to have multiple encrypted versions while all decrypting to the same original message given the same initial key and message.


####Adecode1(config: Config) -> str

Helper function which starts the decryption process and prompts the user to return the path of a valid file path (.txt required in Adecode2).

Parameter config: Config object which allows for config options to be utilized.

Returns: str; File path to be processed by Adecode2

The first step in the Apocrypha decryption process. Given the user input of either a full link, link parameters, a file, or a custom key, gets the user to either open the link, constructs then opens the link, or simply input the filepath or key respectively. Finally gets the user to enter the location of the file either obtained, or previously had. Passes the str of the filepath or key to Adecode2 for processing and use for decryption.


####Adecode2(config: Config, fileloc: str) -> None

Given a valid file location, prompts for an encrypted message to decrypt and print the results.

Parameter config: Config object which allows for config options to be utilized.

Parameter fileloc: valid Path where the file is a .txt file

Returns: None; Final function using the Adecode1 helper function, prints to console.

The second and final step in the Apocrypha decryption process. Takes the filepath or key and validates it or creates it, respectively, then opens it and prepares it for the decryption process. Proceeds to prompt the user for the input of the encrypted message. It then formats the input to be able to process each character to find and then delete from the key in the same way the encryption process created it.


####config_handler(fileloc: str = '') -> Config Looks for an existing config file to read from and set up, otherwise creates a config with default values and returns the resulting configuration from the file.

Parameter fileloc: str; Valid Path of file named "config.json"

Returns: Config object

Function that is used for returning a Config object for the Apoc program to use if a "config.json" file exists, otherwise creates a "config.json" file in the parent directory with the default values as defined above.


####config_subsys(cf: dict) -> dict The subsystem in charge of taking in and modifying a dictionary according to user input. Constantly runs in the subsystem until the user enters the command to quit/exit.

Parameter cf: dict; Dictionary; This dictionary with config options is modified and returned

Returns: dict; Modified cf dictionary post-user processing.

The function that puts the user into a system where they can modify the options of the config file as they please. Called only by the config_handler function via entering c into the A_func prompt in main.


####expanding_hash(invar: str, length: int = 5000) -> str

Iteratively increases a given string into a hash containing characters specified below in the return docstring.

Parameter invar: String to be expanded into a hash used for Apocrypha method encryption as the key.

Parameter length: NonNegInteger; Defaults to 5000

Returns k; Final key, string, contains upper and lowercase letters, numbers, periods, numbers, spaces, and commas.

Expanding hash utilizes several hashing functions within it to create a unique and custom hash with a variable length. The function first goes over one iteration of the custom hashing function which then passes that hash to itself in a loop to complete the rest of the expansion of the hash.

It first creates a large number via taking the floored square root of the numbers present in the blake2b hash of the invar. That number is then used to grab the middle 2 numbers to use as the shifted position of the , in the final construction of the next segment of the expanded hash.

It then creates the first iteration of the hash which is the base64 encryption of the blake3 hash with +, /, and = replaced with ., , and , respectively. By default, due to how base64 encryption works, the = and thus the , is always at the end, which is where the middle two numbers randomly generated are used. It determines where the , is moved in the segments that were generated such that the , isn't a signifier for where each iteration starts and ends.

Finally, it creates a list to pass onto the loop portion of the function which will continue this process until the length parameter is met for the final key. The list passed onto the loop is the result of the first iteration but sliced into 4 equal length segments to then by hashed individually to repeat the process and expand the final key.


####cmd_ln(args: list) -> dict

Given a list of arguments from the command line, returns a dictionary of the program procedure and their values.

Parameter args: List of command line arguments sans "Apocrypha.py"

Returns dict: Dictionary containing prompts 'pre-filled' by command line arguments.

Processes a list of arguments which then builds a dictionary given each prompt of the program along with their values which is then later returned to be processed for functionality. If a given prompt/functional part of the program isn't used given a certain path the arguments follow, fields are by default None in the dictionary.

Contains an internal function err() which is an error handling function which can run the main program if any command line arguments throws an error either in their own right, or from the cmd_ln function in its own right.


####cmd_main(argvs: list) -> None

The main program for the command line.

Parameter argvs: List of command line arguments sans "Apocrypha.py"

Returns None: Calls other functions to carry out functionality as necessary.

The cmd_main function passes the list of arguments to the cmd_ln function which the returned dictionary is used for processing the command line arguments as to what to do functionally.


####main() -> None

The main function of the Apocrypha program which carries out the 3 primary functions of Apocrypha: Encrypt, Decrypt, Config.

Returns: None; Main function of the program which calls other functions to carry out the functionality of Apocrypha.

Prints the current version of the Apocrypha program and prompts the user for input. Valid inputs include: E, D, and C. Any string with these letters as the first character will be considered a valid input and execute the corresponding function.



About

The Apocrypha cipher takes a message and a key given by the user and proceeds to encrypt the message with the key by iteratively changing the key as it encrypts.

Resources

Stars

Watchers

Forks

Packages

No packages published

Languages