├── .gitignore
├── LICENSE
├── README.md
├── pirania-app
└── files
│ ├── etc
│ └── pirania
│ │ └── content.json
│ ├── usr
│ ├── libexec
│ │ └── rpcd
│ │ │ └── pirania-app
│ └── share
│ │ └── rpcd
│ │ └── acl.d
│ │ └── pirania-app.json
│ └── www
│ ├── cgi-bin
│ ├── error_handler
│ └── pirania
│ │ ├── client_ip
│ │ ├── handle_auth
│ │ ├── handle_voucher
│ │ └── portal
│ └── portal
│ ├── admin.html
│ ├── auth.html
│ ├── authenticated.html
│ ├── css
│ ├── loader2.css
│ ├── main.css
│ ├── mobile-icons.css
│ └── normalize.css
│ ├── fail.html
│ ├── index.html
│ ├── info.html
│ └── js
│ ├── admin.js
│ ├── content.js
│ ├── int.js
│ ├── tabs.js
│ ├── ubusFetch.js
│ └── voucher.js
└── pirania
└── files
├── etc
├── config
│ └── pirania
├── init.d
│ ├── pirania
│ ├── pirania-dnsmasq
│ └── pirania-uhttpd
└── pirania
│ └── db.csv
├── usr
├── bin
│ ├── captive-portal
│ └── voucher
├── lib
│ └── lua
│ │ └── voucher
│ │ ├── config.lua
│ │ ├── db.lua
│ │ ├── debugtools.lua
│ │ ├── functools.lua
│ │ ├── hooks.lua
│ │ ├── logic.lua
│ │ └── utils.lua
├── libexec
│ └── rpcd
│ │ └── pirania
└── share
│ └── rpcd
│ └── acl.d
│ └── pirania.json
└── www
└── pirania-redirect
└── redirect
/.gitignore:
--------------------------------------------------------------------------------
1 | .history
2 | .DS_Store
3 | build-router.sh
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2017 LibreMesh.org
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Deprecated! Moved to https://github.com/libremesh/lime-packages/tree/master/packages
2 |
3 | 
4 |
5 | ## Voucher and Captive Portal solution for community networks
6 |
7 | *ALPHA Software, don't use in production*
8 |
9 |
10 | This tool allows an administrator to manage a voucher system to get through the gateway.
11 |
12 | It could be used in a community that wants to share an Internet connection and for that the user's pay a fraction each, but needs the payment from everyone. So the vouchers allows to control the payments via the control of the access to Internet.
13 |
14 | ## Features
15 |
16 | This are the currently implemented features:
17 | * Runs directly from the OpenWRT/LEDE router: no need for extra hardware
18 | * Integrates it's administration with Ubus
19 | * Has a command-line interface for listing, creating and removing vouchers
20 | * Voucher database is shared among nodes in the network
21 |
22 | All planned features are accesible at: https://github.com/libremesh/voucher/issues
23 |
24 | ## Prerequisites
25 |
26 | This software assumes that will be running on a OpenWRT/LEDE distribution (because uses uci for config). Needs `ip6tables-mod-nat` and `ipset` packages installed.
27 |
28 | ## Install
29 |
30 | Not clear yet, but would be something like:
31 | * add the libremesh software feed to opkg
32 | * opkg install pirania
33 | * opkg install pirania-app
34 |
35 | # How it works
36 |
37 | It uses iptables rules to filter inbound connections outside the mesh network.
38 |
39 | ## General overview of file hierarchy and function
40 |
41 | ```
42 | files/
43 | /etc/config/pirania is the UCI config
44 | /etc/pirania/db.csv (default path) contains the database of vouchers
45 | /etc/init.d/pirania-uhttpd starts a uhttpd on port 59080 that replies any request with a redirect towards a preset URL
46 |
47 | /usr/lib/lua/voucher/ contains lua libraries used by /usr/bin/voucher
48 | /usr/bin/voucher is a CLI to manage the db (has functions add_voucher, add_many_vouchers, auth_voucher, get_valid_macs, list_vouchers, remove_voucher and url)
49 | /usr/bin/captive-portal sets up iptables rules to capture traffic
50 |
51 | /usr/libexec/rpcd/pirania ubus pirania API (this is used by the web frontend)
52 | /usr/share/rpcd/acl.d/pirania.json ACL for the ubus pirania API
53 |
54 | luasrc/ contains the luci-app to manage vouchers
55 | ```
56 |
--------------------------------------------------------------------------------
/pirania-app/files/etc/pirania/content.json:
--------------------------------------------------------------------------------
1 | {
2 | "title": "Pirania",
3 | "backgroundColor": "#f0e5de",
4 | "welcome": "Welcome, this is a captive portal",
5 | "body": "In order to have access to Interet you must enter a valid voucher. In the meantime you can continue using the local network services.",
6 | "logo": "",
7 | "rules": "
Welcome to our community network. It's a communication network owned and managed by the community. In case of any problems first learn how to use our app which can help you find any problems and check the status of the whole network.
",
8 | "mediaUrl": ""
9 | }
--------------------------------------------------------------------------------
/pirania-app/files/usr/libexec/rpcd/pirania-app:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env lua
2 | --[[
3 | Copyright 2018 Marcos Gutierrez
4 | Licensed under the Apache License, Version 2.0 (the "License");
5 | you may not use this file except in compliance with the License.
6 | You may obtain a copy of the License at
7 | http://www.apache.org/licenses/LICENSE-3.0
8 | ]]--
9 |
10 | require "ubus"
11 | local json = require 'luci.json'
12 | local path = '/etc/pirania/content.json'
13 |
14 | local function printJson (obj)
15 | print(json.encode(obj))
16 | end
17 |
18 | local conn = ubus.connect()
19 | if not conn then
20 | error("Failed to connect to ubus")
21 | end
22 |
23 | local function shell(command)
24 | local handle = io.popen(command)
25 | local result = handle:read("*a")
26 | handle:close()
27 | return result
28 | end
29 |
30 | local function get_clients(msg)
31 | local result = {}
32 | result.clients = {}
33 | local output = shell("for ip in $(ip n show | grep -v IP | awk '{print $1}' | sort -ut '|' -k 1,2); do grep $ip /tmp/dhcp.leases; done")
34 | for line in output:gmatch("[^\n]+") do
35 | local words = {}
36 | for w in line:gmatch("%S+") do if w ~= "" then table.insert(words, w) end end
37 | local mac = words[2]
38 | local ip = words[3]
39 | local station = words[4]
40 | table.insert(result.clients, { station=station, ip=ip, mac=mac })
41 | end
42 | printJson(result);
43 | end
44 |
45 | local function read_content(msg)
46 | local contents = ""
47 | local myTable = {}
48 | local file = io.open( path, "r" )
49 |
50 | if file then
51 | -- read all contents of file into a string
52 | local contents = file:read( "*a" )
53 | myTable = json.decode(contents)
54 | io.close( file )
55 | end
56 | printJson(myTable)
57 | end
58 |
59 | local function write_content(msg)
60 | local file = io.open(path, "w")
61 | if file then
62 | local contents = json.encode(msg)
63 | file:write( contents )
64 | io.close( file )
65 | read_content()
66 | end
67 | end
68 |
69 | local methods = {
70 | read_content = { no_params = 0 },
71 | get_clients = { no_params = 0 },
72 | write_content = {
73 | title = 'value',
74 | backgroundColor = 'value',
75 | welcome = 'value',
76 | body = 'value',
77 | logo = 'value',
78 | rules = 'value'
79 | }
80 | }
81 |
82 | if arg[1] == 'list' then
83 | printJson(methods)
84 | end
85 |
86 | if arg[1] == 'call' then
87 | local msg = io.read()
88 | msg = json.decode(msg)
89 | if arg[2] == 'read_content' then read_content(msg)
90 | elseif arg[2] == 'get_clients' then get_clients(msg)
91 | elseif arg[2] == 'write_content' then write_content(msg)
92 | else printJson({ error = "Method not found" })
93 | end
94 | end
--------------------------------------------------------------------------------
/pirania-app/files/usr/share/rpcd/acl.d/pirania-app.json:
--------------------------------------------------------------------------------
1 | {
2 | "unauthenticated": {
3 | "description": "Pirania public access",
4 | "write": {
5 | "ubus": {
6 | "pirania-app": [ "get_clients", "read_content" ]
7 | }
8 | }
9 | },
10 | "root": {
11 | "description": "Pirania administration access",
12 | "write": {
13 | "ubus": {
14 | "pirania-app": ["*"]
15 | }
16 | },
17 | "read": {
18 | "ubus": {
19 | "pirania-app": ["*"]
20 | }
21 | }
22 | }
23 | }
--------------------------------------------------------------------------------
/pirania-app/files/www/cgi-bin/error_handler:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | print ("Content-type: text/html\n\n")
4 |
5 | print([[
6 |
7 |
8 |
9 | Pirania
10 |
11 |
12 |
15 |
16 |
17 |
18 | ENTER
19 |
20 |
21 | ]])
--------------------------------------------------------------------------------
/pirania-app/files/www/cgi-bin/pirania/client_ip:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local json = require 'luci.json'
4 | local voucher = require('voucher.logic')
5 |
6 | print ("Content-type: application/json\n")
7 |
8 |
9 | local ipv4AndMac = voucher.getIpv4AndMac()
10 |
11 |
12 | local response = {}
13 | response.ip = ipv4AndMac.ip
14 | response.mac = ipv4AndMac.mac
15 | response.valid = voucher.check_mac_validity(ipv4AndMac.mac) > 0
16 |
17 | print(json.encode(response))
18 |
--------------------------------------------------------------------------------
/pirania-app/files/www/cgi-bin/pirania/handle_auth:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local http = require('luci.http')
4 | local logic = require('voucher.logic')
5 | local dba = require('voucher.db')
6 | local query_string = os.getenv("QUERY_STRING")
7 | local db = dba.load(config.db)
8 | local redirect_page = require('voucher.utils').redirect_page
9 | local uci_cursor = require('uci').cursor()
10 |
11 | local url_authenticated = uci_cursor:get("pirania", "base_config", "url_authenticated")
12 | local url_fail = uci_cursor:get("pirania", "base_config", "url_fail")
13 |
14 | print("Content-type: text/html \n\n")
15 |
16 | local res = logic.getIpv4AndMac()
17 | params = http.urldecode_params(query_string)
18 | local voucher = params['voucher']
19 | local mac = res.mac
20 | local prevUrl = params['prev']
21 | local valid = logic.check_mac_validity(res.mac)
22 |
23 | local url = url_fail
24 | local output
25 | local success = false
26 |
27 | if (voucher and mac) then
28 | local res = logic.auth_voucher(db, mac, voucher)
29 | if (res.success == true or valid > 0) then
30 | if (res.success == true) then
31 | success = true
32 | end
33 | if (prevUrl ~= nil) then
34 | url = 'http://'..prevUrl
35 | else
36 | url = url_authenticated
37 | end
38 | end
39 | end
40 |
41 | print(redirect_page(url))
42 |
43 |
44 | if (success == true) then
45 | dba.save(config.db, db)
46 | end
--------------------------------------------------------------------------------
/pirania-app/files/www/cgi-bin/pirania/handle_voucher:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local http = require('luci.http')
4 | local logic = require('voucher.logic')
5 | local redirect_page = require('voucher.utils').redirect_page
6 | local uci_cursor = require('uci').cursor()
7 | local dba = require('voucher.db')
8 | local query_string = os.getenv("QUERY_STRING")
9 | -- send("Status: 302 \r\n")
10 | print("Content-type: text/html \n\n")
11 | params = http.urldecode_params(query_string)
12 | local res = logic.getIpv4AndMac()
13 | local valid = logic.check_mac_validity(res.mac)
14 | local noJs = params['nojs']
15 |
16 | local url_authenticated = uci_cursor:get("pirania", "base_config", "url_authenticated")
17 | local url_fail = uci_cursor:get("pirania", "base_config", "url_fail")
18 | local url_info = uci_cursor:get("pirania", "base_config", "url_info")
19 |
20 | local url
21 |
22 | if (valid < 1) then
23 | local db = dba.load(config.db)
24 | local output
25 | local mac = res.mac
26 | local voucher = params['voucher']
27 | local prevUrl = params['prev']
28 | if (noJs == 'true') then
29 | local authCommand = "voucher auth_voucher " .. mac .. " ".. voucher
30 | ac = io.popen(authCommand, 'r')
31 | output = ac:read('*l')
32 | ac:close()
33 | local validationOut = tonumber(output)
34 | if (validationOut and validationOut > 0) then
35 | url = url_authenticated
36 | else
37 | url = url_fail
38 | end
39 | else
40 | local setParams = prevUrl and '?voucher='..voucher..'&prev='..prevUrl or'?voucher='..voucher
41 | local voucherValid = logic.check_voucher_validity(voucher, db)
42 | if (voucherValid.valid) then
43 | url = url_info..setParams
44 | else
45 | url = url_fail
46 | end
47 | end
48 |
49 |
50 |
51 | else
52 | url = url_authenticated
53 | end
54 |
55 | print(redirect_page(url))
56 |
57 |
--------------------------------------------------------------------------------
/pirania-app/files/www/cgi-bin/pirania/portal:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local http = require('luci.http')
4 | local logic = require('voucher.logic')
5 | local redirect_page = require('voucher.utils').redirect_page
6 | local uci_cursor = require('uci').cursor()
7 |
8 | local query_string = os.getenv("QUERY_STRING")
9 | print("Content-type: text/html \n\n")
10 | params = http.urldecode_params(query_string)
11 | local prevUrl = params['prev']
12 |
13 | local url_auth = uci_cursor:get("pirania", "base_config", "url_auth")
14 | local url_authenticated = uci_cursor:get("pirania", "base_config", "url_authenticated")
15 |
16 | local res = logic.getIpv4AndMac()
17 | local valid = logic.check_mac_validity(res.mac)
18 |
19 | local url
20 | local setParams = prevUrl and '?prev='..prevUrl or ''
21 |
22 | if (valid > 0) then
23 | url = url_authenticated..setParams
24 | else
25 | url = url_auth..setParams
26 | end
27 |
28 | print(redirect_page(url))
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/admin.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Pirania | Admin
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
25 |
26 |
27 |
55 |
56 |
57 |
59 |
60 |
61 |
62 |
70 |
71 |
72 |
94 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
122 |
123 |
124 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/auth.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Pirania
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
20 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/authenticated.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Pirania
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
20 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
✅
32 |
Authenticated!
33 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/css/loader2.css:
--------------------------------------------------------------------------------
1 | .lds-ring {
2 | display: inline-block;
3 | /* position: relative; */
4 | top: -6.5vh;
5 | width: 25px;
6 | height: 25px;
7 | margin: 0 auto;
8 | text-align: center;
9 | margin-bottom: -6.5vh;
10 | left: 200px;
11 | }
12 |
13 | .lds-ring div {
14 | box-sizing: border-box;
15 | display: block;
16 | position: absolute;
17 | width: 20px;
18 | height: 20px;
19 | margin: 2px;
20 | border: 2px solid #000;
21 | border-radius: 50%;
22 | animation: lds-ring 1.2s cubic-bezier(0.5, 0, 0.5, 1) infinite;
23 | border-color: #000 transparent transparent transparent;
24 | }
25 |
26 | .lds-ring div:nth-child(1) {
27 | animation-delay: -0.45s;
28 | }
29 |
30 | .lds-ring div:nth-child(2) {
31 | animation-delay: -0.3s;
32 | }
33 |
34 | .lds-ring div:nth-child(3) {
35 | animation-delay: -0.15s;
36 | }
37 |
38 | @keyframes lds-ring {
39 | 0% {
40 | transform: rotate(0deg);
41 | }
42 |
43 | 100% {
44 | transform: rotate(360deg);
45 | }
46 | }
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/css/main.css:
--------------------------------------------------------------------------------
1 | body {
2 | border: 10px solid white;
3 | }
4 |
5 | a {
6 | text-decoration: none;
7 | }
8 |
9 | input[type=number] {
10 | /*for absolutely positioning spinners*/
11 | position: relative;
12 | padding: 5px;
13 | padding-right: 25px;
14 | }
15 |
16 | input[type=number]::-webkit-inner-spin-button,
17 | input[type=number]::-webkit-outer-spin-button {
18 | opacity: 1;
19 | }
20 |
21 |
22 | input[type=number]::-webkit-outer-spin-button,
23 | input[type=number]::-webkit-inner-spin-button {
24 | -webkit-appearance: inner-spin-button !important;
25 | width: 25px;
26 | position: absolute;
27 | top: 0;
28 | right: 0;
29 | height: 100%;
30 | }
31 |
32 |
33 | input[type=text],
34 | input[type=number] {
35 | padding: 5px;
36 | border: 2px solid #ccc;
37 | -webkit-border-radius: 15px;
38 | border-radius: 15px;
39 | }
40 |
41 | input[type=text]:focus,
42 | input[type=number]:focus {
43 | border-color: #333;
44 | }
45 |
46 | input[type=submit],
47 | button {
48 | padding: 10px 25px;
49 | background: #ccc;
50 | border: 0 none;
51 | cursor: pointer;
52 | -webkit-border-radius: 5px;
53 | border-radius: 5px;
54 | }
55 |
56 | input,
57 | textarea {
58 | max-width: 100%;
59 | padding: 5px;
60 | border: 2px solid #ccc;
61 | -webkit-border-radius: 15px;
62 | border-radius: 15px;
63 | }
64 |
65 | #link input {
66 | margin-top: 15vh;
67 | }
68 |
69 | video {
70 | min-width: 100%;
71 | }
72 |
73 | #content-media {
74 | margin: 0 auto;
75 | padding: 5vh 0;
76 | }
77 |
78 | #content-media img {
79 | max-width: 300px;
80 | }
81 |
82 | #user-valid {
83 | font-size: 45px;
84 | }
85 |
86 | select {
87 | display: block;
88 | font-family: sans-serif;
89 | color: #444;
90 | line-height: 1.3;
91 | padding: .6em 1.4em .5em .8em;
92 | width: 100%;
93 | max-width: 100%;
94 | box-sizing: border-box;
95 | margin: 0;
96 | border: 1px solid #aaa;
97 | box-shadow: 0 1px 0 1px rgba(0, 0, 0, .04);
98 | border-radius: .5em;
99 | -moz-appearance: none;
100 | -webkit-appearance: none;
101 | appearance: none;
102 | background-color: #fff;
103 | background-image: url('data:image/svg+xml;charset=US-ASCII,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%22292.4%22%20height%3D%22292.4%22%3E%3Cpath%20fill%3D%22%23007CB2%22%20d%3D%22M287%2069.4a17.6%2017.6%200%200%200-13-5.4H18.4c-5%200-9.3%201.8-6.9%205.4A17.6%2017.6%200%200%200%200%2082.2c0%205%201.8%209.3%205.4%2012.9l128%20127.9c3.6%203.6%207.8%205.4%2012.8%205.4s9.2-1.8%2012.8-5.4L287%2095c3.5-3.5%205.4-7.8%205.4-12.8%200-5-1.9-9.2-5.5-12.8z%22%2F%3E%3C%2Fsvg%3E'),
104 | linear-gradient(to bottom, #ffffff 0%, #e5e5e5 100%);
105 | background-repeat: no-repeat, repeat;
106 | background-position: right .7em top 50%, 0 0;
107 | background-size: .65em auto, 100%;
108 | }
109 |
110 | select::-ms-expand {
111 | display: none;
112 | }
113 |
114 | select:hover {
115 | border-color: #888;
116 | }
117 |
118 | select:focus {
119 | border-color: #aaa;
120 | box-shadow: 0 0 1px 3px rgba(59, 153, 252, .7);
121 | box-shadow: 0 0 0 3px -moz-mac-focusring;
122 | color: #222;
123 | outline: none;
124 | }
125 |
126 | select option {
127 | font-weight: normal;
128 | }
129 |
130 | .main {
131 | margin: 0 auto;
132 | max-width: 968px;
133 | padding: 5%;
134 | text-align: center;
135 | animation: ease-in;
136 | min-height: 90vh;
137 | }
138 |
139 | /* Tabs */
140 | .tab-nav {
141 | margin: 0;
142 | padding: 0;
143 | }
144 |
145 | .tab-link {
146 | margin-right: 5px;
147 | padding: 10px 20px;
148 | color: #fff;
149 | display: inline-block;
150 | background-color: #b3b3b3;
151 | }
152 |
153 | .tab-link:last-child {
154 | margin-right: 0;
155 | }
156 |
157 | .tab-link:hover {
158 | color: #6d6d6d;
159 | }
160 |
161 | .tab-link.is-active {
162 | color: lightcoral;
163 | background-color: #e7e7e7;
164 | }
165 |
166 | /* Tab Content */
167 | .tab-contents svg {
168 | height: 15px;
169 | width: 15px;
170 | }
171 |
172 | .tab-content {
173 | display: none;
174 | /* background-color: #e7e7e7; */
175 | }
176 |
177 | .tab-content.is-active {
178 | display: block;
179 | padding: 20px;
180 | }
181 |
182 | .content {
183 | max-width: 968px;
184 | margin: 0 auto;
185 | }
186 |
187 | .login {
188 | position: absolute;
189 | top: 15px;
190 | left: 15px;
191 | text-decoration: none;
192 | color: black;
193 | }
194 |
195 | .voucher {
196 | padding-top: 30px;
197 | }
198 |
199 | .voucher form {
200 | display: flex;
201 | flex-flow: column;
202 | align-items: center;
203 | justify-content: space-around;
204 | height: 33vh;
205 | }
206 |
207 | .voucher form * {
208 | width: 250px;
209 | }
210 |
211 | .admin-login,
212 | #tabs {
213 | margin-top: 50px;
214 | padding: 10px;
215 | }
216 |
217 | .admin-login {
218 | border: 1px solid #444;
219 | }
220 |
221 | .admin-login form,
222 | .admin-create form,
223 | .admin-content form,
224 | .admin-create-many form {
225 | display: flex;
226 | flex-flow: column;
227 | align-items: center;
228 | align-content: center;
229 | justify-content: space-around;
230 | margin: 0 auto;
231 | max-width: 300px;
232 | min-height: 20vh;
233 | }
234 |
235 | .admin-create form {
236 | min-height: 35vh;
237 | }
238 |
239 | .admin-create-many form {
240 | min-height: 45vh;
241 | }
242 |
243 | .admin-content form {
244 | min-height: 100vh;
245 | }
246 |
247 | .voucher-item svg {
248 | width: 12px;
249 | height: 12px;
250 | }
251 |
252 | .voucher-item {
253 | position: relative;
254 | display: flex;
255 | flex-flow: row nowrap;
256 | justify-content: space-between;
257 | align-items: center;
258 | align-content: flex-start;
259 | padding: 15px 0;
260 | }
261 |
262 | .voucher-item-name,
263 | .voucher-item-voucher {
264 | width: 30%
265 | }
266 |
267 | .voucher-item-expires,
268 | .voucher-item-remove,
269 | .voucher-item-renew {
270 | width: 10%;
271 | }
272 |
273 | .voucher-item-voucher {
274 | display: none;
275 | font-family: monospace;
276 | color: #6d6d6d;
277 | }
278 |
279 | .voucher-item-mq {
280 | position: absolute;
281 | left: -20px;
282 | background: #444;
283 | border-radius: 50%;
284 | padding: 0 5px;
285 | color: white;
286 | }
287 |
288 | .voucher-item-macs {
289 | display: none;
290 | }
291 |
292 | .voucher-item-name:hover~.voucher-item-macs {
293 | display: block;
294 | position: absolute;
295 | bottom: -5px;
296 | }
297 |
298 | #voucher-list-button {
299 | position: fixed;
300 | top: 5%;
301 | right: 5%;
302 | border-radius: 25px;
303 | z-index: 999;
304 | }
305 |
306 | #voucher-list-loader {
307 | position: absolute;
308 | top: 50%;
309 | left: 50%;
310 | }
311 |
312 | #error, .error {
313 | margin: 0 auto;
314 | text-align: center;
315 | padding: 10px 0;
316 | background: darkseagreen;
317 | border-radius: 8px;
318 | background: indianred;
319 | }
320 |
321 | .hidden {
322 | position: absolute !important;
323 | top: -9999px !important;
324 | left: -9999px !important;
325 | z-index: -1;
326 | }
327 |
328 | #other-devices {
329 | position: absolute;
330 | top: 30px;
331 | right: 30px;
332 | margin: 0 auto;
333 | border-radius: 25px;
334 | height: 45px;
335 | width: 45px;
336 | display: flex;
337 | flex-flow: row nowrap;
338 | text-align: center;
339 | justify-content: center;
340 | }
341 |
342 | #other-devices span {
343 | font-size: 20px;
344 | position: relative;
345 | right: 10px;
346 | }
347 |
348 | @media only screen and (max-width: 480px) {
349 | .voucher-item-name:hover~.voucher-item-voucher {
350 | display: block;
351 | position: absolute;
352 | top: -5px;
353 | }
354 | }
355 |
356 | @media only screen and (min-width: 480px) {
357 | .tab-contents svg {
358 | height: 32px;
359 | width: 32px;
360 | }
361 |
362 | .voucher-item svg {
363 | width: 20px;
364 | height: 20px;
365 | }
366 |
367 | .voucher-item-voucher {
368 | display: block;
369 | }
370 | }
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/css/mobile-icons.css:
--------------------------------------------------------------------------------
1 | .mobile.icon {
2 | color: #000;
3 | position: absolute;
4 | margin-left: 4px;
5 | margin-top: 0px;
6 | width: 12px;
7 | height: 19px;
8 | border-radius: 2px;
9 | border: solid 1px currentColor;
10 | }
11 |
12 | .mobile.icon:before {
13 | content: '';
14 | position: absolute;
15 | left: 5px;
16 | top: 1px;
17 | width: 2px;
18 | height: 1px;
19 | background-color: currentColor;
20 | }
21 |
22 | .mobile.icon:after {
23 | content: '';
24 | position: absolute;
25 | bottom: 1px;
26 | left: 5px;
27 | height: 2px;
28 | width: 2px;
29 | border-radius: 50%;
30 | background-color: currentColor;
31 | }
32 |
33 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/css/normalize.css:
--------------------------------------------------------------------------------
1 | /*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */
2 |
3 | /* Document
4 | ========================================================================== */
5 |
6 | /**
7 | * 1. Correct the line height in all browsers.
8 | * 2. Prevent adjustments of font size after orientation changes in iOS.
9 | */
10 |
11 | html {
12 | line-height: 1.15;
13 | /* 1 */
14 | -webkit-text-size-adjust: 100%;
15 | /* 2 */
16 | }
17 |
18 | /* Sections
19 | ========================================================================== */
20 |
21 | /**
22 | * Remove the margin in all browsers.
23 | */
24 |
25 | body {
26 | margin: 0;
27 | }
28 |
29 | /**
30 | * Render the `main` element consistently in IE.
31 | */
32 |
33 | main {
34 | display: block;
35 | }
36 |
37 | /**
38 | * Correct the font size and margin on `h1` elements within `section` and
39 | * `article` contexts in Chrome, Firefox, and Safari.
40 | */
41 |
42 | h1 {
43 | font-size: 2em;
44 | margin: 0.67em 0;
45 | }
46 |
47 | /* Grouping content
48 | ========================================================================== */
49 |
50 | /**
51 | * 1. Add the correct box sizing in Firefox.
52 | * 2. Show the overflow in Edge and IE.
53 | */
54 |
55 | hr {
56 | box-sizing: content-box;
57 | /* 1 */
58 | height: 0;
59 | /* 1 */
60 | overflow: visible;
61 | /* 2 */
62 | }
63 |
64 | /**
65 | * 1. Correct the inheritance and scaling of font size in all browsers.
66 | * 2. Correct the odd `em` font sizing in all browsers.
67 | */
68 |
69 | pre {
70 | font-family: monospace, monospace;
71 | /* 1 */
72 | font-size: 1em;
73 | /* 2 */
74 | }
75 |
76 | /* Text-level semantics
77 | ========================================================================== */
78 |
79 | /**
80 | * Remove the gray background on active links in IE 10.
81 | */
82 |
83 | a {
84 | background-color: transparent;
85 | }
86 |
87 | /**
88 | * 1. Remove the bottom border in Chrome 57-
89 | * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.
90 | */
91 |
92 | abbr[title] {
93 | border-bottom: none;
94 | /* 1 */
95 | text-decoration: underline;
96 | /* 2 */
97 | text-decoration: underline dotted;
98 | /* 2 */
99 | }
100 |
101 | /**
102 | * Add the correct font weight in Chrome, Edge, and Safari.
103 | */
104 |
105 | b,
106 | strong {
107 | font-weight: bolder;
108 | }
109 |
110 | /**
111 | * 1. Correct the inheritance and scaling of font size in all browsers.
112 | * 2. Correct the odd `em` font sizing in all browsers.
113 | */
114 |
115 | code,
116 | kbd,
117 | samp {
118 | font-family: monospace, monospace;
119 | /* 1 */
120 | font-size: 1em;
121 | /* 2 */
122 | }
123 |
124 | /**
125 | * Add the correct font size in all browsers.
126 | */
127 |
128 | small {
129 | font-size: 80%;
130 | }
131 |
132 | /**
133 | * Prevent `sub` and `sup` elements from affecting the line height in
134 | * all browsers.
135 | */
136 |
137 | sub,
138 | sup {
139 | font-size: 75%;
140 | line-height: 0;
141 | position: relative;
142 | vertical-align: baseline;
143 | }
144 |
145 | sub {
146 | bottom: -0.25em;
147 | }
148 |
149 | sup {
150 | top: -0.5em;
151 | }
152 |
153 | /* Embedded content
154 | ========================================================================== */
155 |
156 | /**
157 | * Remove the border on images inside links in IE 10.
158 | */
159 |
160 | img {
161 | border-style: none;
162 | }
163 |
164 | /* Forms
165 | ========================================================================== */
166 |
167 | /**
168 | * 1. Change the font styles in all browsers.
169 | * 2. Remove the margin in Firefox and Safari.
170 | */
171 |
172 | button,
173 | input,
174 | optgroup,
175 | select,
176 | textarea {
177 | font-family: inherit;
178 | /* 1 */
179 | font-size: 100%;
180 | /* 1 */
181 | line-height: 1.15;
182 | /* 1 */
183 | margin: 0;
184 | /* 2 */
185 | }
186 |
187 | /**
188 | * Show the overflow in IE.
189 | * 1. Show the overflow in Edge.
190 | */
191 |
192 | button,
193 | input {
194 | /* 1 */
195 | overflow: visible;
196 | }
197 |
198 | /**
199 | * Remove the inheritance of text transform in Edge, Firefox, and IE.
200 | * 1. Remove the inheritance of text transform in Firefox.
201 | */
202 |
203 | button,
204 | select {
205 | /* 1 */
206 | text-transform: none;
207 | }
208 |
209 | /**
210 | * Correct the inability to style clickable types in iOS and Safari.
211 | */
212 |
213 | button,
214 | [type="button"],
215 | [type="reset"],
216 | [type="submit"] {
217 | -webkit-appearance: button;
218 | }
219 |
220 | /**
221 | * Remove the inner border and padding in Firefox.
222 | */
223 |
224 | button::-moz-focus-inner,
225 | [type="button"]::-moz-focus-inner,
226 | [type="reset"]::-moz-focus-inner,
227 | [type="submit"]::-moz-focus-inner {
228 | border-style: none;
229 | padding: 0;
230 | }
231 |
232 | /**
233 | * Restore the focus styles unset by the previous rule.
234 | */
235 |
236 | button:-moz-focusring,
237 | [type="button"]:-moz-focusring,
238 | [type="reset"]:-moz-focusring,
239 | [type="submit"]:-moz-focusring {
240 | outline: 1px dotted ButtonText;
241 | }
242 |
243 | /**
244 | * Correct the padding in Firefox.
245 | */
246 |
247 | fieldset {
248 | padding: 0.35em 0.75em 0.625em;
249 | }
250 |
251 | /**
252 | * 1. Correct the text wrapping in Edge and IE.
253 | * 2. Correct the color inheritance from `fieldset` elements in IE.
254 | * 3. Remove the padding so developers are not caught out when they zero out
255 | * `fieldset` elements in all browsers.
256 | */
257 |
258 | legend {
259 | box-sizing: border-box;
260 | /* 1 */
261 | color: inherit;
262 | /* 2 */
263 | display: table;
264 | /* 1 */
265 | max-width: 100%;
266 | /* 1 */
267 | padding: 0;
268 | /* 3 */
269 | white-space: normal;
270 | /* 1 */
271 | }
272 |
273 | /**
274 | * Add the correct vertical alignment in Chrome, Firefox, and Opera.
275 | */
276 |
277 | progress {
278 | vertical-align: baseline;
279 | }
280 |
281 | /**
282 | * Remove the default vertical scrollbar in IE 10+.
283 | */
284 |
285 | textarea {
286 | overflow: auto;
287 | }
288 |
289 | /**
290 | * 1. Add the correct box sizing in IE 10.
291 | * 2. Remove the padding in IE 10.
292 | */
293 |
294 | [type="checkbox"],
295 | [type="radio"] {
296 | box-sizing: border-box;
297 | /* 1 */
298 | padding: 0;
299 | /* 2 */
300 | }
301 |
302 | /**
303 | * Correct the cursor style of increment and decrement buttons in Chrome.
304 | */
305 |
306 | [type="number"]::-webkit-inner-spin-button,
307 | [type="number"]::-webkit-outer-spin-button {
308 | height: auto;
309 | }
310 |
311 | /**
312 | * 1. Correct the odd appearance in Chrome and Safari.
313 | * 2. Correct the outline style in Safari.
314 | */
315 |
316 | [type="search"] {
317 | -webkit-appearance: textfield;
318 | /* 1 */
319 | outline-offset: -2px;
320 | /* 2 */
321 | }
322 |
323 | /**
324 | * Remove the inner padding in Chrome and Safari on macOS.
325 | */
326 |
327 | [type="search"]::-webkit-search-decoration {
328 | -webkit-appearance: none;
329 | }
330 |
331 | /**
332 | * 1. Correct the inability to style clickable types in iOS and Safari.
333 | * 2. Change font properties to `inherit` in Safari.
334 | */
335 |
336 | ::-webkit-file-upload-button {
337 | -webkit-appearance: button;
338 | /* 1 */
339 | font: inherit;
340 | /* 2 */
341 | }
342 |
343 | /* Interactive
344 | ========================================================================== */
345 |
346 | /*
347 | * Add the correct display in Edge, IE 10+, and Firefox.
348 | */
349 |
350 | details {
351 | display: block;
352 | }
353 |
354 | /*
355 | * Add the correct display in all browsers.
356 | */
357 |
358 | summary {
359 | display: list-item;
360 | }
361 |
362 | /* Misc
363 | ========================================================================== */
364 |
365 | /**
366 | * Add the correct display in IE 10+.
367 | */
368 |
369 | template {
370 | display: none;
371 | }
372 |
373 | /**
374 | * Add the correct display in IE 10.
375 | */
376 |
377 | [hidden] {
378 | display: none;
379 | }
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/fail.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Pirania
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
20 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Redirect
5 |
6 |
7 |
8 |
16 |
17 |
18 |
19 | ENTER
20 |
21 |
22 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/info.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Pirania
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
25 |
26 |
27 |
28 |
29 |
85 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/admin.js:
--------------------------------------------------------------------------------
1 | let session = null
2 | let uploadedLogo = null
3 |
4 | var errorElem = document.getElementById('error')
5 |
6 | Date.daysBetween = function( date1, date2 ) { //Get 1 day in milliseconds
7 | var one_day=1000*60*60*24 // Convert both dates to milliseconds
8 | var date1_ms = date1.getTime()
9 | var date2_ms = date2.getTime() // Calculate the difference in milliseconds
10 | var difference_ms = date2_ms - date1_ms // Convert back to days and return
11 | return Math.round(difference_ms/one_day)
12 | }
13 |
14 |
15 | function xDaysFromNow (days) {
16 | let date = new Date()
17 | let newDate = date.setDate(date.getDate() + parseInt(days))
18 | return newDate.toString()
19 | }
20 |
21 | function makeid(length) {
22 | var text = ""
23 | var possible = "abcdefghijklmnopqrstuvwxyz0123456789"
24 | for (var i = 0; i < length; i++)
25 | text += possible.charAt(Math.floor(Math.random() * possible.length))
26 | return text
27 | }
28 |
29 | function compress(e) {
30 | const fileName = e.target.files[0].name
31 | const reader = new FileReader()
32 | reader.readAsDataURL(e.target.files[0])
33 | reader.onload = event => {
34 | const img = new Image()
35 | img.src = event.target.result
36 | img.onload = () => {
37 | const elem = document.createElement('canvas')
38 | const width = 50
39 | const scaleFactor = width / img.width
40 | elem.width = width
41 | elem.height = img.height * scaleFactor
42 | const ctx = elem.getContext('2d')
43 | ctx.drawImage(img, 0, 0, width, img.height * scaleFactor)
44 | ctx.canvas.toBlob((blob) => {
45 | const file = new File([blob], fileName, {
46 | type: 'image/jpeg',
47 | lastModified: Date.now()
48 | })
49 | }, 'image/jpeg', 1)
50 | console.log(ctx.canvas.toDataURL("image/jpeg"))
51 | uploadedLogo = ctx.canvas.toDataURL("image/jpeg")
52 | document.getElementById("logo-upload").appendChild(elem)
53 | },
54 | reader.onerror = error => console.log(error)
55 |
56 | }
57 | }
58 |
59 | function createManyVouchers () {
60 | const createLoader = document.getElementById('voucher-create-loader')
61 | const key = document.getElementById('adminManyInputKey').value
62 | const days = document.getElementById('adminManyInputDays').value
63 | const numberVouchers = document.getElementById('adminManyInputVouchers').value
64 | const epoc = xDaysFromNow(days)
65 | const vouchers = []
66 | for (let index = 0; index < numberVouchers; index++) {
67 | vouchers.push({
68 | key: `${key}-${makeid(3)}`,
69 | voucher: makeid(8),
70 | epoc,
71 | })
72 | }
73 | console.log('VOUCHERS GO', JSON.stringify(vouchers))
74 | hide(errorElem)
75 | show(createLoader)
76 | ubusFetch('pirania', 'add_many_vouchers', { vouchers }, session)
77 | .then(res => {
78 | console.log('RESPONSE', res)
79 | if (res.success) {
80 | listVouchers()
81 | console.log('VOUCHERS SUCCESS', res.success)
82 | document.getElementById('adminManyInputKey').value = ''
83 | document.getElementById('adminManyInputDays').value = 1
84 | document.getElementById('adminManyInputVouchers').value = 1
85 | document.getElementById('many-result').innerHTML = 'Sucesso!'
86 | }
87 | })
88 | .catch(err => {
89 | console.log('Can find this error ', err)
90 | listVouchers()
91 | document.getElementById('adminManyInputKey').value = ''
92 | document.getElementById('adminManyInputDays').value = 1
93 | document.getElementById('adminManyInputVouchers').value = 1
94 | document.getElementById('many-result').innerHTML = 'Sucesso!'
95 | // show(errorElem)
96 | hide(createLoader)
97 | })
98 | }
99 |
100 | function removeVoucher (name) {
101 | ubusFetch('pirania', 'remove_voucher', { name }, session)
102 | .then(res => {
103 | console.log(res)
104 | listVouchers()
105 | })
106 | .catch(err => {
107 | console.log(err)
108 | show(errorElem)
109 | })
110 | }
111 |
112 | /* Needs to be implemented in shared-state */
113 | // function renewVoucher (name) {
114 | // console.log(name, xDaysFromNow(30))
115 | // ubusFetch('pirania', 'renew_voucher', { name, date: xDaysFromNow(30) }, session)
116 | // .then(res => console.log(res))
117 | // .catch(err => {
118 | // console.log(err)
119 | // show(errorElem)
120 | // })
121 | // }
122 |
123 | function getVoucherName (name) {
124 | return name.split('-')[0]
125 | }
126 |
127 | function listVouchers () {
128 | let voucherList = document.getElementById("voucher-list")
129 | let listLoader = document.getElementById('voucher-list-loader')
130 | hide(errorElem)
131 | voucherList.innerHTML = ''
132 | show(listLoader)
133 | ubusFetch('pirania', 'list_vouchers', {}, session)
134 | .then(res => {
135 | const vouchers = res.vouchers
136 | // document.getElementById('voucher-list-button').style.display = 'none'
137 | vouchers
138 | .sort((a, b) => {
139 | if(parseInt(a.expires) > parseInt(b.expires)) { return -1; }
140 | if(parseInt(a.expires) < parseInt(b.expires)) { return 1; }
141 | return 0;
142 | })
143 | .sort((a, b) => {
144 | if(getVoucherName(a.name) < getVoucherName(b.name)) { return -1; }
145 | if(getVoucherName(a.name) > getVoucherName(b.name)) { return 1; }
146 | return 0;
147 | })
148 | .map(v => {
149 | const date = new Date (parseInt(v.expires))
150 | const dateDiff = Date.daysBetween(new Date(), date)
151 | let container = document.createElement('div')
152 | container.className = 'voucher-item'
153 | let vQuantity = document.createElement('div')
154 | vQuantity.className = 'voucher-item-mq'
155 | vQuantity.innerHTML = v.macs.length > 0 ? v.macs.length : ''
156 | container.appendChild(vQuantity)
157 | let name = document.createElement('div')
158 | name.className = 'voucher-item-name'
159 | name.innerHTML = v.name
160 | container.appendChild(name)
161 | let voucher = document.createElement('div')
162 | voucher.className = 'voucher-item-voucher'
163 | voucher.innerHTML = v.voucher
164 | container.appendChild(voucher)
165 | /* Needs to be implemented in shared-state */
166 | // let renew = document.createElementNS("http://www.w3.org/2000/svg", "svg")
167 | // renew.setAttribute ("viewBox", "0 0 32 32" )
168 | // renew.setAttribute ("stroke", "currentcolor" )
169 | // renew.setAttribute ("stroke-linecap", "round" )
170 | // renew.setAttribute ("stroke-linejoin", "round" )
171 | // renew.setAttribute ("stroke-width", "2" )
172 | // renew.setAttribute ("fill", "none" )
173 | // renew.className = 'voucher-item-renew'
174 | // renew.onclick = () => renewVoucher(v.name)
175 | // let renewPath = document.createElementNS("http://www.w3.org/2000/svg", "path")
176 | // renewPath.setAttribute('d', 'M29 16 C29 22 24 29 16 29 8 29 3 22 3 16 3 10 8 3 16 3 21 3 25 6 27 9 M20 10 L27 9 28 2')
177 | // renew.appendChild(renewPath)
178 | // container.appendChild(renew)
179 | let remove = document.createElementNS("http://www.w3.org/2000/svg", "svg")
180 | remove.setAttribute ("viewBox", "0 0 32 32" )
181 | remove.setAttribute ("stroke", "currentcolor" )
182 | remove.setAttribute ("stroke-linecap", "round" )
183 | remove.setAttribute ("stroke-linejoin", "round" )
184 | remove.setAttribute ("stroke-width", "2" )
185 | remove.setAttribute ("fill", "none" )
186 | remove.className = 'voucher-item-remove'
187 | remove.onclick = () => removeVoucher(v.name)
188 | let removePath = document.createElementNS("http://www.w3.org/2000/svg", "path")
189 | removePath.setAttribute('d', 'M28 6 L6 6 8 30 24 30 26 6 4 6 M16 12 L16 24 M21 12 L20 24 M11 12 L12 24 M12 6 L13 2 19 2 20 6')
190 | remove.appendChild(removePath)
191 | container.appendChild(remove)
192 | let expires = document.createElement('div')
193 | expires.className = 'voucher-item-expires'
194 | expires.innerHTML = dateDiff > 0 ? dateDiff +' '+ int[lang].days : 0
195 | container.appendChild(expires)
196 | let macs = document.createElement('div')
197 | macs.innerHTML = v.macs.toString()
198 | macs.className = 'voucher-item-macs'
199 | listLoader.className = 'hidden'
200 | container.appendChild(macs)
201 | voucherList.appendChild(container)
202 | document.getElementById('voucher-list-button').classList.remove('hidden')
203 | })
204 | })
205 | .catch(err => {
206 | console.log(err)
207 | hide(listLoader)
208 | show(document.getElementById('voucher-list-button'))
209 | errorElem.innerHTML = int[lang].error
210 | show(errorElem)
211 | })
212 | }
213 |
214 | function updateContent () {
215 | const backgroundColor = document.getElementById('adminInputBackground').value
216 | const title = document.getElementById('adminInputTitle').value
217 | const welcome = document.getElementById('adminInputWelcome').value
218 | const body = document.getElementById('adminInputBody').value
219 | let logo = uploadedLogo || content.logo
220 | ubusFetch(
221 | 'pirania-app',
222 | 'write_content',
223 | {
224 | backgroundColor,
225 | title,
226 | welcome,
227 | body,
228 | logo,
229 | },
230 | session
231 | )
232 | .then(res => {
233 | content = res
234 | const { backgroundColor, title, welcome, body, logo } = content
235 | document.body.style.backgroundColor = backgroundColor
236 | const contentLogo = document.getElementById('content-logo')
237 | const contentTitle = document.getElementById('content-title')
238 | const contentWelcome = document.getElementById('content-welcome')
239 | const contentBody = document.getElementById('content-body')
240 | if (contentLogo) contentLogo.src = logo
241 | if (contentTitle) contentTitle.innerHTML = title
242 | if (contentWelcome) contentWelcome.innerHTML = welcome
243 | if (contentBody) contentBody.innerHTML = body
244 | })
245 | .catch(err => {
246 | errorElem.innerHTML = int[lang].error
247 | show(errorElem)
248 | })
249 | }
250 |
251 | function adminAuth () {
252 | const password = document.getElementById('adminInput').value
253 | ubusFetch(
254 | 'session',
255 | 'login',
256 | {
257 | username: 'root',
258 | password,
259 | timeout: 5000,
260 | }
261 | )
262 | .then(res => {
263 | hide(errorElem)
264 | session = res.ubus_rpc_session
265 | document.querySelector('.admin-login').style.display = 'none'
266 | document.querySelector('#tabs').classList.remove('hidden')
267 | const adminContent = document.querySelector('.admin-content')
268 | const { backgroundColor, title, welcome, body } = content
269 | document.getElementById('adminInputTitle').value = title
270 | document.getElementById('adminInputWelcome').value = welcome
271 | document.getElementById('adminInputBody').value = body
272 | document.getElementById('adminInputBackground').value = backgroundColor
273 | listVouchers()
274 | })
275 | .catch(err => {
276 | console.log(err)
277 | show(errorElem)
278 | errorElem.innerHTML = int[lang].wrongPassword
279 | })
280 | }
281 |
282 | document.getElementById("logo-file").addEventListener("change", function (event) {
283 | compress(event)
284 | })
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/content.js:
--------------------------------------------------------------------------------
1 | let loader = document.createElement('div')
2 | loader.className = 'lds-ring'
3 | loader.appendChild(document.createElement('div'))
4 | loader.appendChild(document.createElement('div'))
5 | loader.appendChild(document.createElement('div'))
6 | loader.appendChild(document.createElement('div'))
7 |
8 | const show = elem => elem.classList.remove('hidden')
9 | const hide = elem => (elem.className += ' hidden')
10 |
11 | var nojsElem = document.getElementById('nojs')
12 | if (nojsElem) {
13 | nojsElem.setAttribute('value', false)
14 | }
15 |
16 | var param = '?prev='
17 | var prevUrl = window.location.search.split(param)[1]
18 | if (prevUrl) {
19 | var prevElem = document.createElement('input')
20 | prevElem.setAttribute('value', prevUrl)
21 | prevElem.setAttribute('id', 'prev')
22 | prevElem.setAttribute('name', 'prev')
23 | document.getElementById('voucher').appendChild(prevElem)
24 | }
25 |
26 | let content = {
27 | backgroundColor: 'white',
28 | title: '',
29 | welcome: '',
30 | body: '',
31 | logo: '',
32 | rules: '',
33 | mediaUrl: ''
34 | }
35 |
36 | function getContent () {
37 | ubusFetch('pirania-app', 'read_content')
38 | .then(res => {
39 | content = res
40 | const { backgroundColor, title, welcome, body, logo, rules, mediaUrl } = content
41 | document.body.style.backgroundColor = backgroundColor
42 | const contentLogo = document.getElementById('content-logo')
43 | const contentTitle = document.getElementById('content-title')
44 | const contentWelcome = document.getElementById('content-welcome')
45 | const contentBody = document.getElementById('content-body')
46 | const contentRules = document.getElementById('content-rules')
47 | const contentMedia = document.getElementById('content-media')
48 |
49 | if (contentLogo) {
50 | show(contentLogo)
51 | contentLogo.src = logo
52 | }
53 | if (contentTitle) contentTitle.innerHTML = title
54 | if (contentWelcome) contentWelcome.innerHTML = welcome
55 | if (contentBody) contentBody.innerHTML = body
56 | if (contentRules) contentRules.innerHTML = rules
57 | if (contentMedia) {
58 | var mediaType = mediaUrl.split('.')[mediaUrl.split('.').length -1]
59 | if (mediaType === 'mp4' || mediaType === 'webm' || mediaType === 'avi') {
60 | var videoContainerElem = document.createElement('video')
61 | var videoElem = document.createElement('source')
62 | contentMedia.append(videoContainerElem)
63 | videoElem.setAttribute('src', mediaUrl)
64 | videoElem.setAttribute('type', 'video/'+mediaType)
65 | videoContainerElem.appendChild(videoElem)
66 | } else if (mediaType === 'jpg' || mediaType === 'png' || mediaType === 'jpeg' || mediaType === 'gif' || mediaType === 'svg') {
67 | var imageElem = document.createElement('img')
68 | contentMedia.append(imageElem)
69 | imageElem.setAttribute('src', mediaUrl)
70 | imageElem.setAttribute('type', 'image/'+mediaType)
71 | }
72 | }
73 | })
74 | .catch(err => {
75 | document.getElementById('error').innerHTML = int[lang].error
76 | })
77 | }
78 |
79 | getContent()
80 |
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/int.js:
--------------------------------------------------------------------------------
1 | const userLang = navigator.language || navigator.userLanguage
2 | const lang = userLang.split('-')[0] || userLang || 'en'
3 |
4 | const int = {
5 | pt: {
6 | selectVoucher: 'Entre o voucher',
7 | createNewVoucher: 'Criar novo voucher',
8 | createManyVouchers: 'Criar muitos vouchers',
9 | changeContent: 'Mudar o conteúdo',
10 | title: 'Título',
11 | welcome: 'Bem vindo',
12 | body: 'Texto principal',
13 | backgroundColor: 'Cor de fundo',
14 | rules: 'Regras da rede',
15 | listVouchers: 'Listar vouchers',
16 | success: 'Sucesso',
17 | error: 'Erro',
18 | invalid: 'Código incorreto',
19 | wrongPassword: 'Senha incorreta',
20 | name: 'Nome',
21 | days: 'Dias',
22 | numberOfVouchers: 'Número de vouchers',
23 | authenticated: 'Seu dispositivo está autenticado',
24 | wait: 'Aguarde',
25 | continue: 'Continuar',
26 | info: 'Mais informações',
27 | },
28 | es: {
29 | selectVoucher: 'Entre el voucher',
30 | createNewVoucher: 'Crear nuevo voucher',
31 | createManyVouchers: 'Crear muchos vouchers',
32 | changeContent: 'Cambiar el contenido',
33 | title: 'Título',
34 | welcome: 'Bienvenido',
35 | body: 'Texto principal',
36 | backgroundColor: 'Color de fondo',
37 | rules: 'Reglas de la rede',
38 | listVouchers: 'Listar vouchers',
39 | success: 'Sucesso',
40 | error: 'Erro',
41 | invalid: 'Código incorrecto',
42 | wrongPassword: 'Contraseña incorrecta',
43 | name: 'Nombre',
44 | days: 'Dias',
45 | numberOfVouchers: 'Cantidad de vouchers',
46 | authenticated: 'Tu dispositivo esta autenticado',
47 | wait: 'Espere',
48 | continue: 'Seguir',
49 | info: 'Mas informaciones',
50 | },
51 | en: {
52 | selectVoucher: 'Enter a voucher',
53 | createNewVoucher: 'Create new voucher',
54 | createManyVouchers: 'Create many vouchers',
55 | changeContent: 'Change content',
56 | title: 'Title',
57 | welcome: 'Welcome text',
58 | body: 'Main text',
59 | backgroundColor: 'Background color',
60 | rules: 'Network rules',
61 | listVouchers: 'List vouchers',
62 | success: 'Success',
63 | error: 'Error',
64 | invalid: 'Invalid voucher',
65 | wrongPassword: 'Wrong password',
66 | name: 'Nome',
67 | days: 'Dias',
68 | numberOfVouchers: 'Number of vouchers',
69 | authenticated: "You're device is authenticated!",
70 | wait: 'Wait',
71 | continue: 'Continue',
72 | info: 'More information',
73 | }
74 | }
75 |
76 | Object.keys(int[lang]).map(text => {
77 | Array.from(document.getElementsByClassName(`int-${text}`)).map(
78 | element => {
79 | if (element.tagName === 'INPUT') {
80 | element.value = int[lang][text]
81 | } else {
82 | element.innerHTML = int[lang][text]
83 | }
84 | }
85 | )
86 | })
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/tabs.js:
--------------------------------------------------------------------------------
1 | // Start tabs.js
2 | (function() {
3 |
4 | 'use strict';
5 |
6 | /**
7 | * tabs
8 | *
9 | * @description The Tabs component.
10 | * @param {Object} options The options hash
11 | */
12 | var tabs = function(options) {
13 |
14 | var el = document.querySelector(options.el);
15 | var tabNavigationLinks = el.querySelectorAll(options.tabNavigationLinks);
16 | var tabContentContainers = el.querySelectorAll(options.tabContentContainers);
17 | var activeIndex = 0;
18 | var initCalled = false;
19 |
20 | /**
21 | * init
22 | *
23 | * @description Initializes the component by removing the no-js class from
24 | * the component, and attaching event listeners to each of the nav items.
25 | * Returns nothing.
26 | */
27 | var init = function() {
28 | if (!initCalled) {
29 | initCalled = true;
30 | el.classList.remove('no-js');
31 |
32 | for (var i = 0; i < tabNavigationLinks.length; i++) {
33 | var link = tabNavigationLinks[i];
34 | handleClick(link, i);
35 | }
36 | }
37 | };
38 |
39 | /**
40 | * handleClick
41 | *
42 | * @description Handles click event listeners on each of the links in the
43 | * tab navigation. Returns nothing.
44 | * @param {HTMLElement} link The link to listen for events on
45 | * @param {Number} index The index of that link
46 | */
47 | var handleClick = function(link, index) {
48 | link.addEventListener('click', function(e) {
49 | e.preventDefault();
50 | goToTab(index);
51 | });
52 | };
53 |
54 | /**
55 | * goToTab
56 | *
57 | * @description Goes to a specific tab based on index. Returns nothing.
58 | * @param {Number} index The index of the tab to go to
59 | */
60 | var goToTab = function(index) {
61 | if (index !== activeIndex && index >= 0 && index <= tabNavigationLinks.length) {
62 | tabNavigationLinks[activeIndex].classList.remove('is-active');
63 | tabNavigationLinks[index].classList.add('is-active');
64 | tabContentContainers[activeIndex].classList.remove('is-active');
65 | tabContentContainers[index].classList.add('is-active');
66 | activeIndex = index;
67 | }
68 | };
69 |
70 | /**
71 | * Returns init and goToTab
72 | */
73 | return {
74 | init: init,
75 | goToTab: goToTab
76 | };
77 |
78 | };
79 |
80 | /**
81 | * Attach to global namespace
82 | */
83 | window.tabs = tabs;
84 |
85 | })();
86 | // End tabs.js
87 |
88 | // Initialise at bottom of HTML
89 | var myTabs = tabs({
90 | el: '#tabs',
91 | tabNavigationLinks: '.tab-link',
92 | tabContentContainers: '.tab-content'
93 | });
94 |
95 | myTabs.init();
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/ubusFetch.js:
--------------------------------------------------------------------------------
1 | const url = 'http://thisnode.info/ubus'
2 | let ubusError = false
3 |
4 | function parseJSON(response) {
5 | return response.json()
6 | }
7 |
8 | const ubusFetch = (call, action, params, session) => new Promise ((resolve, reject) => {
9 | const form = {
10 | id: 99,
11 | jsonrpc: '2.0',
12 | method: 'call',
13 | params:[
14 | session || '00000000000000000000000000000000',
15 | call,
16 | action,
17 | params || {},
18 | ]
19 | }
20 | fetch(url, {
21 | method: 'POST',
22 | body: JSON.stringify(form),
23 | headers: {
24 | 'Access-Control-Allow-Origin': 'http://thisnode.info'
25 | },
26 | })
27 | .then(parseJSON)
28 | .then((res) => {
29 | if (res && res.result[1]) {
30 | resolve(res.result[1])
31 | } else {
32 | ubusError = true
33 | reject(int[lang].error)
34 | }
35 | })
36 | .catch((err) => {
37 | console.log('Ubus error ', err)
38 | ubusError = true
39 | reject(int[lang].error)
40 | })
41 | })
--------------------------------------------------------------------------------
/pirania-app/files/www/portal/js/voucher.js:
--------------------------------------------------------------------------------
1 | var validMacs = []
2 | var userIp = null
3 | var userMac = null
4 | var userIsValid = null
5 |
6 | var voucherButton = document.getElementById('voucherInput-submit')
7 | let voucherElem = document.getElementById('voucher')
8 |
9 | const validMacsForm = {
10 | id: 99,
11 | jsonrpc: '2.0',
12 | method: 'call',
13 | params: [
14 | '00000000000000000000000000000000',
15 | 'pirania',
16 | 'print_valid_macs',
17 | {}
18 | ]
19 | }
20 |
21 | const validGetClients = {
22 | id: 99,
23 | jsonrpc: '2.0',
24 | method: 'call',
25 | params: [
26 | '00000000000000000000000000000000',
27 | 'pirania-app',
28 | 'get_clients',
29 | {}
30 | ]
31 | }
32 |
33 | async function loadAsyncData () {
34 | await getIp()
35 | await getValidClients()
36 | await getValidMacs()
37 | }
38 |
39 | function init () {
40 | console.log('Welcome to Pirania!')
41 | // Add responses
42 | var error = document.createElement('h4')
43 | var result = document.createElement('p')
44 | var form = document.getElementsByClassName('voucher')[0]
45 | form.append(error)
46 | form.append(result)
47 | error.setAttribute('id', 'error')
48 | result.setAttribute('id', 'result')
49 | error.className = 'hidden'
50 | result.className = 'hidden'
51 |
52 | // Add list
53 | var deviceList = document.createElement('button')
54 | var add = document.createElement('span')
55 | var icon = document.createElement('div')
56 | document.body.appendChild(deviceList)
57 | deviceList.appendChild(add)
58 | deviceList.appendChild(icon)
59 | icon.className = 'mobile icon'
60 | add.innerHTML = '+'
61 | deviceList.setAttribute('id', 'other-devices')
62 | deviceList.addEventListener('click', async function (e) {
63 | e.preventDefault()
64 | showingList = !showingList
65 | if (showingList) {
66 | show(stationList)
67 | show(voucherButton)
68 | show(voucherElem)
69 | deviceList.style.backgroundColor = '#A593E0'
70 | } else {
71 | hide(stationList)
72 | deviceList.style.backgroundColor = ''
73 | }
74 | await loadAsyncData()
75 | })
76 | }
77 |
78 | init()
79 |
80 | var showingList = false
81 | var stationList = document.getElementById('station-list')
82 | var errorElem = document.getElementById('error')
83 | var resultElem = document.getElementById('result')
84 |
85 | function prepareResult (res) {
86 | if (res.error) {
87 | console.log(res.error)
88 | errorElem.innerHTML = res.error
89 | show(errorElem)
90 | ubusError = true
91 | } else if (res && res.result[1]) return res.result[1]
92 | else return false
93 | }
94 |
95 | function authVoucher () {
96 | if (!userMac) return
97 | let mac
98 | if (showingList) {
99 | mac = document.getElementById('stations').value || userMac
100 | } else {
101 | mac = userMac
102 | }
103 | let voucher = voucherElem.value.toLowerCase()
104 | voucherElem.after(loader)
105 | show(loader)
106 | const authVoucherForm = {
107 | id: 99,
108 | jsonrpc: '2.0',
109 | method: 'call',
110 | params: [
111 | '00000000000000000000000000000000',
112 | 'pirania',
113 | 'auth_voucher',
114 | {
115 | voucher,
116 | mac
117 | }
118 | ]
119 | }
120 | fetch(url, {
121 | method: 'POST',
122 | body: JSON.stringify(authVoucherForm),
123 | headers: {
124 | 'Access-Control-Allow-Origin': 'http://thisnode.info'
125 | }
126 | })
127 | .then(parseJSON)
128 | .then(prepareResult)
129 | .then(res => {
130 | hide(loader)
131 | show(voucherButton)
132 | if (res && res.success) {
133 | result.innerHTML = int[lang].success
134 | show(result)
135 | hide(errorElem)
136 | loadAsyncData()
137 | } else if (res && !res.success) {
138 | errorElem.innerHTML = int[lang].invalid
139 | show(errorElem)
140 | }
141 | voucherElem.value = ''
142 | })
143 | .catch(err => {
144 | console.log('UBUS error:', err)
145 | errorElem.innerHTML = err
146 | show(errorElem)
147 | ubusError = true
148 | })
149 | }
150 |
151 | function getIp () {
152 | return fetch('/cgi-bin/pirania/client_ip', {
153 | headers: {
154 | 'Access-Control-Allow-Origin': 'http://thisnode.info'
155 | }
156 | })
157 | .then(async i => {
158 | const res = await i.json()
159 | userIp = res.ip
160 | userMac = res.mac
161 | userIsValid = res.valid
162 | })
163 | .catch(err => {
164 | console.log('Error fetching mac:', err)
165 | ubusError = true
166 | })
167 | }
168 |
169 | function getValidClients () {
170 | if (!ubusError) {
171 | const myDiv = document.getElementById('station-list')
172 | const exists = document.getElementById('stations')
173 | if (!exists) {
174 | const select = document.createElement('select')
175 | select.id = 'stations'
176 | myDiv.appendChild(select)
177 | }
178 | }
179 | return fetch(url, {
180 | method: 'POST',
181 | body: JSON.stringify(validGetClients),
182 | headers: {
183 | 'Access-Control-Allow-Origin': 'http://thisnode.info'
184 | }
185 | })
186 | .then(parseJSON)
187 | .then(prepareResult)
188 | .then(res => {
189 | if (res && !ubusError) {
190 | document.getElementById('stations').innerHTML = ''
191 | res.clients.map(i => {
192 | const valid = validMacs.filter(valid => i.mac === valid).length > 0
193 | const node = document.createElement('option')
194 | let textnode = document.createTextNode('')
195 | if (userIp === i.ip) {
196 | userMac = i.mac
197 | node.selected = true
198 | }
199 | const isIp = userIp === i.ip ? '📱 ' : ''
200 | textnode.nodeValue = valid
201 | ? isIp + i.station + ' ✅'
202 | : isIp + i.station
203 | node.value = i.mac
204 | node.appendChild(textnode)
205 | return document.getElementById('stations').appendChild(node)
206 | })
207 | }
208 | })
209 | .catch(err => {
210 | console.log(int[lang].error, err)
211 | errorElem.innerHTML = int[lang].error
212 | show(errorElem)
213 | ubusError = true
214 | })
215 | }
216 |
217 | function getValidMacs () {
218 | fetch(url, {
219 | method: 'POST',
220 | body: JSON.stringify(validMacsForm),
221 | headers: {
222 | 'Access-Control-Allow-Origin': 'http://thisnode.info'
223 | }
224 | })
225 | .then(parseJSON)
226 | .then(res => {
227 | if (res && res.result[1]) {
228 | validMacs = res.result[1].macs
229 | if (validMacs.length > 0) {
230 | getValidClients()
231 | }
232 | } else if (res.error) {
233 | console.log(res.error)
234 | errorElem.innerHTML = int[lang].error
235 | ubusError = true
236 | }
237 | })
238 | .catch(err => {
239 | console.log(int[lang].error, err)
240 | errorElem.innerHTML = int[lang].error
241 | ubusError = true
242 | })
243 | }
244 |
245 | voucherButton.addEventListener('click', function (e) {
246 | hide(voucherButton)
247 | show(loader)
248 | if (showingList) {
249 | e.preventDefault()
250 | authVoucher()
251 | }
252 | })
253 |
--------------------------------------------------------------------------------
/pirania/files/etc/config/pirania:
--------------------------------------------------------------------------------
1 | config base_config 'base_config'
2 | option enabled '0'
3 | option portal_url 'http://thisnode.info/portal/'
4 | option url_auth '/portal/auth.html'
5 | option url_authenticated '/portal/authenticated.html'
6 | option url_info '/portal/info.html'
7 | option url_fail '/portal/fail.html'
8 | option db_path '/etc/pirania/db.csv'
9 | option elections_path '/etc/pirania/elections.json'
10 | option hooks_path '/etc/pirania/hooks/'
11 | option uploadlimit '10000000'
12 | option downloadlimit '50000'
13 | option append_ipt_rules '0' # if set to 1, iptables rules will be Appended instead of Inserted
14 | list whitelist_ipv4 '10.0.0.0/8'
15 | list whitelist_ipv4 '172.16.0.0/12'
16 | list whitelist_ipv4 '192.168.0.0/16'
17 | list whitelist_ipv6 '2a00:1508:0a00::/40'
18 | list catch_interfaces 'br-lan'
19 | list catch_interfaces 'anygw'
20 |
--------------------------------------------------------------------------------
/pirania/files/etc/init.d/pirania:
--------------------------------------------------------------------------------
1 | #!/bin/sh /etc/rc.common
2 |
3 | START=99
4 | NAME=pirania
5 | PROG=/usr/bin/captive-portal
6 |
7 | start() {
8 | if [ $(uci get pirania.base_config.enabled) == 1 ]; then
9 | logger -t pirania "Running portal captive"
10 | $PROG
11 | /usr/lib/lua/voucher/hooks.lua start
12 | else
13 | logger -t pirania "Portal captive is disabled"
14 | fi
15 | }
16 |
17 | stop () {
18 | /usr/lib/lua/voucher/hooks.lua stop
19 | $PROG stop
20 | logger -t pirania "Portal captive stopped"
21 | }
22 |
--------------------------------------------------------------------------------
/pirania/files/etc/init.d/pirania-dnsmasq:
--------------------------------------------------------------------------------
1 | #!/bin/sh /etc/rc.common
2 |
3 | START=50
4 |
5 | USE_PROCD=1
6 |
7 | DNSMASQ_BIN="/usr/sbin/dnsmasq"
8 |
9 | start_service() {
10 | procd_open_instance
11 | procd_set_param command $DNSMASQ_BIN \
12 | --keep-in-foreground \
13 | --port=59053 \
14 | --no-resolv \
15 | --server=/$(uci get dhcp.@dnsmasq[0].domain)/$(uci get network.lm_net_br_lan_anygw_if.ipaddr) \
16 | --addn-hosts=/var/hosts/shared-state-dnsmasq_hosts \
17 | --address=/thisnode.info/$(uci get network.lm_net_br_lan_anygw_if.ipaddr) \
18 | --address=/\#/1.2.3.4
19 |
20 | # respawn automatically if something died, be careful if you have an alternative process supervisor
21 | # if process dies sooner than respawn_threshold, it is considered crashed and after 5 retries the service is stopped
22 | procd_set_param respawn ${respawn_threshold:-3600} ${respawn_timeout:-5} ${respawn_retry:-5}
23 |
24 | procd_set_param stderr 1 # forward stderr of the command to logd
25 | procd_close_instance
26 | }
--------------------------------------------------------------------------------
/pirania/files/etc/init.d/pirania-uhttpd:
--------------------------------------------------------------------------------
1 | #!/bin/sh /etc/rc.common
2 |
3 | START=50
4 |
5 | USE_PROCD=1
6 |
7 | UHTTPD_BIN="/usr/sbin/uhttpd"
8 |
9 | start_service() {
10 | procd_open_instance
11 | procd_set_param command $UHTTPD_BIN -f -h /www/pirania-redirect/ -E / -l / -L /www/pirania-redirect/redirect -n 20 -p 59080
12 | # respawn automatically if something died, be careful if you have an alternative process supervisor
13 | # if process dies sooner than respawn_threshold, it is considered crashed and after 5 retries the service is stopped
14 | procd_set_param respawn ${respawn_threshold:-3600} ${respawn_timeout:-5} ${respawn_retry:-5}
15 |
16 | procd_set_param stderr 1 # forward stderr of the command to logd
17 | procd_close_instance
18 | }
19 |
--------------------------------------------------------------------------------
/pirania/files/etc/pirania/db.csv:
--------------------------------------------------------------------------------
1 | key,voucher,expiretime,uploadlimit,downloadlimit,amountofmacsallowed,usedmacs,
2 |
--------------------------------------------------------------------------------
/pirania/files/usr/bin/captive-portal:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # requires ip6tables-mod-nat and ipset
3 |
4 | clean_tables () {
5 | echo "Cleaning captive-portal rules"
6 | for ipvX in ipv4 ipv6 ; do
7 | if [ "$ipvX" = "ipv4" ] ; then
8 | iptables=iptables
9 | family=inet
10 | ipaddr=ipaddr
11 | else
12 | iptables=ip6tables
13 | family=inet6
14 | ipaddr=ip6addr
15 | fi
16 |
17 | ### Cleanup
18 | for interface in br-lan anygw; do
19 | $iptables -t mangle -D PREROUTING -i $interface -j pirania
20 | done
21 |
22 | $iptables -t nat -D PREROUTING -j pirania
23 | $iptables -t filter -D FORWARD -j pirania
24 |
25 | for table in mangle nat filter; do
26 | $iptables -t $table -F pirania
27 | $iptables -t $table -X pirania
28 | done
29 | done
30 | }
31 |
32 | clean_sets () {
33 | ipset flush pirania-auth-macs
34 | for ipvX in ipv4 ipv6 ; do
35 | ipset flush pirania-whitelist-$ipvX
36 | done
37 | }
38 |
39 | set_iptables () {
40 | echo "Apply captive-portal rules"
41 |
42 | append_ipt_rules=$(uci get pirania.base_config.append_ipt_rules 2> /dev/null)
43 | if [ "$append_ipt_rules" = "1" ] ; then
44 | AorI="A"
45 | else
46 | AorI="I"
47 | fi
48 |
49 | for ipvX in ipv4 ipv6 ; do
50 | if [ "$ipvX" = "ipv4" ] ; then
51 | iptables=iptables
52 | family=inet
53 | ipaddr=ipaddr
54 | anygw=$(uci get network.lm_net_br_lan_anygw_if.ipaddr)
55 | else
56 | iptables=ip6tables
57 | family=inet6
58 | ipaddr=ip6addr
59 | anygw=[$(uci get network.lan.ip6addr | cut -d/ -f1)]
60 | fi
61 |
62 | ### Buildup
63 | for table in mangle nat filter; do
64 | $iptables -t $table -N pirania
65 | done
66 |
67 | $iptables -t nat -$AorI PREROUTING -j pirania
68 | $iptables -t filter -$AorI FORWARD -j pirania
69 |
70 | for interface in $(uci get pirania.base_config.catch_interfaces); do
71 | $iptables -t mangle -$AorI PREROUTING -i $interface -j pirania
72 | done
73 |
74 | $iptables -t nat -A pirania -p udp -m set ! --match-set pirania-whitelist-$ipvX src -m set ! --match-set pirania-auth-macs src --dport 53 -j DNAT --to-destination $anygw:59053
75 |
76 | $iptables -t mangle -A pirania -m set --match-set pirania-auth-macs src -j RETURN
77 | $iptables -t mangle -A pirania -m set --match-set pirania-whitelist-$ipvX dst -j RETURN
78 | $iptables -t mangle -A pirania -j MARK --set-mark 0x66/0xff # everything not auth nor whitelisted will be marked for REJECT
79 | $iptables -t mangle -A pirania -p tcp -m tcp --dport 80 -j MARK --set-mark 0x80/0xff # unless is dport 80, re-set mark for REDIRECT
80 |
81 | $iptables -t nat -A pirania -p tcp -m tcp -m mark --mark 0x80/0xff -j REDIRECT --to-ports 59080
82 | $iptables -t filter -A pirania -p tcp -m mark --mark 0x66/0xff -j REJECT --reject-with tcp-reset
83 | $iptables -t filter -A pirania -m mark --mark 0x66/0xff -j REJECT
84 | done
85 | }
86 |
87 | set_ipsets () {
88 | ipset -exist create pirania-auth-macs hash:mac timeout 0
89 | for mac in $(voucher print_valid_macs) ; do
90 | ipset -exist add pirania-auth-macs $mac
91 | done
92 | for ipvX in ipv4 ipv6 ; do
93 | if [ "$ipvX" = "ipv4" ] ; then
94 | family=inet
95 | else
96 | family=inet6
97 | fi
98 | ipset -exist create pirania-whitelist-$ipvX hash:net family $family
99 | for item in $(uci get pirania.base_config.whitelist_$ipvX); do
100 | ipset -exist add pirania-whitelist-$ipvX $item
101 | done
102 | done
103 | }
104 |
105 | # check if captive-portal is enabled in /etc/config/pirania
106 | enabled=$(uci get pirania.base_config.enabled)
107 |
108 | if [ "$1" = "start" ]; then
109 | echo "Running captive-portal"
110 | clean_tables
111 | clean_sets
112 | set_ipsets
113 | set_iptables
114 | exit
115 | elif [ "$1" = "update" ] ; then
116 | clean_sets
117 | set_ipsets
118 | exit
119 | elif [ "$1" = "clean" ] || [ "$1" = "stop" ] ; then
120 | clean_tables
121 | clean_sets
122 | exit
123 | elif [ "$enabled" = "1" ]; then
124 | clean_tables
125 | clean_sets
126 | set_ipsets
127 | set_iptables
128 | exit
129 | else
130 | echo "Pirania captive-portal is disabled. Try running captive-portal start"
131 | exit
132 | fi
133 |
134 |
--------------------------------------------------------------------------------
/pirania/files/usr/bin/voucher:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local config = require('voucher.config')
4 | local dba = require('voucher.db')
5 | local logic = require('voucher.logic')
6 | local ft = require('voucher.functools')
7 | local utils = require('voucher.utils')
8 | local uci = require('uci')
9 | local json = require 'luci.json'
10 |
11 | local uci_cursor = uci.cursor()
12 | local arguments
13 | local action
14 | local context
15 |
16 | captive_portal = {}
17 |
18 | local function printJson (obj)
19 | print(json.encode(obj))
20 | end
21 |
22 | local function shell(command)
23 | local handle = io.popen(command)
24 | local result = handle:read("*a")
25 | handle:close()
26 | return result
27 | end
28 |
29 | local function split(inputstr, sep)
30 | if sep == nil then
31 | sep = "+"
32 | end
33 | local t={}
34 | for str in string.gmatch(inputstr, "([^"..sep.."]+)") do
35 | table.insert(t, str)
36 | end
37 | return t
38 | end
39 |
40 | -- Show or change portal url
41 | captive_portal.url = function(context)
42 | local url = ''
43 | if (context[1] == 'show') then
44 | url = uci_cursor:get("pirania", "base_config", "portal_url")
45 | else
46 | url = context[1] and context[1] or "http://thisnode.info/portal"
47 | uci_cursor:set("pirania", "base_config", "portal_url", url)
48 | uci_cursor:commit("pirania")
49 | end
50 | print(url)
51 | end
52 |
53 | -- Update voucher date
54 | local function updateVoucher(name, date)
55 | local result = {}
56 | local db = dba.load(config.db)
57 | local toRenew = {}
58 | for _, voucher in pairs (db.data) do
59 | if (voucher[1] == name) then
60 | toRenew = voucher
61 | end
62 | end
63 | if (#toRenew > 0) then
64 | result.success = true
65 | toRenew[3] = tonumber(date)
66 | local validVouchers = ft.filter(function(row, index) return row[1] ~= name end, db.data)
67 | local newDb = db
68 | newDb.data = validVouchers
69 | table.insert(newDb.data, toRenew)
70 | dba.save(config.db, newDb)
71 | else result.success = false
72 | end
73 | return result
74 | end
75 |
76 |
77 | --[[
78 | --Checks if a context(defined by a mac and a voucher code)
79 | --is authorized to be used, and associates the mac to the
80 | --voucher code if needed.
81 | --]]
82 | captive_portal.auth_voucher = function(context)
83 | local mac = context[1]
84 | local voucher = context[2]
85 |
86 | local db = dba.load(config.db)
87 | local res = logic.auth_voucher(db, mac, voucher)
88 | print ( res.limit[1], res.limit[2], res.limit[3], res.limit[4] )
89 | if (res.success == true) then
90 | print('Saving to DB')
91 | print('...')
92 | dba.save(config.db, db)
93 | print('saved!')
94 | end
95 | end
96 |
97 | --[[
98 | --Checks if the mac of the given context is allowed to browse.
99 | --]]
100 | captive_portal.status = function(context)
101 | local mac = context[1]
102 |
103 | local db = dba.load(config.db)
104 | print ( logic.status(db, mac) )
105 | end
106 |
107 | -- List all
108 | captive_portal.list_vouchers = function()
109 | local result = {}
110 | local vName = 1
111 | local vSecret = 2
112 | local vExpire = 3
113 | local vMacsAllowed = 6
114 | local usedMacs = 7
115 | local db = dba.load(config.db)
116 | for _, v in pairs(db.data) do
117 | result[_] = {}
118 | result[_].name = v[vName]
119 | result[_].expires = v[vExpire]
120 | result[_].voucher = v[vSecret]
121 | result[_].macsAllowed = v[vMacsAllowed]
122 | result[_].macs = split(v[usedMacs])
123 | end
124 | printJson(result)
125 | end
126 |
127 | -- Renew voucher
128 | captive_portal.renew_voucher = function(context)
129 | local voucherName = context[1]
130 | local renewDate = context[2]
131 | printJson(updateVoucher(voucherName, renewDate))
132 | end
133 |
134 | -- Remove voucher
135 | captive_portal.remove_voucher = function(context)
136 | local voucherName = context[1]
137 | printJson(updateVoucher(voucherName, 0))
138 | end
139 |
140 | --[[
141 | --Adds a voucher to the db with the params defined by context.
142 | --]]
143 | captive_portal.add_voucher = function(context)
144 | local key = context[1]
145 | local voucher = context[2]
146 | local epoc = context[3]
147 | local upload = context[4] and context[4] or '0'
148 | local download = context[5] and context[5] or '0'
149 | local amountofmacsallowed = context[6] and context[6] or '0'
150 | local exists = false
151 |
152 | local db = dba.load(config.db)
153 | for _, v in pairs (db.data) do
154 | if (v[1] == key) then
155 | exists = true
156 | end
157 | end
158 | if exists == false then
159 | local retval = { logic.add_voucher(db, key, voucher, epoc, upload, download, amountofmacsallowed)}
160 | dba.save(config.db, db)
161 | print ( unpack(retval) )
162 | else
163 | print(0)
164 | end
165 | end
166 |
167 | captive_portal.add_many_vouchers = function(context, data)
168 | local result = {}
169 | local table = utils.split(data, "\n")
170 | local db = dba.load(config.db)
171 | local error = false
172 | for _,val in pairs (table) do
173 | local jdata = json.decode(val)
174 | local upload = jdata.upload and jdata.upload or '10'
175 | local download = jdata.download and jdata.download or '10'
176 | local amountofmacsallowed = jdata.amountofmacsallowed and jdata.amountofmacsallowed or '1'
177 | local res = logic.add_voucher(db, jdata.key, jdata.voucher, jdata.epoc, upload, download, amountofmacsallowed)
178 | if (res == nil) then error = true end
179 | end
180 | result.success = not error
181 | dba.save(config.db, db)
182 | printJson(result)
183 | end
184 |
185 | -- TODO refactor eliminate bash portion awk sed bash-isms
186 | captive_portal.print_valid_macs = function()
187 | local db = dba.load(config.db)
188 | local macs = logic.valid_macs(db)
189 | for _, mac in ipairs(macs) do
190 | print ( mac )
191 | end
192 | end
193 |
194 | -- if is main
195 | if debug.getinfo(2).name == nil then
196 | arguments = { ... }
197 | action = arguments[1]
198 | context = ft.filter(function(row, index) return index > 1 end, arguments)
199 | if (action == 'add_many_vouchers') then stdin = io.stdin:read("*all") end
200 | captive_portal[action](context, stdin)
201 | end
202 |
203 | return captive_portal
204 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/config.lua:
--------------------------------------------------------------------------------
1 | local uci = require("uci")
2 | local pirania_config = 'pirania'
3 |
4 | ucicursor = uci.cursor()
5 |
6 | config = {
7 | db = ucicursor:get(pirania_config, 'base_config', 'db_path'),
8 | uploadlimit = ucicursor:get(pirania_config, 'base_config', 'uploadlimit'),
9 | downloadlimit = ucicursor:get(pirania_config, 'base_config', 'downloadlimit'),
10 | hooksDir = ucicursor:get(pirania_config, 'base_config', 'hooks_path')
11 | }
12 |
13 | return config
14 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/db.lua:
--------------------------------------------------------------------------------
1 | #!/bin/lua
2 |
3 | local utils = require('voucher.utils')
4 | local ft = require('voucher.functools')
5 | local hooks = require('voucher.hooks')
6 |
7 | dba = {}
8 |
9 | local function read_db_from_csv(dbinfo)
10 | local rawtable = utils.from_csv_to_table(dbinfo);
11 | if rawtable == nil then
12 | fho,err = io.open(dbinfo, "w")
13 | fho:write('key,voucher,expiretime,uploadlimit,downloadlimit,amountofmacsallowed,usedmacs,')
14 | rawtable = utils.from_csv_to_table(dbinfo);
15 | end
16 | local table = {
17 | headers = rawtable[1],
18 | data = ft.filter(function(row, index) return index > 1 end, rawtable)
19 | }
20 | return table
21 | end
22 |
23 | local function write_db_to_csv(csvname, db)
24 | data = {}
25 | data[1] = db.headers
26 | local idx = 2
27 |
28 | for _, v in pairs(db.data) do
29 | data[idx] = v
30 | idx = idx + 1
31 | end
32 |
33 | utils.from_table_to_csv(csvname, data)
34 | hooks("db_change")
35 | end
36 |
37 | function dba.get_vouchers_by_voucher(db, voucherid)
38 | local voucher_column = functools.search(function(val) return val == 'voucher' end, db.headers)
39 | return ft.filter(function(voucher) return voucher[voucher_column] == voucherid end, db.data)
40 | end
41 |
42 | function dba.get_vouchers_by_mac(db, mac)
43 | local mac_column = functools.search(function(val) return val == 'usedmacs' end, db.headers)
44 | return ft.filter(function(maclist)
45 | if(maclist[mac_column]) then
46 | local search_result = string.find(maclist[mac_column], mac)
47 | return search_result ~= nil
48 | end
49 | return false
50 | end, db.data)
51 | end
52 |
53 | function dba.get_all_vouchers(db)
54 | return db.data
55 | end
56 |
57 | function dba.describe_values(db, row)
58 | local described_row = {}
59 | for i, v in pairs(db.headers) do
60 | described_row[v] = row[i]
61 | end
62 |
63 | return described_row
64 | end
65 |
66 | function dba.add_voucher(db, key, voucher, epoc, upload, download, amountofmacsallowed)
67 | local data = {key, voucher, epoc, upload, download, amountofmacsallowed, ''}
68 | table.insert(db.data, data)
69 | return data
70 | end
71 |
72 | dba.load = read_db_from_csv
73 | dba.save = write_db_to_csv
74 |
75 | return dba
76 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/debugtools.lua:
--------------------------------------------------------------------------------
1 | #!/bin/lua
2 |
3 | debugtools = {}
4 |
5 | debugtools.print_r = function ( t )
6 | local print_r_cache={}
7 | local function sub_print_r(t,indent)
8 | if (print_r_cache[tostring(t)]) then
9 | print(indent.."*"..tostring(t))
10 | else
11 | print_r_cache[tostring(t)]=true
12 | if (type(t)=="table") then
13 | for pos,val in pairs(t) do
14 | if (type(val)=="table") then
15 | print(indent.."["..pos.."] => "..tostring(t).." {")
16 | sub_print_r(val,indent..string.rep(" ",string.len(pos)+8))
17 | print(indent..string.rep(" ",string.len(pos)+6).."}")
18 | elseif (type(val)=="string") then
19 | print(indent.."["..pos..'] => "'..val..'"')
20 | else
21 | print(indent.."["..pos.."] => "..tostring(val))
22 | end
23 | end
24 | else
25 | print(indent..tostring(t))
26 | end
27 | end
28 | end
29 | if (type(t)=="table") then
30 | print(tostring(t).." {")
31 | sub_print_r(t," ")
32 | print("}")
33 | else
34 | sub_print_r(t," ")
35 | end
36 | print()
37 | end
38 |
39 | return debugtools
40 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/functools.lua:
--------------------------------------------------------------------------------
1 | functools = {}
2 |
3 | -- Lua implementation of the curry function
4 | -- Developed by tinylittlelife.org
5 | -- released under the WTFPL (http://sam.zoy.org/wtfpl/)
6 |
7 | -- curry(func, num_args) : take a function requiring a tuple for num_args arguments
8 | -- and turn it into a series of 1-argument functions
9 | -- e.g.: you have a function dosomething(a, b, c)
10 | -- curried_dosomething = curry(dosomething, 3) -- we want to curry 3 arguments
11 | -- curried_dosomething (a1) (b1) (c1) -- returns the result of dosomething(a1, b1, c1)
12 | -- partial_dosomething1 = curried_dosomething (a_value) -- returns a function
13 | -- partial_dosomething2 = partial_dosomething1 (b_value) -- returns a function
14 | -- partial_dosomething2 (c_value) -- returns the result of dosomething(a_value, b_value, c_value)
15 | function functools.curry(func, num_args)
16 |
17 | -- currying 2-argument functions seems to be the most popular application
18 | num_args = num_args or 2
19 |
20 | -- helper
21 | local function curry_h(argtrace, n)
22 | if 0 == n then
23 | -- reverse argument list and call function
24 | return func(functools.reverse(argtrace()))
25 | else
26 | -- "push" argument (by building a wrapper function) and decrement n
27 | return function (onearg)
28 | return curry_h(function () return onearg, argtrace() end, n - 1)
29 | end
30 | end
31 | end
32 |
33 | -- no sense currying for 1 arg or less
34 | if num_args > 1 then
35 | return curry_h(function () return end, num_args)
36 | else
37 | return func
38 | end
39 | end
40 |
41 | -- reverse(...) : take some tuple and return a tuple of elements in reverse order
42 | --
43 | -- e.g. "reverse(1,2,3)" returns 3,2,1
44 | function functools.reverse(...)
45 |
46 | --reverse args by building a function to do it, similar to the unpack() example
47 | local function reverse_h(acc, v, ...)
48 | if 0 == select('#', ...) then
49 | return v, acc()
50 | else
51 | return reverse_h(function () return v, acc() end, ...)
52 | end
53 | end
54 |
55 | -- initial acc is the end of the list
56 | return reverse_h(function () return end, ...)
57 | end
58 |
59 |
60 | function functools.map(func, tbl)
61 | local newtbl = {}
62 | for i,v in pairs(tbl) do
63 | newtbl[i] = func(v)
64 | end
65 | return newtbl
66 | end
67 |
68 | function functools.filter(func, tbl)
69 | local newtbl= {}
70 | local index=1;
71 | for i,v in pairs(tbl) do
72 | if func(v, i) then
73 | newtbl[index]=v
74 | index = index + 1
75 | end
76 | end
77 | return newtbl
78 | end
79 |
80 | function functools.search(func, tbl)
81 | for i,v in pairs(tbl) do
82 | if func(v, i) then
83 | return i
84 | end
85 | end
86 |
87 | return 0
88 | end
89 |
90 | return functools
91 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/hooks.lua:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local config = require('voucher.config')
4 | local fs = require("nixio.fs")
5 |
6 | local hooks = function(action)
7 | local hookPath = config.hooksDir..action..'/'
8 | local files = fs.dir(hookPath) or pairs({})
9 | for file in files do
10 | os.execute(hookPath..file)
11 | end
12 |
13 | end
14 |
15 | if debug.getinfo(2).name == nil then
16 | arguments = { ... }
17 | if (arguments ~= nil and arguments[1] ~= nil) then
18 | hooks(arguments[1])
19 | end
20 | end
21 |
22 | return hooks
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/logic.lua:
--------------------------------------------------------------------------------
1 | #!/bin/lua
2 |
3 | local dba = require('voucher.db')
4 | local config = require('voucher.config')
5 |
6 | logic = {}
7 |
8 | local function shell(command)
9 | local handle = io.popen(command)
10 | local result = handle:read("*a")
11 | handle:close()
12 | return result
13 | end
14 |
15 | local function isMac(string)
16 | local string = string:match("%w%w:%w%w:%w%w:%w%w:%w%w:%w%w")
17 | if string then
18 | return true
19 | else
20 | return false
21 | end
22 | end
23 |
24 |
25 | local function dateNow()
26 | local output = shell('date +%s000')
27 | local parsed = string.gsub(output, "%s+", "")
28 | local dateNow = tonumber(parsed)
29 | return dateNow
30 | end
31 |
32 |
33 |
34 | local function use_voucher(db, voucher, mac)
35 | macs = voucher[7]
36 |
37 | if (string.find(macs, mac) == nil) then
38 | if (macs == '') then
39 | voucher[7] = mac
40 | else
41 | voucher[7] = macs .. '+' .. mac
42 | end
43 | end
44 | end
45 |
46 | local function get_valid_rawvoucher(db, rawvouchers)
47 | local voucher, expiretime, uploadlimit, downloadlimit
48 |
49 | for _, rawvoucher in ipairs( rawvouchers ) do
50 | if (logic.check_valid_voucher(db, rawvoucher)) then
51 | return rawvoucher
52 | end
53 | end
54 |
55 | return
56 | end
57 |
58 | local function get_limit_from_rawvoucher(db, rawvoucher)
59 | local voucher, expiretime, uploadlimit, downloadlimit
60 |
61 | if (rawvoucher ~= nil) then
62 | voucher = dba.describe_values (db, rawvoucher)
63 |
64 | if tonumber(voucher.expiretime) ~= nil then
65 | expiretime = tostring( tonumber( voucher.expiretime ) - dateNow())
66 | uploadlimit = voucher.uploadlimit ~= '0' and voucher.uploadlimit or config.uploadlimit
67 | downloadlimit = voucher.downloadlimit ~= '0' and voucher.downloadlimit or config.downloadlimit
68 | currentmacs = table.getn(utils.string_split(voucher.usedmacs, '+'))
69 | valid = currentmacs < tonumber(voucher.amountofmacsallowed) and '1' or '0'
70 | return expiretime, uploadlimit, downloadlimit, valid
71 | end
72 | end
73 |
74 | return '0', '0', '0', '0'
75 | end
76 |
77 | local function checkIfIpv4(ip)
78 | if ip == nil or type(ip) ~= "string" then
79 | return 0
80 | end
81 | -- check for format 1.11.111.111 for ipv4
82 | local chunks = {ip:match("(%d+)%.(%d+)%.(%d+)%.(%d+)")}
83 | if (#chunks == 4) then
84 | for _,v in pairs(chunks) do
85 | if (tonumber(v) < 0 or tonumber(v) > 255) then
86 | return 0
87 | end
88 | end
89 | return true
90 | else
91 | return false
92 | end
93 | end
94 |
95 | function logic.getIpv4AndMac()
96 | local address = os.getenv('REMOTE_ADDR')
97 | local isIpv4 = checkIfIpv4(address)
98 | if (isIpv4) then
99 | local ipv4macCommand = "cat /proc/net/arp | grep "..address.." | awk -F ' ' '{print $4}' | head -n 1"
100 | fd = io.popen(ipv4macCommand, 'r')
101 | ipv4mac = fd:read('*l')
102 | fd:close()
103 | local res = {}
104 | res.ip = address
105 | res.mac = ipv4mac
106 | return res
107 | else
108 | local ipv6macCommand = "ip neighbor | grep "..address.." | awk -F ' ' '{print $5}' | head -n 1"
109 | fd6 = io.popen(ipv6macCommand, 'r')
110 | ipv6mac = fd6:read('*l')
111 | fd6:close()
112 | local ipv4Command = "cat /proc/net/arp | grep "..ipv6mac.." | awk -F ' ' '{print $1}' | head -n 1"
113 | fd4 = io.popen(ipv4Command, 'r')
114 | ipv4 = fd4:read('*l')
115 | fd4:close()
116 | local res = {}
117 | res.ip = ipv4
118 | res.mac = ipv6mac
119 | return res
120 | end
121 | end
122 |
123 | function logic.check_valid_voucher(db, row)
124 | local expireDate = tonumber(dba.describe_values(db, row).expiretime) or 0
125 | if (expireDate ~= nil) then
126 | return expireDate > dateNow()
127 | end
128 | end
129 |
130 | function logic.check_voucher_validity(voucherid, db)
131 | local res = {}
132 | res.valid = false
133 | local rawvouchers = dba.get_vouchers_by_voucher(db, voucherid)
134 | if (rawvouchers ~= nil) then
135 | local voucher = get_valid_rawvoucher(db, rawvouchers)
136 | local expiretime, uploadlimit, downloadlimit, valid = get_limit_from_rawvoucher(db, voucher)
137 | if(voucher ~= nil and valid == '1' and tonumber(expiretime) > 0) then
138 | res.valid = true
139 | res.voucher = voucher
140 | end
141 | end
142 | return res
143 | end
144 |
145 | local function setIpset(mac, expiretime)
146 | -- ipset only supports timeout up to 4294967
147 | if tonumber(expiretime) > 4294967 then expiretime = 4294967 end
148 | os.execute("ipset -exist add pirania-auth-macs " .. mac .. " timeout ".. expiretime)
149 | end
150 |
151 | function logic.auth_voucher(db, mac, voucherid)
152 | local response = {
153 | success=false,
154 | limit={'0', '0', '0', '0'}
155 | }
156 | local res = logic.check_voucher_validity(voucherid, db)
157 | if (res.valid) then
158 | use_voucher(db, res.voucher, mac)
159 | setIpset(mac, res.voucher[3])
160 | response.limit={ res.voucher[3], res.voucher[4], res.voucher[5], res.voucher[6] }
161 | response.success=true
162 | end
163 | return response
164 | end
165 |
166 | function logic.check_mac_validity(mac)
167 | local command = 'voucher print_valid_macs | grep -o '..mac..' | wc -l | grep "[^[:blank:]]"'
168 | fd = io.popen(command, 'r')
169 | local output = fd:read('*all')
170 | fd:close()
171 | return tonumber(output)
172 | end
173 |
174 | function logic.add_voucher(db, key, voucher, epoc, upload, download, amountofmacsallowed)
175 | local rawvoucher = dba.add_voucher(db, key, voucher, epoc, upload, download, amountofmacsallowed)
176 |
177 | return get_limit_from_rawvoucher(db, rawvoucher)
178 | end
179 |
180 | function logic.valid_macs(db)
181 | local rawvouchers, rawvoucher, macs, currentmacs
182 | macs = {}
183 | rawvouchers = dba.get_all_vouchers(db)
184 |
185 | for _, rawvoucher in ipairs( rawvouchers ) do
186 | if logic.check_valid_voucher(db, rawvoucher) then
187 | local voucher = dba.describe_values(db, rawvoucher)
188 | currentmacs = utils.string_split(voucher.usedmacs, '+')
189 | for _, mac in ipairs( currentmacs ) do
190 | local intAmount = tonumber(voucher.amountofmacsallowed)
191 | local amountofmacs = 0
192 | if (intAmount ~= nil) then
193 | amountofmacs = intAmount
194 | end
195 | if (_ <= amountofmacs and isMac(mac)) then
196 | table.insert(macs, mac)
197 | end
198 | end
199 | end
200 | end
201 |
202 | return macs
203 | end
204 |
205 | function logic.status(db, mac)
206 | local rawvouchers, rawvoucher
207 | rawvouchers = dba.get_vouchers_by_mac(db, mac)
208 |
209 | for _, rawvoucher in ipairs( rawvouchers ) do
210 | if logic.check_valid_voucher(db, rawvoucher) then
211 | return get_limit_from_rawvoucher(db, rawvoucher)
212 | end
213 | end
214 |
215 | return '0', '0', '0', '0'
216 | end
217 |
218 | return logic
219 |
--------------------------------------------------------------------------------
/pirania/files/usr/lib/lua/voucher/utils.lua:
--------------------------------------------------------------------------------
1 | #!/bin/lua
2 |
3 | utils = {}
4 |
5 | -- Used to escape "'s by toCSV
6 | local function escapeCSV (s)
7 | if string.find(s, '[,"]') then
8 | s = '"' .. string.gsub(s, '"', '""') .. '"'
9 | end
10 | return s
11 | end
12 |
13 | -- Convert from CSV string to table (converts a single line of a CSV file)
14 | local function fromCSV (s)
15 | s = s .. ',' -- ending comma
16 | local t = {} -- table to collect fields
17 | local fieldstart = 1
18 | repeat
19 | -- next field is quoted? (start with `"'?)
20 | if string.find(s, '^"', fieldstart) then
21 | local a, c
22 | local i = fieldstart
23 | repeat
24 | -- find closing quote
25 | a, i, c = string.find(s, '"("?)', i+1)
26 | until c ~= '"' -- quote not followed by quote?
27 | if not i then error('unmatched "') end
28 | local f = string.sub(s, fieldstart+1, i-1)
29 | table.insert(t, (string.gsub(f, '""', '"')))
30 | fieldstart = string.find(s, ',', i) + 1
31 | else -- unquoted; find next comma
32 | local nexti = string.find(s, ',', fieldstart)
33 | table.insert(t, string.sub(s, fieldstart, nexti-1))
34 | fieldstart = nexti + 1
35 | end
36 | until fieldstart > string.len(s)
37 | return t
38 | end
39 |
40 | -- Convert from table to CSV string
41 | local function toCSV (tt)
42 | local s = ""
43 | for _,p in ipairs(tt) do
44 | s = s .. "," .. escapeCSV(p)
45 | end
46 | return string.sub(s, 2) -- remove first comma
47 | end
48 |
49 | utils.from_csv_to_table = function(filename)
50 | local line, lines, fh, err
51 |
52 | lines = {}
53 |
54 | fh, err = io.open(filename)
55 | if err then print("OOps"); return; end
56 |
57 | while true do
58 | line = fh:read()
59 | if line == nil or line == '' then break end
60 |
61 | table.insert(lines, fromCSV(line))
62 | end
63 |
64 | fh:close()
65 |
66 | return lines
67 | end
68 |
69 | utils.from_table_to_csv = function(filename, table)
70 | local fho, err
71 | -- Open a file for write
72 | fho,err = io.open(filename, "w")
73 |
74 | for _, line in pairs( table ) do
75 | fho:write(toCSV(line))
76 | fho:write('\n')
77 | end
78 |
79 | fho:close()
80 | end
81 |
82 | utils.string_split = function(inputstr, sep)
83 | if sep == nil then
84 | sep = "%s"
85 | end
86 | local t={} ; i=1
87 | for str in string.gmatch(inputstr, "([^"..sep.."]+)") do
88 | t[i] = str
89 | i = i + 1
90 | end
91 | return t
92 | end
93 |
94 | utils.split = function(string, sep)
95 | local ret = {}
96 | for token in string.gmatch(string, "[^"..sep.."]+") do table.insert(ret, token) end
97 | return ret
98 | end
99 |
100 | utils.redirect_page = function(url)
101 | return string.format([[
102 |
103 |
104 |
105 | Redirect
106 |
107 |
108 |
109 |
112 |
113 |
114 |
115 | ENTER
116 |
117 |
118 | ]], url, url , url)
119 | end
120 |
121 | return utils
122 |
--------------------------------------------------------------------------------
/pirania/files/usr/libexec/rpcd/pirania:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env lua
2 | --[[
3 | Copyright 2018 Marcos Gutierrez
4 | Licensed under the Apache License, Version 2.0 (the "License");
5 | you may not use this file except in compliance with the License.
6 | You may obtain a copy of the License at
7 | http://www.apache.org/licenses/LICENSE-3.0
8 | ]]--
9 |
10 | require "ubus"
11 | local json = require 'luci.json'
12 | local ft = require('voucher.functools')
13 | local utils = require('voucher.utils')
14 | local logic = require('voucher.logic')
15 | local dba = require('voucher.db')
16 |
17 | local uci = require('uci')
18 |
19 | local uci_cursor = uci.cursor()
20 |
21 | local function printJson (obj)
22 | print(json.encode(obj))
23 | end
24 |
25 | local conn = ubus.connect()
26 | if not conn then
27 | error("Failed to connect to ubus")
28 | end
29 |
30 | local function shell(command)
31 | -- TODO(nicoechaniz): sanitize or evaluate if this is a security risk
32 | local handle = io.popen(command)
33 | local result = handle:read("*a")
34 | handle:close()
35 | return result
36 | end
37 |
38 | local function list_vouchers(msg)
39 | local result = {}
40 | result = json.decode(shell('voucher list_vouchers'))
41 | printJson({ vouchers = result })
42 | end
43 |
44 | local function renew_voucher(msg)
45 | local name = msg.name
46 | local date = msg.date
47 | local result = {}
48 | result = json.decode(shell('voucher renew_voucher '..name..' '..date))
49 | printJson(result)
50 | end
51 |
52 | local function remove_voucher(msg)
53 | local name = msg.name
54 | local result = {}
55 | result = json.decode(shell('voucher remove_voucher '..name))
56 | printJson(result)
57 | end
58 |
59 | local function add_many_vouchers(msg)
60 | local result = {}
61 | local vouchers = ft.map(function(val) return json.encode(val) end, msg.vouchers)
62 | local voucher = io.popen("voucher add_many_vouchers", 'w')
63 | result.success = assert(voucher:write(table.concat(vouchers, '\n')))
64 | printJson(result)
65 | end
66 |
67 | local function add_voucher(msg)
68 | local result = {}
69 | local key = msg.key
70 | local secret = msg.secret
71 | local epoc = msg.epoc
72 | local upload = msg.upload
73 | local download = msg.download
74 | local amountofmacsallowed = msg.amountofmacsallowed
75 | local output = shell('voucher add_voucher '..key..' '..secret..' '..epoc..' '..download..' '..upload..' '..amountofmacsallowed)
76 | local parsed = string.gsub(output, "%s+", "")
77 | if tonumber(parsed) == 0 then
78 | result.success = false
79 | else
80 | result.key = key
81 | result.secret = secret
82 | result.epoc = epoc
83 | result.upload = upload
84 | result.download = download
85 | result.amountofmacsallowed = amountofmacsallowed
86 | end
87 | printJson(result)
88 | end
89 |
90 | local function auth_voucher(msg)
91 | local result = {}
92 | local mac = msg.mac
93 | local voucher = msg.voucher
94 | result.mac = mac
95 | result.voucher = voucher
96 | local db = dba.load(config.db)
97 | local res = logic.auth_voucher(db, mac, voucher)
98 | result.success = res.success
99 | -- BUG: Saving to DB causes ubus to timeout
100 | dba.save(config.db, db)
101 | printJson(result)
102 | end
103 |
104 | local function print_valid_macs(msg)
105 | local result = {}
106 | result.macs = {}
107 | local output = shell('voucher print_valid_macs')
108 | for line in output:gmatch("[^\n]+") do
109 | -- local words = {}
110 | -- for w in line:gmatch("%S+") do if w ~= "" then table.insert(words, w) end end
111 | -- local mac = words[2]
112 | table.insert(result.macs, line)
113 | end
114 | printJson(result);
115 | end
116 |
117 | local function show_url(msg)
118 | local result = {}
119 | local url = uci_cursor:get("pirania", "base_config", "portal_url")
120 | result.url = url
121 | printJson(result);
122 | end
123 |
124 | local function change_url(msg)
125 | local result = {}
126 | local url = msg.url
127 | uci_cursor:set("pirania", "base_config", "portal_url", url)
128 | uci_cursor:commit("pirania")
129 | result.url = url
130 | printJson(result);
131 | end
132 |
133 |
134 | local methods = {
135 | add_voucher = {
136 | key = 'value',
137 | secret = 'value',
138 | epoc = 0,
139 | upload = 0,
140 | download = 0,
141 | amountofmacsallowed = 0
142 | },
143 | add_many_vouchers = { vouchers = 'value' },
144 | auth_voucher = { mac = 'value', voucher = 'value' },
145 | print_valid_macs = { no_params = 0 },
146 | list_vouchers = { no_params = 0 },
147 | remove_voucher = { name = 'value' },
148 | renew_voucher = { name = 'value', date = 0 },
149 | show_url = { no_params = 0 },
150 | change_url = { url = 'value' },
151 | }
152 |
153 | if arg[1] == 'list' then
154 | printJson(methods)
155 | end
156 |
157 | if arg[1] == 'call' then
158 | local msg = io.read()
159 | msg = json.decode(msg)
160 | if arg[2] == 'add_voucher' then add_voucher(msg)
161 | elseif arg[2] == 'add_many_vouchers' then add_many_vouchers(msg)
162 | elseif arg[2] == 'auth_voucher' then auth_voucher(msg)
163 | elseif arg[2] == 'print_valid_macs' then print_valid_macs(msg)
164 | elseif arg[2] == 'list_vouchers' then list_vouchers(msg)
165 | elseif arg[2] == 'remove_voucher' then remove_voucher(msg)
166 | elseif arg[2] == 'renew_voucher' then renew_voucher(msg)
167 | elseif arg[2] == 'show_url' then show_url(msg)
168 | elseif arg[2] == 'change_url' then change_url(msg)
169 | else printJson({ error = "Method not found" })
170 | end
171 | end
172 |
--------------------------------------------------------------------------------
/pirania/files/usr/share/rpcd/acl.d/pirania.json:
--------------------------------------------------------------------------------
1 | {
2 | "unauthenticated": {
3 | "description": "pirania voucher public access",
4 | "write": {
5 | "ubus": {
6 | "pirania": [ "auth_voucher", "get_clients", "print_valid_macs", "show_url" ]
7 | }
8 | }
9 | },
10 | "root": {
11 | "description": "pirania administration access",
12 | "write": {
13 | "ubus": {
14 | "pirania": ["*"]
15 | }
16 | },
17 | "read": {
18 | "ubus": {
19 | "pirania": ["*"]
20 | }
21 | }
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/pirania/files/www/pirania-redirect/redirect:
--------------------------------------------------------------------------------
1 | #!/usr/bin/lua
2 |
3 | local uci = require('uci')
4 | local json = require('luci.json')
5 | local uci_cursor = uci.cursor()
6 | local redirect_page = require('voucher.utils').redirect_page
7 |
8 | function handle_request (env)
9 | local origin_url = env.HTTP_HOST
10 | local portal_url = uci_cursor:get("pirania", "base_config", "portal_url")
11 | if origin_url ~= portal_url then
12 | local setParams = prevUrl and '?prev='..origin_url or ''
13 | local url = portal_url..setParams
14 | local send = uhttpd.send
15 | send("Status: 302 \r\n")
16 | send("Content-type: text/html \n\n")
17 | send(print(redirect_page(url)))
18 | end
19 | end
20 |
--------------------------------------------------------------------------------