Compare commits

...
This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.

4 Commits

Author SHA1 Message Date
Diogo Cordeiro
2f2b6af671 brilliant hack to make everything work... 2019-04-05 10:53:46 +01:00
Diogo Cordeiro
dff282567d Fix weird instruction on token system and add some more debugging outputs to production release 2019-04-03 21:20:14 +01:00
Diogo Cordeiro
59215f4d38 Add production binary 2019-03-22 17:38:56 +00:00
Diogo Cordeiro
1c59afa216 Special MostraUP release: Add Authentication
Translated interface to portuguese as this is a portuguese event
2019-03-22 16:35:03 +00:00
10 changed files with 546 additions and 58 deletions

View File

@ -63,6 +63,8 @@ Installing the basic Fun with Binary in offline mode is relatively easy.
4. Minify the script.js with http://lisperator.net/uglifyjs/ and replace the 4. Minify the script.js with http://lisperator.net/uglifyjs/ and replace the
line after <!-- JS --> in the index.html with the result inside the tags line after <!-- JS --> in the index.html with the result inside the tags
<script>*result*</script> <script>*result*</script>
Note: If you have problems minifying try replacing `let` by `var`, then
minify, and then reverting the replace in the minified version.
5. Make the resulting HTM usable with the following tool: 5. Make the resulting HTM usable with the following tool:
https://github.com/CytronTechnologies/ESP8266-WiFi-Module/raw/master/convertHtml/convertHtml.jar https://github.com/CytronTechnologies/ESP8266-WiFi-Module/raw/master/convertHtml/convertHtml.jar

View File

@ -25,7 +25,15 @@ There are no special prerequisites for the offline mode.
At least: Arduino Uno with ESP8266-01, 6 leds (and 6 resistors), wires, breadboard. At least: Arduino Uno with ESP8266-01, 6 leds (and 6 resistors), wires, breadboard.
Some instruction of how to build it can be found on my blog post about this project: [https://blog.diogo.site/posts/fun-with-binary](https://blog.diogo.site/posts/fun-with-binary) You have instructions of how to build your own FwB in my blog: [https://blog.diogo.site/posts/fun-with-binary](https://blog.diogo.site/posts/fun-with-binary)
## MostraUP edition
This is a special edition that adds a password mechanism to force single player
in online mode.
Every output messages were translated to Portuguese due to
University of Porto's Exhibition (MostraUP) being a portuguese event.
For regular usage (during a lecture) you will probably find standard release 2.0
a better option: https://www.diogo.site/projects/fun_with_binary#releases
## Versioning ## Versioning
@ -63,18 +71,3 @@ Additional library software has been made available. All of it is Free Software
and can be distributed under liberal terms, but those terms may differ in detail and can be distributed under liberal terms, but those terms may differ in detail
from the AGPL's particulars. See each package's license file in their official from the AGPL's particulars. See each package's license file in their official
repository for additional terms. repository for additional terms.
## New this version
This is version 2.0 of Fun with Binary and includes the following (key) changes
from the previous one:
- Client <-> Server <-> Arduino was replaced by a Client <-> Arduino structure
- ESP8266 is now required for Access Point purposes
- Both modes have become PHP independent (mostly relevant in the offline one)
The last release, 2.0, gave us these improvements:
- Significant visual improvement
- 2 Powers Label switch functionality
- Offline mode is now lighter and more portable (no computer with webserver required anymore)

70
auth/index.html Normal file
View File

@ -0,0 +1,70 @@
<!--
Fun with Binary - a fun way of introducing binary
Copyright (C) 2018, Diogo Cordeiro.
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
-->
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=800" />
<title>Fun with Binary</title>
<meta name="description" content="A fun way of introducing binary">
<meta name="author" content="Diogo Cordeiro">
<!-- CSS -->
<link rel="stylesheet" type="text/css" href="style.css" />
</head>
<body class="container">
<!-- auth_html_message_Busy -->
<div id="statement">
<p class="text" id="statement_text">
Já está alguém a jogar, espera pela tua vez! :)
</p>
</div>
<!-- auth_html_message_readyToPLay -->
<div id="statement">
<p class="text" id="statement_text">
Acende as lâmpadas conforme a caixa! :)
</p>
</div>
<!-- auth_html_login_dialog -->
<div class="images" id="leds">
</div>
<br>
<ul id="settings">
<li>
<div class="button">
<input type="button" onclick="submit_auth_form()" value="Vamos Jogar!" id="play_button">
</div>
</li>
</ul>
<!-- Credits -->
<p class="text" id="footer">Feito com <span style="color:
#e25555;">&#9829;</span> por
<a href="https://www.diogo.site/">Diogo Cordeiro</a></p>
<!-- JS -->
<script src="script.js"></script>
</body>
</html>

152
auth/script.js Normal file
View File

@ -0,0 +1,152 @@
/*
* Fun with Binary - a fun way of introducing binary
* Copyright (C) 2018, Diogo Cordeiro.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* @category UI interaction
* @package Fun with Binary
* @author Diogo Cordeiro <up201705417@fc.up.pt>
* @copyright 2018 Diogo Cordeiro.
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
* @link https://www.diogo.site/projects/fun_with_binary/
*/
/***********************************
* EXTEND JAVASCRIPT FUNCTIONALITY *
***********************************/
/**
* Returns a random integer between min (inclusive) and max (inclusive)
* Using Math.round() would give a non-uniform distribution!
*/
function getRandomInt (min, max)
{
return Math.floor(Math.random() * (max - min + 1)) + min;
}
/**
* Replace char at given position
*/
String.prototype.replaceAt=function(index, replacement)
{
return this.substr(0, index) + replacement+ this.substr(index +
replacement.length);
}
/**
* Checks if session storage is available
*/
function is_session_storage_available ()
{
return window.sessionStorage != null;
}
//***********************************
/***********************************
* GAME SETTINGS *
***********************************/
// Box IP
let URL = "http://42.42.42.42/";
// Number of leds
let NLEDS = 6;
// ON and OFF leds srcs
let LED_OFF = "";
let LED_ON = "";
//***********************************
/**********************************
* ACTUAL GAME *
**********************************/
/**
* Global variables
*/
// Empty Answer
empty_answer = '';
while (empty_answer.length < NLEDS)
{
empty_answer += '0';
}
// current answer
current_answer = empty_answer;
/**
* Create leds
*/
for (let i = 0; i < NLEDS; ++i)
{
// Led holder
let figure = document.createElement ("figure");
figure.setAttribute ("class", "box");
// Led
let img = document.createElement ("img");
img.src = LED_OFF;
img.setAttribute ("value", "OFF");
img.id = "led" + i;
img.setAttribute ("class", "led");
img.setAttribute ("onclick", "switch_led("+i+")");
figure.appendChild (img);
document.getElementById("leds").appendChild(figure);
}
/**
* Turn current answer bits accordingly
*/
function switch_led (led_id)
{
let image = document.getElementById("led"+led_id);
if (image.getAttribute("value") == "ON")
{
image.setAttribute ("value", "OFF");
image.src = LED_OFF;
current_answer = current_answer.replaceAt(led_id, "0");
}
else
{
image.setAttribute ("value", "ON");
image.src = LED_ON;
current_answer = current_answer.replaceAt(led_id, "1");
}
}
function submit_auth_form ()
{
let xhttp = new XMLHttpRequest();
xhttp.onreadystatechange = function() {
if (this.readyState == 4 && this.status == 200) {
if (this.responseText == "ok")
window.location.replace (URL+"game");
else
alert (this.responseText);
}
};
xhttp.open("POST", URL+"auth", true);
xhttp.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
sessionStorage.token=getRandomInt (42, 31337)
xhttp.send("password="+current_answer+"&token="+sessionStorage.token);
}

Binary file not shown.

View File

@ -38,9 +38,9 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
<!-- Statement --> <!-- Statement -->
<div id="statement"> <div id="statement">
<p class="text" id="statement_text"> <p class="text" id="statement_text">
How do you represent the number Como representas o número
<span id="statement_value"></span> <span id="statement_value"></span>
in binary? em binário?
</p> </p>
</div> </div>
@ -48,7 +48,8 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
<ul id="settings"> <ul id="settings">
<li> <li>
<div class="button" id="label_switch_button"> <div class="button" id="label_switch_button">
<p class="text" id="settings_label_label">Labels</p> <p class="text"
id="settings_label_label">Potências de 2</p>
<label class="switch"> <label class="switch">
<input id="label_switch" <input id="label_switch"
class="toggle" class="toggle"
@ -63,7 +64,8 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
</ul> </ul>
<!-- Credits --> <!-- Credits -->
<p class="text" id="footer">Made with <span style="color: #e25555;">&#9829;</span> by <p class="text" id="footer">Feito com <span style="color:
#e25555;">&#9829;</span> por
<a href="https://www.diogo.site/">Diogo Cordeiro</a></p> <a href="https://www.diogo.site/">Diogo Cordeiro</a></p>
<!-- JS --> <!-- JS -->
<script src="script.js"></script> <script src="script.js"></script>

View File

@ -56,12 +56,11 @@ function is_session_storage_available()
/*********************************** /***********************************
* GAME SETTINGS * * GAME SETTINGS *
***********************************/ ***********************************/
// Online Mode (only used in Online Mode) // Online Mode (only used in Online Mode)
ONLINEMODE = false; ONLINEMODE = true;
// Box IP (only used in Online Mode) // Box IP
ONLINEURL = "http://42.42.42.42/"; ONLINEURL = "http://42.42.42.42/game/";
// Number of leds // Number of leds
NLEDS = 6; NLEDS = 6;
@ -87,7 +86,11 @@ LED_ON = "
* Global variables * Global variables
*/ */
// Empty Answer // Empty Answer
for (empty_answer = ''; empty_answer.length < NLEDS; empty_answer += '0'); empty_answer = '';
while (empty_answer.length < NLEDS)
{
empty_answer += '0';
}
// correct answer // correct answer
correct_answer = empty_answer; correct_answer = empty_answer;
@ -214,9 +217,9 @@ function switch_led(led_id)
if (ONLINEMODE) if (ONLINEMODE)
{ {
let xhttp = new XMLHttpRequest(); let xhttp = new XMLHttpRequest();
xhttp.open("GET", ONLINEURL+"switch_state?led=" + led_id, true); xhttp.open("GET", ONLINEURL+"switch_state?led=" + led_id +
"&token=" + sessionStorage.token, true);
xhttp.send(); xhttp.send();
} }
// If right answer, finish game, restart a new one // If right answer, finish game, restart a new one
@ -333,7 +336,8 @@ function end_game()
if (ONLINEMODE) if (ONLINEMODE)
{ {
let xhttp = new XMLHttpRequest(); let xhttp = new XMLHttpRequest();
xhttp.open("GET", ONLINEURL+"won", true); xhttp.open("GET", ONLINEURL+"won" +
"?token=" + sessionStorage.token, true);
xhttp.send(); xhttp.send();
} }

82
server/HTML.cpp Normal file
View File

@ -0,0 +1,82 @@
#include "HTML.h"
String cssHTML ()
{
const String cssHTML =
"";
return cssHTML;
}
String gameHTML1 ()
{
const String gameHTML1=
"\r\n"
"\r\n"
"\r\n";
return gameHTML1;
}
String gameHTML2 ()
{
const String gameHTML2=
"\r\n"
"\r\n"
"\r\n";
return gameHTML2;
}
String gameHTML3 ()
{
const String gameHTML3=
"\r\n"
"\r\n"
"\r\n";
return gameHTML3;
}
String gameHTML4 ()
{
const String gameHTML4=
"\r\n"
"\r\n"
"\r\n";
return gameHTML4;
}
String authHTML1 ()
{
const String authHTML1=
"\r\n"
"\r\n"
"\r\n";
return authHTML1;
}
String authHTML2 ()
{
const String authHTML2=
"\r\n"
"\r\n"
"\r\n";
return authHTML2;
}
String authHTML3 ()
{
const String authHTML3=
"\r\n"
"\r\n"
"\r\n";
return authHTML3;
}
String authHTML4 ()
{
const String authHTML4=
"\r\n"
"\r\n"
"\r\n";
return authHTML4;
}

18
server/HTML.h Normal file
View File

@ -0,0 +1,18 @@
// Due to https://github.com/esp8266/Arduino/issues/3205#issuecomment-299763849
#ifndef html_h
#define html_h
#include "Arduino.h"
String cssHTML ();
String gameHTML1 ();
String gameHTML2 ();
String gameHTML3 ();
String gameHTML4 ();
String authHTML1 ();
String authHTML2 ();
String authHTML3 ();
String authHTML4 ();
#endif

View File

@ -28,7 +28,7 @@
#include <ESP8266WebServer.h> #include <ESP8266WebServer.h>
// Pages // Pages
#include "file1.h" #include "HTML.h"
// Define a dns server for Captive Portal // Define a dns server for Captive Portal
const byte DNS_PORT = 53; const byte DNS_PORT = 53;
@ -40,7 +40,7 @@ IPAddress netMsk(255, 255, 255, 0);
// These are the WiFi access point settings. Update them to your liking // These are the WiFi access point settings. Update them to your liking
const char *ssid = "Fun with Binary"; const char *ssid = "Fun with Binary";
//const char *password = ""; //const char *wifi_password = "";
// Define a web server at port 80 for HTTP // Define a web server at port 80 for HTTP
ESP8266WebServer webServer(80); ESP8266WebServer webServer(80);
@ -49,11 +49,25 @@ ESP8266WebServer webServer(80);
#define NLEDS 6 #define NLEDS 6
// LEDs board outputs // LEDs board outputs
int bits[NLEDS] = { D0, D1, D2, D3, D4, D5 }; // Orange, Yellow, Green, Blue, Purple, Gray
// D0 D1 D2 D5 D6 D7
int bits[NLEDS] = { D2, D3, D4, D5, D6, D7 };
// Memorizes bits state // Memorizes bits state
int current_answer[NLEDS] = { 0 }; int current_answer[NLEDS] = { 0 };
// Current state of the game
bool is_game_running = false;
// Current system password
String password = "000000";
// Current system token
int token = 0;
// This is an emergency endpoint in case some of students decides to leave
// in the middle of a game, you should rename it in production as a security
// measure
const String FORCERESTARENDPOINT = "force_restart";
/** /**
* Turns all leds on * Turns all leds on
* **Doesn't update current_answer** * **Doesn't update current_answer**
@ -86,32 +100,33 @@ turn_all_off()
void void
handleRoot() handleRoot()
{ {
webServer.send(200, "text/html",
webServer.send(200, "text/html", file1); "<meta http-equiv=\"refresh\" content=\"0; url=http://42.42.42.42/auth\" />");
} }
/** /**
* 404 - Page Not Found endpoint * Game endpoint
*/ */
void void
handleNotFound() handleGame()
{ {
String message = "File Not Found\n\n"; webServer.setContentLength(gameHTML1().length() +
message += "URI: "; gameHTML2().length() +
message += webServer.uri(); gameHTML3().length() +
message += "\nMethod: "; gameHTML4().length());
message +=(webServer.method() == HTTP_GET) ? "GET" : "POST"; webServer.send(200, "text/html", gameHTML1());
message += "\nArguments: "; webServer.sendContent(gameHTML2());
message += webServer.args(); webServer.sendContent(gameHTML3());
message += "\n"; webServer.sendContent(gameHTML4());
}
for (uint8_t i = 0; i < webServer.args(); ++i) { /**
message += * CSS endpoint
" " + webServer.argName(i) + ": " + webServer.arg(i) + */
"\n"; void
} return_css()
{
webServer.send(404, "text/plain", message); webServer.send(200, "text/css", cssHTML());
} }
/** /**
@ -144,6 +159,78 @@ handleInputError()
webServer.send(500, "text/plain", message); webServer.send(500, "text/plain", message);
} }
/**
* Auth endpoint
*/
void
handleAuth()
{
if (webServer.method() == HTTP_GET)
{
webServer.setContentLength(authHTML1().length() +
authHTML2().length() +
authHTML3().length() +
authHTML4().length());
webServer.send(200, "text/html", authHTML1());
webServer.sendContent(authHTML2());
webServer.sendContent(authHTML3());
webServer.sendContent(authHTML4());
}
else if (webServer.method() == HTTP_POST)
{
if (!is_game_running)
{
if (password == webServer.arg("password"))
{
is_game_running = true;
token = webServer.arg("token").toInt();
webServer.send(200, "text/html", "ok");
turn_all_off();
Serial.print("Current valid token is: ");
Serial.println(token);
}
else
{
webServer.send(200, "text/html",
"Esta palavra-passe está errada.");
}
}
else
{
webServer.send(200, "text/html",
"Já está alguém a jogar, espera pela tua vez! :)");
}
}
else
{
// Invalid HTTP request method
handleInputError();
}
}
/**
* 404 - Page Not Found endpoint
*/
void
handleNotFound()
{
String message = "File Not Found\n\n";
message += "URI: ";
message += webServer.uri();
message += "\nMethod: ";
message +=(webServer.method() == HTTP_GET) ? "GET" : "POST";
message += "\nArguments: ";
message += webServer.args();
message += "\n";
for (uint8_t i = 0; i < webServer.args(); ++i)
message +=
" " + webServer.argName(i) + ": " + webServer.arg(i) +
"\n";
webServer.send(404, "text/plain", message);
}
/** /**
* Switches a led state (a.k.a. bit of current answer) * Switches a led state (a.k.a. bit of current answer)
* *
@ -152,27 +239,67 @@ handleInputError()
void void
switch_led() switch_led()
{ {
int incomingByte = webServer.arg("led").toInt(); if (!is_game_running || token != webServer.arg("token").toInt())
{
webServer.send(200, "text/plain", "unauthorized");
Serial.print("Invalid token tried to play: ");
Serial.println(webServer.arg("token").toInt());
return;
}
int incomingByte = webServer.arg("led").toInt();
Serial.print("Mandaram-me alterar o estado da lampada: ");
Serial.println(incomingByte);
// Validate input
if (incomingByte >= NLEDS || incomingByte < 0) if (incomingByte >= NLEDS || incomingByte < 0)
{ {
handleInputError(); handleInputError();
return; return;
} }
String message = "Led ";
if (current_answer[incomingByte] == 1) if (current_answer[incomingByte] == 1)
{ // Turn LED off { // Turn LED off
message += incomingByte;
message += " off!";
digitalWrite(bits[incomingByte], LOW); digitalWrite(bits[incomingByte], LOW);
current_answer[incomingByte] = 0; current_answer[incomingByte] = 0;
} }
else else
{ // Turn LED on { // Turn LED on
message += incomingByte;
message += " on!";
digitalWrite(bits[incomingByte], HIGH); digitalWrite(bits[incomingByte], HIGH);
current_answer[incomingByte] = 1; current_answer[incomingByte] = 1;
} }
webServer.send(200, "text/plain", message);
}
webServer.send(200, "text/plain", "ok"); /**
* Generates a binary string password with NLEDS length
* Result is stored in global password variable
*/
void
generate_password()
{
unsigned int dec_password = random(1, pow(2, NLEDS)-1);
char tmp_password[] = "000000";
for (int i = NLEDS-1; i > 0; --i)
{
tmp_password[i] =(dec_password & 1) + '0';
dec_password >>= 1;
}
for (int i = 0; i < NLEDS; ++i)
{
if (tmp_password[i] == '1')
{
password.setCharAt(i, '1');
}
}
Serial.print("Password is: ");
Serial.println(password);
} }
/** /**
@ -183,6 +310,8 @@ switch_led()
void void
print_binary_string(String which) print_binary_string(String which)
{ {
Serial.print("Eu estou a imprimir esta string: ");
Serial.println(which);
for (int i = 0; i < NLEDS; ++i) for (int i = 0; i < NLEDS; ++i)
{ {
if (which.charAt(i) == '1') if (which.charAt(i) == '1')
@ -208,6 +337,14 @@ reset_game()
{ {
current_answer[i] = 0; current_answer[i] = 0;
} }
is_game_running = false;
token = 0;
generate_password();
// print password
print_binary_string(password);
} }
/** /**
@ -216,18 +353,32 @@ reset_game()
void void
won_game() won_game()
{ {
if (!is_game_running || token != webServer.arg("token").toInt())
{
webServer.send(200, "text/plain", "unauthorized");
return;
}
for (int i = 0; i < 3; ++i) for (int i = 0; i < 3; ++i)
{ {
turn_all_off(); turn_all_off();
delay(500); delay(500);
print_binary_string(current_answer); for (int i = 0; i < NLEDS; ++i)
{
if (current_answer[i] == 1)
{
digitalWrite(bits[i], HIGH);
}
}
delay(500); delay(500);
} }
reset_game(); reset_game();
webServer.send(200, "text/plain", "ok");
} }
/** /**
@ -252,7 +403,7 @@ setup()
WiFi.softAPConfig(apIP, apIP, IPAddress(255, 255, 255, 0)); // subnet FF FF FF 00 WiFi.softAPConfig(apIP, apIP, IPAddress(255, 255, 255, 0)); // subnet FF FF FF 00
// You can remove/add the password parameter if you want the AP to be open/closed. // You can remove/add the password parameter if you want the AP to be open/closed.
WiFi.softAP(ssid); //, password); WiFi.softAP(ssid); //, wifi_password);
// if DNSServer is started with "*" for domain name, it will reply with // if DNSServer is started with "*" for domain name, it will reply with
// provided IP to all DNS request // provided IP to all DNS request
@ -269,16 +420,30 @@ setup()
webServer.on("/fwlink", handleRoot); // Microsoft captive portal. webServer.on("/fwlink", handleRoot); // Microsoft captive portal.
webServer.on("/", handleRoot); webServer.on("/", handleRoot);
webServer.on("/auth", handleAuth);
webServer.on("/"+FORCERESTARENDPOINT,[]()
{
reset_game();
webServer.send(200, "text/plain",
"A caixa foi reiniciada com sucesso!");});
webServer.on("/game", handleGame);
webServer.on("/style.css", return_css);
webServer.on("/switch_state", switch_led); webServer.on("/game/switch_state", switch_led);
webServer.on("/won", won_game); webServer.on("/game/won", won_game);
webServer.onNotFound(handleNotFound); //webServer.onNotFound(handleNotFound);
webServer.onNotFound(handleRoot);
// Start WebServer // Start WebServer
webServer.begin(); webServer.begin();
Serial.println("HTTP server started"); Serial.println("HTTP server started");
/*** Start game ***/
generate_password();
// print password
print_binary_string(password);
} }
/** /**