esp8266_deauther/esp8266_deauther/CLI.cpp

1280 lines
44 KiB
C++

/* This software is licensed under the MIT License: https://github.com/spacehuhntech/esp8266_deauther */
#include "CLI.h"
#include <LittleFS.h>
#include "settings.h"
#include "wifi.h"
/*
Shitty code used less resources so I will keep this clusterfuck as it is,
but if you're interested I made a library for this: github.com/spacehuhn/SimpleCLI
*/
CLI::CLI() {
list = new SimpleList<String>;
queue = new SimpleList<String>;
}
CLI::~CLI() {}
void CLI::load() {
String defaultValue = str(CLI_DEFAULT_AUTOSTART);
checkFile(execPath, defaultValue);
execFile(execPath);
}
void CLI::load(String filepath) {
execPath = filepath;
load();
}
void CLI::enable() {
enabled = true;
prntln(CLI_SERIAL_ENABLED);
}
void CLI::disable() {
enabled = true;
prntln(CLI_SERIAL_DISABLED);
}
void CLI::update() {
// when serial available, read input
if (Serial.available() > 0) {
String input = Serial.readStringUntil('\n');
exec(input);
}
// when queue is not empty, delay is off and no scan is active, run it
else if ((queue->size() > 0) && !delayed && !scan.isScanning() && !attack.isRunning()) {
String s = queue->shift();
exec(s);
}
}
void CLI::stop() {
queue->clear();
prntln(CLI_STOPPED_SCRIPT);
}
void CLI::enableDelay(uint32_t delayTime) {
delayed = true;
this->delayTime = delayTime;
delayStartTime = millis();
}
void CLI::exec(String input) {
// quick exit when input is empty
if (input.length() == 0) return;
// check delay
if (delayed && (millis() - delayStartTime > delayTime)) {
delayed = false;
prntln(CLI_RESUMED);
}
// when delay is on, add it to queue, else run it
if (delayed) {
queue->add(input);
} else {
runLine(input);
}
}
void CLI::execFile(String path) {
String input;
if (readFile(path, input)) {
String tmpLine;
char tmpChar;
input += '\n';
while (!queue->isEmpty()) {
input += queue->shift();
input += '\n';
}
for (int i = 0; i < input.length(); i++) {
tmpChar = input.charAt(i);
if (tmpChar == '\n') {
queue->add(tmpLine);
tmpLine = String();
} else {
tmpLine += tmpChar;
}
}
queue->add(tmpLine);
}
}
void CLI::error(String message) {
prnt(CLI_ERROR);
prntln(message);
}
void CLI::parameterError(String parameter) {
prnt(CLI_ERROR_PARAMETER);
prnt(parameter);
prntln(DOUBLEQUOTES);
}
bool CLI::isInt(String str) {
if (eqls(str, STR_TRUE) || eqls(str, STR_FALSE)) return true;
for (uint32_t i = 0; i < str.length(); i++)
if (!isDigit(str.charAt(i))) return false;
return true;
}
int CLI::toInt(String str) {
if (eqls(str, STR_TRUE)) return 1;
else if (eqls(str, STR_FALSE)) return 0;
else return str.toInt();
}
uint32_t CLI::getTime(String time) {
int value = time.toInt();
if (value < 0) value = -value;
if (time.substring(time.length() - 1).equalsIgnoreCase(String(S))) value *= 1000;
else if (time.substring(time.length() - 3).equalsIgnoreCase(str(STR_MIN)) ||
(time.charAt(time.length() - 1) == M)) value *= 60000;
return value;
}
bool CLI::eqlsCMD(int i, const char* keyword) {
return eqls(list->get(i).c_str(), keyword);
}
void CLI::runLine(String input) {
String tmp;
for (int i = 0; i < input.length(); i++) {
// when 2 semicolons in a row without a backslash escaping the first
if ((input.charAt(i) == SEMICOLON) && (input.charAt(i + 1) == SEMICOLON) &&
(input.charAt(i - 1) != BACKSLASH)) {
runCommand(tmp);
tmp = String();
i++;
} else {
tmp += input.charAt(i);
}
}
tmp.replace(BACKSLASH + SEMICOLON + SEMICOLON, SEMICOLON + SEMICOLON);
if (tmp.length() > 0) runCommand(tmp);
}
void CLI::runCommand(String input) {
input.replace(String(NEWLINE), String());
input.replace(String(CARRIAGERETURN), String());
list->clear();
// parse/split input in list
String tmp;
bool withinQuotes = false;
bool escaped = false;
char c;
for (uint32_t i = 0; i < input.length() && i < 512; i++) {
c = input.charAt(i);
// when char is an unescaped
if (!escaped && (c == BACKSLASH)) {
escaped = true;
}
// (when char is a unescaped space AND it's not within quotes) OR char is \r or \n
else if (((c == SPACE) && !escaped && !withinQuotes) || (c == CARRIAGERETURN) || (c == NEWLINE)) {
// when tmp string isn't empty, add it to the list
if (tmp.length() > 0) {
list->add(tmp);
tmp = String(); // reset tmp string
}
}
// when char is an unescaped "
else if ((c == DOUBLEQUOTES) && !escaped) {
// update wheter or not the following chars are within quotes or not
withinQuotes = !withinQuotes;
if ((tmp.length() == 0) && !withinQuotes) tmp += SPACE; // when exiting quotes and tmp string is empty, add
// a space
}
// add character to tmp string
else {
tmp += c;
escaped = false;
}
}
// add string if something is left from the loop above
if (tmp.length() > 0) list->add(tmp);
// stop when input is empty/invalid
if (list->size() == 0) return;
// print comments
if (list->get(0) == str(CLI_COMMENT)) {
prntln(input);
return;
}
if (settings::getCLISettings().serial_echo) {
// print command
prnt(CLI_INPUT_PREFIX);
prntln(input);
}
if (list->size() == 0) return;
// ===== HELP ===== //
if (eqlsCMD(0, CLI_HELP)) {
prntln(CLI_HELP_HEADER);
prntln(CLI_HELP_HELP);
prntln(CLI_HELP_SCAN);
prntln(CLI_HELP_SHOW);
prntln(CLI_HELP_SELECT);
prntln(CLI_HELP_DESELECT);
prntln(CLI_HELP_SSID_A);
prntln(CLI_HELP_SSID_B);
prntln(CLI_HELP_SSID_C);
prntln(CLI_HELP_NAME_A);
prntln(CLI_HELP_NAME_B);
prntln(CLI_HELP_NAME_C);
prntln(CLI_HELP_SET_NAME);
prntln(CLI_HELP_ENABLE_RANDOM);
prntln(CLI_HELP_DISABLE_RANDOM);
prntln(CLI_HELP_LOAD);
prntln(CLI_HELP_SAVE);
prntln(CLI_HELP_REMOVE_A);
prntln(CLI_HELP_REMOVE_B);
prntln(CLI_HELP_ATTACK);
prntln(CLI_HELP_ATTACK_STATUS);
prntln(CLI_HELP_STOP);
prntln(CLI_HELP_SYSINFO);
prntln(CLI_HELP_CLEAR);
prntln(CLI_HELP_FORMAT);
prntln(CLI_HELP_PRINT);
prntln(CLI_HELP_DELETE);
prntln(CLI_HELP_REPLACE);
prntln(CLI_HELP_COPY);
prntln(CLI_HELP_RENAME);
prntln(CLI_HELP_RUN);
prntln(CLI_HELP_WRITE);
prntln(CLI_HELP_GET);
prntln(CLI_HELP_SET);
prntln(CLI_HELP_RESET);
prntln(CLI_HELP_CHICKEN);
prntln(CLI_HELP_REBOOT);
prntln(CLI_HELP_INFO);
prntln(CLI_HELP_COMMENT);
prntln(CLI_HELP_SEND_DEAUTH);
prntln(CLI_HELP_SEND_BEACON);
prntln(CLI_HELP_SEND_PROBE);
prntln(CLI_HELP_LED_A);
prntln(CLI_HELP_LED_B);
prntln(CLI_HELP_DRAW);
prntln(CLI_HELP_SCREEN_ON);
prntln(CLI_HELP_SCREEN_MODE);
prntln(CLI_HELP_FOOTER);
}
// ===== SCAN ===== //
// scan [<mode>] [-t <time>] [-c <continue-time>] [-ch <channel>]
else if (eqlsCMD(0, CLI_SCAN)) {
uint8_t scanMode = SCAN_MODE_ALL;
uint8_t nextmode = SCAN_MODE_OFF;
uint8_t channel = wifi_channel;
bool channelHop = true;
uint32_t time = 15000;
uint32_t continueTime = 10000;
for (int i = 1; i < list->size(); i++) {
if (eqlsCMD(i, CLI_AP)) scanMode = SCAN_MODE_APS;
else if (eqlsCMD(i, CLI_STATION)) scanMode = SCAN_MODE_STATIONS;
else if (eqlsCMD(i, CLI_ALL)) scanMode = SCAN_MODE_ALL;
else if (eqlsCMD(i, CLI_WIFI)) scanMode = SCAN_MODE_SNIFFER;
else if (eqlsCMD(i, CLI_TIME)) {
i++;
time = getTime(list->get(i));
} else if (eqlsCMD(i, CLI_CONTINUE)) {
i++;
nextmode = scanMode;
continueTime = getTime(list->get(i));
} else if (eqlsCMD(i, CLI_CHANNEL)) {
i++;
if (!eqlsCMD(i, CLI_ALL)) {
channelHop = false;
channel = list->get(i).toInt();
}
} else {
parameterError(list->get(i));
}
}
scan.start(scanMode, time, nextmode, continueTime, channelHop, channel);
}
// ===== SHOW ===== //
else if (eqlsCMD(0, CLI_SHOW)) {
// show selected [<all/aps/stations/names/ssids>]
if (eqlsCMD(1, CLI_SELECT)) {
if (list->size() > 2) {
for (int i = 2; i < list->size(); i++) {
if (eqlsCMD(i, CLI_AP)) accesspoints.printSelected();
else if (eqlsCMD(i, CLI_STATION)) stations.printSelected();
else if (eqlsCMD(i, CLI_NAME)) names.printSelected();
else if (eqlsCMD(i, CLI_ALL)) scan.printSelected();
else parameterError(list->get(i));
}
} else {
scan.printSelected();
}
}
// show [<all/aps/stations/names/ssids>]
else {
if (list->size() > 1) {
for (int i = 1; i < list->size(); i++) {
if (eqlsCMD(i, CLI_AP)) accesspoints.printAll();
else if (eqlsCMD(i, CLI_STATION)) stations.printAll();
else if (eqlsCMD(i, CLI_NAME)) names.printAll();
else if (eqlsCMD(i, CLI_SSID)) ssids.printAll();
else if (eqlsCMD(i, CLI_ALL)) scan.printAll();
else parameterError(list->get(i));
}
} else {
scan.printAll();
}
}
}
// ===== (DE)SELECT ===== //
// select [<type>] [<id>]
// deselect [<type>] [<id>]
else if (eqlsCMD(0, CLI_SELECT) || eqlsCMD(0, CLI_DESELECT)) {
bool select = eqlsCMD(0, CLI_SELECT);
int mode = 0; // aps = 0, stations = 1, names = 2
int id = -1; // -1 = all, -2 name string
if ((list->size() == 1) || eqlsCMD(1, CLI_ALL)) {
select ? scan.selectAll() : scan.deselectAll();
return;
}
if ((list->size() == 2) || eqlsCMD(2, CLI_ALL)) id = -1;
else if (!isInt(list->get(2))) id = -2;
else id = list->get(2).toInt();
if (eqlsCMD(1, CLI_AP)) mode = 0;
else if (eqlsCMD(1, CLI_STATION)) mode = 1;
else if (eqlsCMD(1, CLI_NAME)) mode = 2;
else parameterError(list->get(1));
if (id >= 0) {
if (mode == 0) select ? accesspoints.select(id) : accesspoints.deselect(id);
else if (mode == 1) select ? stations.select(id) : stations.deselect(id);
else if (mode == 2) select ? names.select(id) : names.deselect(id);
} else if (id == -1) {
if (mode == 0) select ? accesspoints.selectAll() : accesspoints.deselectAll();
else if (mode == 1) select ? stations.selectAll() : stations.deselectAll();
else if (mode == 2) select ? names.selectAll() : names.deselectAll();
} else if ((id == -2)) {
String name = list->get(2);
if (mode == 0) select ? accesspoints.select(name) : accesspoints.deselect(name);
else if (mode == 1) select ? stations.select(name) : stations.deselect(name);
else if (mode == 2) select ? names.select(name) : names.deselect(name);
} else {
parameterError(list->get(1) + SPACE + list->get(2));
}
}
// ===== ADD ===== //
else if ((list->size() >= 3) && eqlsCMD(0, CLI_ADD) && eqlsCMD(1, CLI_SSID)) {
// add ssid -s [-f]
if (eqlsCMD(2, CLI_SELECT)) {
bool force = eqlsCMD(3, CLI_FORCE);
ssids.cloneSelected(force);
}
// add ssid <ssid> [-wpa2] [-cl <clones>] [-f]
// add ssid -ap <id> [-cl <clones>] [-f]
else {
String ssid = list->get(2);
bool wpa2 = false;
bool force = false;
int clones = 1;
int i = 3;
if (eqlsCMD(2, CLI_AP)) {
ssid = accesspoints.getSSID(list->get(3).toInt());
wpa2 = accesspoints.getEncStr(list->get(3).toInt()) != " - ";
i = 4;
}
while (i < list->size()) {
if (eqlsCMD(i, CLI_WPA2)) wpa2 = true;
else if (eqlsCMD(i, CLI_FORCE)) force = true;
else if (eqlsCMD(i, CLI_CLONES)) {
clones = list->get(i + 1).toInt();
i++;
} else parameterError(list->get(i));
i++;
}
ssids.add(ssid, wpa2, clones, force);
}
}
// add name <name> [-ap <id>] [-s] [-f]
// add name <name> [-st <id>] [-s] [-f]
// add name <name> [-m <mac>] [-ch <channel>] [-b <bssid>] [-s] [-f]
else if ((list->size() >= 3) && eqlsCMD(0, CLI_ADD) && eqlsCMD(1, CLI_NAME)) {
String name = list->get(2);
String mac;
uint8_t channel = wifi_channel;
String bssid;
bool selected = false;
bool force = false;
for (int i = 3; i < list->size(); i++) {
if (eqlsCMD(i, CLI_MAC)) mac = list->get(i + 1);
else if (eqlsCMD(i, CLI_AP)) mac = accesspoints.getMacStr(list->get(i + 1).toInt());
else if (eqlsCMD(i, CLI_STATION)) {
mac = stations.getMacStr(list->get(i + 1).toInt());
bssid = stations.getAPMacStr(list->get(i + 1).toInt());
}
else if (eqlsCMD(i, CLI_CHANNEL)) channel = (uint8_t)list->get(i + 1).toInt();
else if (eqlsCMD(i, CLI_BSSID)) bssid = list->get(i + 1);
else if (eqlsCMD(i, CLI_SELECT)) {
selected = true;
i--;
} else if (eqlsCMD(i, CLI_FORCE)) {
force = true;
i--;
} else {
parameterError(list->get(i));
i--;
}
i++;
}
if (name.length() == 0) prntln(CLI_ERROR_NAME_LEN);
else if (mac.length() == 0) prntln(CLI_ERROR_MAC_LEN);
else names.add(mac, name, bssid, channel, selected, force);
}
// ===== SET NAME ==== //
// set name <id> <newname>
else if ((list->size() == 4) && eqlsCMD(0, CLI_SET) && eqlsCMD(1, CLI_NAME)) {
names.setName(list->get(2).toInt(), list->get(3));
}
// ===== REPLACE ===== //
// replace name <id> [-n <name>} [-m <mac>] [-ch <channel>] [-b <bssid>] [-s]
else if ((list->size() >= 4) && eqlsCMD(0, CLI_REPLACE) && eqlsCMD(1, CLI_NAME)) {
int id = list->get(2).toInt();
String name = names.getName(id);
String mac = names.getMacStr(id);
uint8_t channel = names.getCh(id);
String bssid = names.getBssidStr(id);
bool selected = names.getSelected(id);
for (int i = 3; i < list->size(); i++) {
if (eqlsCMD(i, CLI_NAME)) name = list->get(i + 1);
else if (eqlsCMD(i, CLI_MAC)) mac = list->get(i + 1);
else if (eqlsCMD(i, CLI_CHANNEL)) channel = (uint8_t)list->get(i + 1).toInt();
else if (eqlsCMD(i, CLI_BSSID)) bssid = list->get(i + 1);
else if (eqlsCMD(i, CLI_SELECT)) {
selected = true;
i--;
} else {
parameterError(list->get(i));
i--;
}
i++;
}
names.replace(id, mac, name, bssid, channel, selected);
}
// replace ssid <id> [-n <name>} [-wpa2]
else if ((list->size() >= 3) && eqlsCMD(0, CLI_REPLACE) && eqlsCMD(1, CLI_SSID)) {
int id = list->get(2).toInt();
String name = ssids.getName(id);
bool wpa2 = false;
for (int i = 3; i < list->size(); i++) {
if (eqlsCMD(i, CLI_NAME)) {
name = list->get(i + 1);
i++;
} else if (eqlsCMD(i, CLI_WPA2)) {
wpa2 = true;
}
}
ssids.replace(id, name, wpa2);
}
// ===== REMOVE ===== //
// remove <type> [-a]
// remove <type> <id>
else if ((list->size() >= 2) && eqlsCMD(0, CLI_REMOVE)) {
if ((list->size() == 2) || (eqlsCMD(2, CLI_ALL))) {
if (eqlsCMD(1, CLI_SSID)) ssids.removeAll();
else if (eqlsCMD(1, CLI_NAME)) names.removeAll();
else if (eqlsCMD(1, CLI_AP)) accesspoints.removeAll();
else if (eqlsCMD(1, CLI_STATION)) stations.removeAll();
else parameterError(list->get(1));
} else {
if (eqlsCMD(1, CLI_SSID)) ssids.remove(list->get(2).toInt());
else if (eqlsCMD(1, CLI_NAME)) names.remove(list->get(2).toInt());
else if (eqlsCMD(1, CLI_AP)) accesspoints.remove(list->get(2).toInt());
else if (eqlsCMD(1, CLI_STATION)) stations.remove(list->get(2).toInt());
else parameterError(list->get(1));
}
}
// ===== RANDOM ===== //
// enable random <interval>
else if (eqlsCMD(0, CLI_ENABLE) && eqlsCMD(1, CLI_RANDOM) && (list->size() == 3)) {
ssids.enableRandom(getTime(list->get(2)));
}
// disable random
else if (eqlsCMD(0, CLI_DISABLE) && eqlsCMD(1, CLI_RANDOM)) {
ssids.disableRandom();
}
// ====== RICE ===== //
// => have you tried putting it into a bowl rice?
// funny command to mess with people, please don't share the info thanks <3
else if (eqlsCMD(0, CLI_RICE)) {
prntln(CLI_RICE_START);
uint32_t i = 0;
uint8_t multi = 1;
uint8_t end = random(80, 99);
if (list->get(1).toInt() > 1) multi = list->get(1).toInt();
while (true) {
if ((i % 10 == 0) && (i > 0)) {
char s[100];
sprintf(s, str(CLI_RICE_OUTPUT).c_str(), i / 10);
prnt(String(s));
if ((i / 10) == end) {
prnt(CLI_RICE_ERROR);
prnt(String(random(16, 255), HEX));
prnt(String(random(16, 255), HEX));
prnt(String(random(16, 255), HEX));
prntln(String(random(16, 255), HEX));
for (int i = 1; i <= 32; i++) {
for (int i = 0; i < 2; i++) {
for (int i = 1; i <= 8; i++) {
prnt(String(random(16, 255), HEX));
prnt(SPACE);
}
prnt(SPACE);
}
prntln();
}
ESP.reset();
} else if ((i / 10) % 10 == 0) {
prnt(CLI_RICE_MEM);
prnt(String(random(16, 255), HEX));
prnt(String(random(16, 255), HEX));
prnt(String(random(16, 255), HEX));
prntln(String(random(16, 255), HEX));
}
}
prnt(POINT);
delay(100 * multi * multi);
i++;
}
}
// ===== LOAD/SAVE ===== //
// save [<type>] [<file>]
// load [<type>] [<file>]
else if ((eqlsCMD(0, CLI_LOAD) || eqlsCMD(0, CLI_SAVE)) && (list->size() >= 1) && (list->size() <= 3)) {
bool load = eqlsCMD(0, CLI_LOAD);
if ((list->size() == 1) || eqlsCMD(1, CLI_ALL)) {
load ? ssids.load() : ssids.save(false);
load ? names.load() : names.save(false);
load ? settings::load() : settings::save(false);
if (!load) scan.save(false);
return;
}
if (list->size() == 3) { // Todo: check if -f or filename
if (eqlsCMD(1, CLI_SSID)) load ? ssids.load(list->get(2)) : ssids.save(true, list->get(2));
else if (eqlsCMD(1, CLI_NAME)) load ? names.load(list->get(2)) : names.save(true, list->get(2));
// else if (eqlsCMD(1, CLI_SETTING)) load ? settings::load(list->get(2)) : settings::save(true, list->get(2));
else parameterError(list->get(1));
} else {
if (eqlsCMD(1, CLI_SSID)) load ? ssids.load() : ssids.save(true);
else if (eqlsCMD(1, CLI_NAME)) load ? names.load() : names.save(true);
else if (eqlsCMD(1, CLI_SETTING)) load ? settings::load() : settings::save(true);
else if ((eqlsCMD(1, CLI_SCAN) || eqlsCMD(1, CLI_AP) || eqlsCMD(1, CLI_STATION)) && !load) scan.save(true);
else parameterError(list->get(1));
}
}
// ===== ATTACK ===== //
// attack [-b] [-d] [-da] [p] [-t <timeout>]
// attack status [<on/off>]
else if (eqlsCMD(0, CLI_ATTACK)) {
if (eqlsCMD(1, CLI_STATUS)) {
if (list->size() == 2) {
attack.status();
} else {
if (eqlsCMD(2, CLI_ON)) attack.enableOutput();
else if (eqlsCMD(2, CLI_OFF)) attack.disableOutput();
else parameterError(list->get(2));
}
return;
}
bool beacon = false;
bool deauth = false;
bool deauthAll = false;
bool probe = false;
bool output = true;
uint32_t timeout = settings::getAttackSettings().timeout * 1000;
for (int i = 1; i < list->size(); i++) {
if (eqlsCMD(i, CLI_BEACON)) beacon = true;
else if (eqlsCMD(i, CLI_DEAUTH)) deauth = true;
else if (eqlsCMD(i, CLI_DEAUTHALL)) deauthAll = true;
else if (eqlsCMD(i, CLI_PROBE)) probe = true;
else if (eqlsCMD(i, CLI_NOOUTPUT)) output = false;
else if (eqlsCMD(i, CLI_TIMEOUT)) {
timeout = getTime(list->get(i + 1));
i++;
}
else parameterError(list->get(i));
}
attack.start(beacon, deauth, deauthAll, probe, output, timeout);
}
// ===== GET/SET ===== //
// get <setting>
else if (eqlsCMD(0, CLI_GET) /*&& (list->size() == 2)*/) {
String _tmp = list->get(1);
const char* str = _tmp.c_str();
if (eqls(str, "settings")) settings::print();
// Version
else if (eqls(str, S_JSON_VERSION)) prntln(DEAUTHER_VERSION);
else if (eqls(str, S_JSON_AUTOSAVE)) prntln(settings::getAutosaveSettings().enabled);
else if (eqls(str, S_JSON_AUTOSAVETIME)) prntln(settings::getAutosaveSettings().time);
// Attack
else if (eqls(str, S_JSON_BEACONCHANNEL)) prntln((int)settings::getAttackSettings().attack_all_ch);
else if (eqls(str, S_JSON_RANDOMTX)) prntln(settings::getAttackSettings().random_tx);
else if (eqls(str, S_JSON_ATTACKTIMEOUT)) prntln(settings::getAttackSettings().timeout);
else if (eqls(str, S_JSON_DEAUTHSPERTARGET)) prntln(settings::getAttackSettings().deauths_per_target);
else if (eqls(str, S_JSON_DEAUTHREASON)) prntln(settings::getAttackSettings().deauth_reason);
else if (eqls(str, S_JSON_BEACONINTERVAL)) prntln((bool)settings::getAttackSettings().beacon_interval);
else if (eqls(str, S_JSON_PROBESPERSSID)) prntln(settings::getAttackSettings().probe_frames_per_ssid);
// WiFi
else if (eqls(str, S_JSON_CHANNEL)) prntln(settings::getWifiSettings().channel);
else if (eqls(str, S_JSON_MACST)) prntln(macToStr(settings::getWifiSettings().mac_st));
else if (eqls(str, S_JSON_MACAP)) prntln(macToStr(settings::getWifiSettings().mac_ap));
// Sniffer
else if (eqls(str, S_JSON_CHTIME)) prntln(settings::getSnifferSettings().channel_time);
else if (eqls(str, S_JSON_MIN_DEAUTHS)) prntln(settings::getSnifferSettings().min_deauth_frames);
// AP
else if (eqls(str, S_JSON_SSID)) prntln(settings::getAccessPointSettings().ssid);
else if (eqls(str, S_JSON_PASSWORD)) prntln(settings::getAccessPointSettings().password);
else if (eqls(str, S_JSON_HIDDEN)) prntln(settings::getAccessPointSettings().hidden);
else if (eqls(str, S_JSON_IP)) prntln(settings::getAccessPointSettings().ip);
// Web
else if (eqls(str, S_JSON_WEBINTERFACE)) prntln(settings::getWebSettings().enabled);
else if (eqls(str, S_JSON_CAPTIVEPORTAL)) prntln(settings::getWebSettings().captive_portal);
else if (eqls(str, S_JSON_WEB_SPIFFS)) prntln(settings::getWebSettings().use_spiffs);
else if (eqls(str, S_JSON_LANG)) prntln(settings::getWebSettings().lang, 3);
// CLI
else if (eqls(str, S_JSON_SERIALINTERFACE)) prntln(settings::getCLISettings().enabled);
else if (eqls(str, S_JSON_SERIAL_ECHO)) prntln(settings::getCLISettings().serial_echo);
// LED
else if (eqls(str, S_JSON_LEDENABLED)) prntln(settings::getLEDSettings().enabled);
// Display
else if (eqls(str, S_JSON_DISPLAYINTERFACE)) prntln(settings::getDisplaySettings().enabled);
else if (eqls(str, S_JSON_DISPLAY_TIMEOUT)) prntln(settings::getDisplaySettings().timeout);
else {
prnt(_tmp);
prntln(" setting not found");
}
}
// set <setting> <value>
else if (eqlsCMD(0, CLI_SET) && (list->size() == 3)) {
String _tmp = list->get(1);
const char* str = _tmp.c_str();
String strVal = list->get(2);
bool boolVal = s2b(strVal);
int intVal = strVal.toInt();
uint32_t unsignedVal = intVal < 0 ? 0 : (uint32_t)intVal;
settings_t newSettings = settings::getAllSettings();
// Autosave
if (eqls(str, S_JSON_AUTOSAVE)) newSettings.autosave.enabled = boolVal;
else if (eqls(str, S_JSON_AUTOSAVETIME)) newSettings.autosave.time = unsignedVal;
// Attack
else if (eqls(str, S_JSON_BEACONCHANNEL)) newSettings.attack.attack_all_ch = boolVal;
else if (eqls(str, S_JSON_RANDOMTX)) newSettings.attack.random_tx = boolVal;
else if (eqls(str, S_JSON_ATTACKTIMEOUT)) newSettings.attack.timeout = unsignedVal;
else if (eqls(str, S_JSON_DEAUTHSPERTARGET)) newSettings.attack.deauths_per_target = unsignedVal;
else if (eqls(str, S_JSON_DEAUTHREASON)) newSettings.attack.deauth_reason = unsignedVal;
else if (eqls(str, S_JSON_BEACONINTERVAL)) newSettings.attack.beacon_interval = (beacon_interval_t)boolVal;
else if (eqls(str, S_JSON_PROBESPERSSID)) newSettings.attack.probe_frames_per_ssid = unsignedVal;
// WiFi
else if (eqls(str, S_JSON_CHANNEL)) newSettings.wifi.channel = unsignedVal;
else if (eqls(str, S_JSON_MACST)) strToMac(strVal, newSettings.wifi.mac_st);
else if (eqls(str, S_JSON_MACAP)) strToMac(strVal, newSettings.wifi.mac_ap);
// Sniffer
else if (eqls(str, S_JSON_CHTIME)) newSettings.sniffer.channel_time = unsignedVal;
else if (eqls(str, S_JSON_MIN_DEAUTHS)) newSettings.sniffer.min_deauth_frames = unsignedVal;
// AP
else if (eqls(str, S_JSON_SSID)) strncpy(newSettings.ap.ssid, strVal.c_str(), 32);
else if (eqls(str, S_JSON_PASSWORD)) strncpy(newSettings.ap.password, strVal.c_str(), 64);
else if (eqls(str, S_JSON_HIDDEN)) newSettings.ap.hidden = boolVal;
else if (eqls(str, S_JSON_IP)) strToIP(strVal, newSettings.ap.ip);
// Web
else if (eqls(str, S_JSON_WEBINTERFACE)) newSettings.web.enabled = boolVal;
else if (eqls(str, S_JSON_CAPTIVEPORTAL)) newSettings.web.captive_portal = boolVal;
else if (eqls(str, S_JSON_WEB_SPIFFS)) newSettings.web.use_spiffs = boolVal;
else if (eqls(str, S_JSON_LANG)) strncpy(newSettings.web.lang, strVal.c_str(), 3);
// CLI
else if (eqls(str, S_JSON_SERIALINTERFACE)) newSettings.cli.enabled = boolVal;
else if (eqls(str, S_JSON_SERIAL_ECHO)) newSettings.cli.serial_echo = boolVal;
// LED
else if (eqls(str, S_JSON_LEDENABLED)) newSettings.led.enabled = boolVal;
// Display
else if (eqls(str, S_JSON_DISPLAYINTERFACE)) newSettings.display.enabled = boolVal;
else if (eqls(str, S_JSON_DISPLAY_TIMEOUT)) newSettings.display.timeout = unsignedVal;
else {
prnt(str);
prntln(" not found");
return;
}
prnt("Set ");
prnt(str);
prnt(" = ");
prntln(strVal);
settings::setAllSettings(newSettings);
}
// ====== CHICKEN ===== //
else if (eqlsCMD(0, CLI_CHICKEN)) {
prntln(CLI_CHICKEN_OUTPUT);
}
// ===== STOP ===== //
// stop [<mode>]
else if (eqlsCMD(0, CLI_STOP)) {
led::setMode(IDLE, true);
if ((list->size() >= 2) && !(eqlsCMD(1, CLI_ALL))) {
for (int i = 1; i < list->size(); i++) {
if (eqlsCMD(i, CLI_SCAN)) scan.stop();
else if (eqlsCMD(i, CLI_ATTACK)) attack.stop();
else if (eqlsCMD(i, CLI_SCRIPT)) this->stop();
else parameterError(list->get(i));
}
} else {
scan.stop();
attack.stop();
this->stop();
}
}
// ===== SYSTEM ===== //
// sysinfo
else if (eqlsCMD(0, CLI_SYSINFO)) {
prntln(CLI_SYSTEM_INFO);
char s[150];
sprintf(s, str(CLI_SYSTEM_OUTPUT).c_str(), 81920 - system_get_free_heap_size(),
100 - system_get_free_heap_size() / (81920 / 100), system_get_free_heap_size(),
system_get_free_heap_size() / (81920 / 100), 81920);
prntln(String(s));
prnt(CLI_SYSTEM_CHANNEL);
prntln(settings::getWifiSettings().channel);
uint8_t mac[6];
prnt(CLI_SYSTEM_AP_MAC);
wifi_get_macaddr(SOFTAP_IF, mac);
prntln(macToStr(mac));
prnt(CLI_SYSTEM_ST_MAC);
wifi_get_macaddr(STATION_IF, mac);
prntln(macToStr(mac));
FSInfo fs_info;
LittleFS.info(fs_info);
sprintf(s, str(
CLI_SYSTEM_RAM_OUT).c_str(), fs_info.usedBytes, fs_info.usedBytes / (fs_info.totalBytes / 100), fs_info.totalBytes - fs_info.usedBytes,
(fs_info.totalBytes - fs_info.usedBytes) / (fs_info.totalBytes / 100), fs_info.totalBytes);
prnt(String(s));
sprintf(s, str(CLI_SYSTEM_SPIFFS_OUT).c_str(), fs_info.blockSize, fs_info.pageSize);
prnt(String(s));
prntln(CLI_FILES);
Dir dir = LittleFS.openDir(String(SLASH));
while (dir.next()) {
prnt(String(SPACE) + String(SPACE) + dir.fileName() + String(SPACE));
File f = dir.openFile("r");
prnt(int(f.size()));
prntln(str(CLI_BYTES));
}
wifi::printStatus();
prntln(CLI_SYSTEM_FOOTER);
}
// ===== RESET ===== //
// reset
else if (eqlsCMD(0, CLI_RESET)) {
settings::reset();
}
// ===== CLEAR ===== //
// clear
else if (eqlsCMD(0, CLI_CLEAR)) {
for (int i = 0; i < 100; i++) prnt(HASHSIGN);
for (int i = 0; i < 60; i++) prntln();
}
// ===== REBOOT ===== //
// reboot
else if (eqlsCMD(0, CLI_REBOOT)) {
ESP.reset();
}
// ===== FORMAT ==== //
// format
else if (eqlsCMD(0, CLI_FORMAT)) {
prnt(CLI_FORMATTING_SPIFFS);
LittleFS.format();
prntln(SETUP_OK);
}
// ===== DELETE ==== //
// delete <file> [<lineFrom>] [<lineTo>]
else if ((list->size() >= 2) && eqlsCMD(0, CLI_DELETE)) {
if (list->size() == 2) {
// remove whole file
if (removeFile(list->get(1))) {
prnt(CLI_REMOVED);
prntln(list->get(1));
} else {
prnt(CLI_ERROR_REMOVING);
prntln(list->get(1));
}
} else {
// remove certain lines
int beginLine = list->get(2).toInt();
int endLine = list->size() == 4 ? list->get(3).toInt() : beginLine;
if (removeLines(list->get(1), beginLine, endLine)) {
prnt(CLI_REMOVING_LINES);
prnt(beginLine);
prnt(String(SPACE) + String(DASH) + String(SPACE));
prnt(endLine);
prntln(String(SPACE) + list->get(1));
} else {
prnt(CLI_ERROR_REMOVING);
prntln(list->get(1));
}
}
}
// ===== COPY ==== //
// delete <file> <newfile>
else if ((list->size() == 3) && eqlsCMD(0, CLI_COPY)) {
if (copyFile(list->get(1), list->get(2))) {
prntln(CLI_COPIED_FILES);
} else {
prntln(CLI_ERROR_COPYING);
}
}
// ===== RENAME ==== //
// delete <file> <newfile>
else if ((list->size() == 3) && eqlsCMD(0, CLI_RENAME)) {
if (renameFile(list->get(1), list->get(2))) {
prntln(CLI_RENAMED_FILE);
} else {
prntln(CLI_ERROR_RENAMING_FILE);
}
}
// ===== WRITE ==== //
// write <file> <commands>
else if ((list->size() >= 3) && eqlsCMD(0, CLI_WRITE)) {
String path = list->get(1);
String buf = String();
int listSize = list->size();
for (int i = 2; i < listSize; i++) {
buf += list->get(i);
if (i < listSize - 1) buf += SPACE;
}
prnt(CLI_WRITTEN);
prnt(buf);
prnt(CLI_TO);
prntln(list->get(1));
buf += NEWLINE;
appendFile(path, buf);
}
// ===== REPLACE ==== //
// replace <file> <line> <new-content>
else if ((list->size() >= 4) && eqlsCMD(0, CLI_REPLACE)) {
int line = list->get(2).toInt();
String tmp = String();
for (int i = 3; i < list->size(); i++) {
tmp += list->get(i);
if (i < list->size() - 1) tmp += SPACE;
}
if (replaceLine(list->get(1), line, tmp)) {
prnt(CLI_REPLACED_LINE);
prnt(line);
prnt(CLI_WITH);
prntln(list->get(1));
} else {
prnt(CLI_ERROR_REPLACING_LINE);
prntln(list->get(1));
}
}
// ===== RUN ==== //
// run <file> [continue <num>]
else if ((list->size() >= 2) && eqlsCMD(0, CLI_RUN)) {
execFile(list->get(1));
}
// ===== PRINT ==== //
// print <file> [<lines>]
else if ((list->size() >= 2) && eqlsCMD(0, CLI_PRINT)) {
readFileToSerial(list->get(1), eqlsCMD(2, CLI_LINE));
prntln();
}
// ===== INFO ===== //
// info
else if (eqlsCMD(0, CLI_INFO)) {
prntln(CLI_INFO_HEADER);
prnt(CLI_INFO_SOFTWARE);
prntln(DEAUTHER_VERSION);
prntln(CLI_INFO_COPYRIGHT);
prntln(CLI_INFO_LICENSE);
prntln(CLI_INFO_ADDON);
prntln(CLI_INFO_HEADER);
}
// ===== SEND ===== //
// send deauth <apMac> <stMac> <rason> <channel>
else if (eqlsCMD(0, CLI_SEND) && (list->size() == 6) && eqlsCMD(1, CLI_DEAUTH)) {
uint8_t apMac[6];
uint8_t stMac[6];
strToMac(list->get(2), apMac);
strToMac(list->get(3), stMac);
uint8_t reason = list->get(4).toInt();
uint8_t channel = list->get(5).toInt();
prnt(CLI_DEAUTHING);
prnt(macToStr(apMac));
prnt(CLI_ARROW);
prntln(macToStr(stMac));
attack.deauthDevice(apMac, stMac, reason, channel);
}
// send beacon <mac> <ssid> <ch> [wpa2]
else if (eqlsCMD(0, CLI_SEND) && (list->size() >= 5) && eqlsCMD(1, CLI_BEACON)) {
uint8_t mac[6];
strToMac(list->get(2), mac);
uint8_t channel = list->get(4).toInt();
String ssid = list->get(3);
for (int i = ssid.length(); i < 32; i++) ssid += SPACE;
prnt(CLI_SENDING_BEACON);
prnt(list->get(3));
prntln(DOUBLEQUOTES);
attack.sendBeacon(mac, ssid.c_str(), channel, eqlsCMD(5, CLI_WPA2));
}
// send probe <mac> <ssid> <ch>
else if (eqlsCMD(0, CLI_SEND) && (list->size() == 5) && eqlsCMD(1, CLI_PROBE)) {
uint8_t mac[6];
strToMac(list->get(2), mac);
uint8_t channel = list->get(4).toInt();
String ssid = list->get(3);
for (int i = ssid.length(); i < 32; i++) ssid += SPACE;
prnt(CLI_SENDING_PROBE);
prnt(list->get(3));
prntln(DOUBLEQUOTES);
attack.sendProbe(mac, ssid.c_str(), channel);
}
// send custom <packet>
else if (eqlsCMD(0, CLI_SEND) && eqlsCMD(1, CLI_CUSTOM)) {
String packetStr = list->get(2);
packetStr.replace(String(DOUBLEQUOTES), String());
uint16_t counter = 0;
uint16_t packetSize = packetStr.length() / 2;
uint8_t packet[packetSize];
for (int i = 0; i < packetSize; i++) packet[i] = strtoul((packetStr.substring(i * 2,
i * 2 + 2)).c_str(), NULL, 16);
if (attack.sendPacket(packet, packetSize, wifi_channel, 10)) {
prntln(CLI_CUSTOM_SENT);
counter++;
} else {
prntln(CLI_CUSTOM_FAILED);
}
}
// ===== LED ===== //
// led <r> <g> <b> [<brightness>]
else if ((list->size() == 4) && eqlsCMD(0, CLI_LED)) {
led::setColor(list->get(1).toInt(), list->get(2).toInt(), list->get(3).toInt());
}
// led <#rrggbb> [<brightness>]
else if ((list->size() == 2) &&
eqlsCMD(0, CLI_LED) && (list->get(1).charAt(0) == HASHSIGN)) {
uint8_t c[3];
strToColor(list->get(1), c);
led::setColor(c[0], c[1], c[2]);
}
// ===== DELAY ===== //
else if ((list->size() == 2) && eqlsCMD(0, CLI_DELAY)) {
uint32_t endTime = currentTime + getTime(list->get(1));
while (currentTime < endTime) {
// ------- loop function ----- //
currentTime = millis();
wifi::update(); // manage access point
scan.update(); // run scan
attack.update(); // run attacks
ssids.update(); // run random mode, if enabled
led::update(); // update LED color
// auto-save
if (settings::getAutosaveSettings().enabled && (currentTime - autosaveTime > settings::getAutosaveSettings().time)) {
autosaveTime = currentTime;
names.save(false);
ssids.save(false);
settings::save(false);
}
// ------- loop function end ----- //
yield();
}
}
// ===== DRAW ===== //
else if (eqlsCMD(0, CLI_DRAW)) {
int height = 25;
int width = 2;
if (list->size() >= 2) height = list->get(1).toInt();
if (list->size() >= 3) width = list->get(2).toInt();
double scale = scan.getScaleFactor(height);
prnt(String(DASH) + String(DASH) + String(DASH) + String(DASH) + String(VERTICALBAR)); // ----|
for (int j = 0; j < SCAN_PACKET_LIST_SIZE; j++) {
for (int k = 0; k < width; k++) prnt(EQUALS);
}
prntln(VERTICALBAR);
for (int i = height; i >= 0; i--) {
char s[200];
if (i == height) sprintf(s, str(CLI_DRAW_OUTPUT).c_str(),
scan.getMaxPacket() > (uint32_t)height ? scan.getMaxPacket() : (uint32_t)height);
else if (i == height / 2) sprintf(s, str(CLI_DRAW_OUTPUT).c_str(),
scan.getMaxPacket() >
(uint32_t)height ? scan.getMaxPacket() / 2 : (uint32_t)height / 2);
else if (i == 0) sprintf(s, str(CLI_DRAW_OUTPUT).c_str(), 0);
else {
s[0] = SPACE;
s[1] = SPACE;
s[2] = SPACE;
s[3] = SPACE;
s[4] = ENDOFLINE;
}
prnt(String(s));
prnt(VERTICALBAR);
for (int j = 0; j < SCAN_PACKET_LIST_SIZE; j++) {
if (scan.getPackets(j) * scale > i) {
for (int k = 0; k < width; k++) prnt(HASHSIGN);
} else {
for (int k = 0; k < width; k++) prnt(SPACE);
}
}
prntln(VERTICALBAR);
}
prnt(String(DASH) + String(DASH) + String(DASH) + String(DASH) + String(VERTICALBAR)); // ----|
for (int j = 0; j < SCAN_PACKET_LIST_SIZE; j++) {
for (int k = 0; k < width; k++) prnt(EQUALS);
}
prntln(VERTICALBAR);
prnt(String(SPACE) + String(SPACE) + String(SPACE) + String(SPACE) + String(VERTICALBAR));
for (int j = 0; j < SCAN_PACKET_LIST_SIZE; j++) {
char s[6];
String helper = String(PERCENT) + DASH + (String)width + D;
if (j == 0) sprintf(s, helper.c_str(), SCAN_PACKET_LIST_SIZE - 1);
else if (j == SCAN_PACKET_LIST_SIZE / 2) sprintf(s, helper.c_str(), SCAN_PACKET_LIST_SIZE / 2);
else if (j == SCAN_PACKET_LIST_SIZE - 1) sprintf(s, helper.c_str(), 0);
else {
int k;
for (k = 0; k < width; k++) s[k] = SPACE;
s[k] = ENDOFLINE;
}
prnt(s);
}
prntln(VERTICALBAR);
}
// ===== START/STOP AP ===== //
// startap [-p <path][-s <ssid>] [-pswd <password>] [-ch <channel>] [-h] [-cp]
else if (eqlsCMD(0, CLI_STARTAP)) {
String path = String(F("/web"));
String ssid = settings::getAccessPointSettings().ssid;
String password = settings::getAccessPointSettings().password;
int ch = wifi_channel;
bool hidden = settings::getAccessPointSettings().hidden;
bool captivePortal = settings::getWebSettings().captive_portal;
for (int i = 1; i < list->size(); i++) {
if (eqlsCMD(i, CLI_PATH)) {
i++;
path = list->get(i);
} else if (eqlsCMD(i, CLI_SSID)) {
i++;
ssid = list->get(i);
} else if (eqlsCMD(i, CLI_PASSWORD)) {
i++;
password = list->get(i);
} else if (eqlsCMD(i, CLI_CHANNEL)) {
i++;
ch = list->get(i).toInt();
} else if (eqlsCMD(i, CLI_HIDDEN)) {
hidden = true;
} else if (eqlsCMD(i, CLI_CAPTIVEPORTAL)) {
captivePortal = true;
} else {
parameterError(list->get(1));
}
}
wifi::startNewAP(path, ssid, password, ch, hidden, captivePortal);
}
// stopap
else if (eqlsCMD(0, CLI_STOPAP)) {
wifi::stopAP();
}
// ===== SCREEN ===== //
// screen mode <menu/packetmonitor/buttontest/loading>
else if (eqlsCMD(0, CLI_SCREEN) && eqlsCMD(1, CLI_MODE)) {
if (eqlsCMD(2, CLI_MODE_BUTTONTEST)) displayUI.mode = DISPLAY_MODE::BUTTON_TEST;
else if (eqlsCMD(2, CLI_MODE_PACKETMONITOR)) displayUI.mode = DISPLAY_MODE::PACKETMONITOR;
else if (eqlsCMD(2, CLI_MODE_LOADINGSCREEN)) displayUI.mode = DISPLAY_MODE::LOADSCAN;
else if (eqlsCMD(2, CLI_MODE_MENU)) displayUI.mode = DISPLAY_MODE::MENU;
else parameterError(list->get(2));
prntln(CLI_CHANGED_SCREEN);
}
// screen <on/off>
else if (eqlsCMD(0, CLI_SCREEN) && (eqlsCMD(1, CLI_ON) || eqlsCMD(1, CLI_OFF))) {
if (eqlsCMD(1, CLI_ON)) {
displayUI.on();
} else if (eqlsCMD(1, CLI_OFF)) {
displayUI.off();
}
}
// ===== NOT FOUND ===== //
else {
prnt(CLI_ERROR_NOT_FOUND_A);
prnt(input);
prntln(CLI_ERROR_NOT_FOUND_B);
// some debug stuff
/*
Serial.println(list->get(0));
for(int i=0;i<input.length();i++){
Serial.print(input.charAt(i), HEX);
Serial.print(' ');
}
*/
}
}