Skip to content

Instantly share code, notes, and snippets.

@jRiest
Last active January 19, 2022 13:07
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save jRiest/7893cf10c550057ce1ff53f270683e1c to your computer and use it in GitHub Desktop.
Save jRiest/7893cf10c550057ce1ff53f270683e1c to your computer and use it in GitHub Desktop.
Cloudflare worker and terraform config for partyparrot.business. Inspired by https://github.com/shrugs/partyparrot
provider "cloudflare" {}
variable "zone" {
default = "partyparrot.business"
}
resource "cloudflare_worker_script" "main_script" {
zone = "${var.zone}"
content = "${file("party_parrot_worker.js")}"
}
resource "cloudflare_worker_route" "catch_all_route" {
zone = "${var.zone}"
pattern = "*${var.zone}/*"
enabled = true
depends_on = ["cloudflare_worker_script.main_script"]
}
/*
Copyright (c) 2018, Cloudflare. All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
addEventListener('fetch', event => {
event.respondWith(handleRequest(event.request));
});
function render(opts) {
const {grid} = opts;
const colCount = grid[0].length;
const maxSize = 30;
return `<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Party Parrot</title>
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
html {
height: 100%;
}
body {
min-height: 100%;
position: relative;
padding: 20px 0 40px;
font-family: sans-serif;
}
.fill {
background-repeat: no-repeat;
background-position: center center;
background-size: contain;
}
/* images from https://www.reddit.com/r/PartyParrot/wiki/hdpartyparrots */
.p0 {
/* https://i.redd.it/o0e6ehe0ay3y.gif */
background-image: url();
}
.p1 {
/* https://i.redd.it/qaufs16x9y3y.gif */
background-image: url();
}
.wrapper {
min-height: ${maxSize * 5}px;
margin: 0 20px;
}
.char {
height: 0;
padding-top: calc((1/${colCount}) * 100%);
max-width: calc((1/${colCount}) * 100%);
width: ${maxSize}px;
}
.chars {
display: flex;
align-items: center;
flex-direction: column;
}
.row {
display: flex;
max-width: 100%;
}
form {
display: flex;
margin: 100px auto 20px;
max-width: 400px;
padding: 0 20px;
}
input {
display: block;
width: 100%;
flex: 1 1 auto;
border: 1px solid #7272E9;
border-radius: 0;
border-right-width: 0;
border-top-left-radius: 6px;
border-bottom-left-radius: 6px;
font-size: 14px;
padding: 4px 12px;
}
button {
display: block;
flex: 0 0 auto;
background: #7272E9;
border: 0;
color: #fff;
padding: 15px;
border-top-right-radius: 6px;
border-bottom-right-radius: 6px;
}
.credit {
position: absolute;
bottom: 0;
right: 0;
padding: 20px;
}
a, a:visited, a:hover, a:active {
color: #7272E9;
text-decoration: none;
}
a:hover, a:active {
text-decoration: underline;
}
</style>
</head>
<body>
<div class="wrapper">
<div class="chars">
${grid
.map(row => {
return (
'<div class="row">' +
row
.map(
(fill, colIndex) =>
// alternate between left (p0) and right (p1) parrot for every character
`<div class="char${fill ? ` fill p${Math.floor(colIndex / 6) % 2}` : ''}"></div>`,
)
.join('') +
'</div>'
);
})
.join('')}
</div>
</div>
<form action="/" method="get">
<input placeholder="custom message" name="say" autocomplete="off">
<button>Go</button>
</form>
<p class="credit">Party Parrot images from <a href="https://www.reddit.com/r/PartyParrot/wiki/hdpartyparrots">Reddit</a></p>
</body>
</html>`;
}
// Combine all characters into one grid
function getGrid(text) {
const grid = [[], [], [], [], []];
const chars = text.split('');
if (text.length > 200) {
throw new Error('Text is too long, maximum length is 200 characters');
}
chars.forEach((char, charIndex) => {
const charGrid = ALL_CHARACTERS[char];
if (!charGrid) {
throw new Error(
`Unsupported character: "${char}" only the letters A-Z, digits 0-9, and spaces are supported.`,
);
}
// add space between characters
if (charIndex > 0) {
for (const row of grid) {
row.push(0);
}
}
charGrid.forEach((row, rowIndex) => {
row.forEach((shouldFill, colIndex) => {
const gridCol = colIndex + charIndex * 6;
grid[rowIndex][gridCol] = shouldFill;
});
});
});
return grid;
}
async function handleRequest(request) {
const url = new URL(request.url);
if (url.pathname !== '/') {
return new Response('Not Found', {
status: 404,
headers: {
'content-type': 'text/plain',
},
});
}
const text = (url.searchParams.get('say') || 'hello').toLowerCase().replace(/\s+/g, ' ');
let grid;
try {
grid = getGrid(text);
} catch (e) {
return new Response(e.message, {
status: 400,
headers: {
'content-type': 'text/plain',
},
});
}
const html = render({grid});
return new Response(html, {
headers: {
'content-type': 'text/html',
},
});
}
// Character mappings from https://github.com/shrugs/partyparrot/blob/d7b1992fc2096ecbb519f80aed35c94f9ad6d1de/all_characters.py#L1
/*
Copyright (c) 2015 Matt Condon
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
const ALL_CHARACTERS = {
' ': [[0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0]],
a: [[0, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1]],
b: [[1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 1, 1, 1, 0]],
c: [[0, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0], [0, 1, 1, 1, 1]],
d: [[1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 0]],
e: [[1, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 1, 1, 1, 1]],
f: [[1, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 1, 1, 0, 0], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0]],
g: [[0, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 0, 1, 1, 1], [1, 0, 0, 0, 1], [0, 1, 1, 1, 1]],
h: [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1]],
i: [[0, 0, 1, 0, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0]],
j: [[0, 0, 1, 1, 1], [0, 0, 0, 0, 1], [0, 0, 0, 0, 1], [1, 0, 0, 0, 1], [0, 1, 1, 1, 0]],
k: [[1, 0, 0, 1, 0], [1, 0, 1, 0, 0], [1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1]],
l: [[1, 0, 0, 0, 0], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0], [1, 1, 1, 1, 1]],
m: [[1, 0, 0, 0, 1], [1, 1, 0, 1, 1], [1, 0, 1, 0, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1]],
n: [[1, 1, 0, 0, 1], [1, 0, 1, 0, 1], [1, 0, 1, 0, 1], [1, 0, 1, 0, 1], [1, 0, 0, 1, 1]],
o: [[0, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [0, 1, 1, 1, 0]],
p: [[1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 1, 1, 1, 0], [1, 0, 0, 0, 0], [1, 0, 0, 0, 0]],
q: [[0, 1, 1, 0, 0], [1, 0, 0, 1, 0], [1, 0, 0, 1, 0], [1, 0, 0, 1, 0], [0, 1, 1, 1, 1]],
r: [[1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [1, 1, 1, 1, 0], [1, 0, 0, 1, 0], [1, 0, 0, 0, 1]],
s: [[0, 1, 1, 1, 1], [1, 0, 0, 0, 0], [0, 1, 1, 1, 0], [0, 0, 0, 0, 1], [1, 1, 1, 1, 0]],
t: [[1, 1, 1, 1, 1], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0]],
u: [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [0, 1, 1, 1, 0]],
v: [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [0, 1, 0, 1, 0], [0, 1, 0, 1, 0], [0, 0, 1, 0, 0]],
w: [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 0, 1, 0, 1], [1, 1, 0, 1, 1], [1, 0, 0, 0, 1]],
x: [[1, 0, 0, 0, 1], [0, 1, 0, 1, 0], [0, 0, 1, 0, 0], [0, 1, 0, 1, 0], [1, 0, 0, 0, 1]],
y: [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [0, 1, 1, 1, 0], [0, 0, 1, 0, 0], [0, 0, 1, 0, 0]],
z: [[1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [0, 1, 1, 1, 0], [1, 0, 0, 0, 0], [1, 1, 1, 1, 1]],
'0': [[0, 1, 1, 1, 0], [1, 0, 0, 1, 1], [1, 0, 1, 0, 1], [1, 1, 0, 0, 1], [0, 1, 1, 1, 0]],
'1': [[0, 0, 0, 1, 0], [0, 0, 1, 1, 0], [0, 0, 0, 1, 0], [0, 0, 0, 1, 0], [1, 1, 1, 1, 1]],
'2': [[1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [1, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 1, 1, 1, 1]],
'3': [[1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [0, 1, 1, 1, 1], [0, 0, 0, 0, 1], [1, 1, 1, 1, 1]],
'4': [[1, 0, 0, 0, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [0, 0, 0, 0, 1]],
'5': [[1, 1, 1, 1, 1], [1, 0, 0, 0, 0], [1, 1, 1, 1, 0], [0, 0, 0, 0, 1], [1, 1, 1, 1, 0]],
'6': [[0, 0, 1, 1, 1], [0, 1, 0, 0, 0], [1, 1, 1, 1, 0], [1, 0, 0, 0, 1], [0, 1, 1, 1, 0]],
'7': [[1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [0, 0, 0, 1, 0], [0, 0, 1, 0, 0], [0, 1, 0, 0, 0]],
'8': [[1, 1, 1, 1, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1]],
'9': [[1, 1, 1, 1, 1], [1, 0, 0, 0, 1], [1, 1, 1, 1, 1], [0, 0, 0, 0, 1], [0, 0, 0, 0, 1]],
};
@lpappone
Copy link

Could you point me toward some documentation on setting up Zones? I can't find much, and when I try to run terraform apply with the above config, I get the error: cloudflare_worker_script.main_script: error finding zone "partyparrot.business": Zone could not be found

@maxsap
Copy link

maxsap commented Nov 25, 2019

Not sure if you have already checked this -> https://blog.cloudflare.com/deploy-workers-using-terraform/

@davidfernandezm
Copy link

Has anyone tried running this with mjs instead of vanilla JS? it gives error:

Error: error creating worker script: HTTP status 400: Uncaught SyntaxError: Unexpected token 'export'
  at line 285

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment