Skip to content

Commit

Permalink
wrangler init
Browse files Browse the repository at this point in the history
  • Loading branch information
willswire committed Apr 23, 2023
1 parent 33c7617 commit b452d2d
Show file tree
Hide file tree
Showing 5 changed files with 4,318 additions and 517 deletions.
251 changes: 14 additions & 237 deletions index.js
Original file line number Diff line number Diff line change
@@ -1,238 +1,15 @@
/**
* Receives a HTTP request and replies with a response.
* @param {Request} request
* @returns {Promise<Response>}
*/
async function handleRequest(request) {
const { protocol, pathname } = new URL(request.url);

// Require HTTPS (TLS) connection to be secure.
if (
"https:" !== protocol ||
"https" !== request.headers.get("x-forwarded-proto")
) {
throw new BadRequestException("Please use a HTTPS connection.");
}

switch (pathname) {

case "/nic/update":
case "/update":
if (request.headers.has("Authorization")) {
const { username, password } = basicAuthentication(request);

// Throws exception when query parameters aren't formatted correctly
const url = new URL(request.url);
verifyParameters(url);

// Only returns this response when no exception is thrown.
const response = await informAPI(url, username, password);
return response;
}

throw new BadRequestException("Please provide valid credentials.");

case "/favicon.ico":
case "/robots.txt":
return new Response(null, { status: 204 });
}

return new Response("Not Found.", { status: 404 });
}

/**
* Pass the request info to the Cloudflare API Handler
* @param {URL} url
* @param {String} name
* @param {String} token
* @returns {Promise<Response>}
*/
async function informAPI(url, name, token) {
// Parse Url
const hostnames = url.searchParams.get("hostname").split(",");
// Get the IP address. This can accept two query parameters, this will
// use the "ip" query parameter if it is set, otherwise falling back to "myip".
const ip = url.searchParams.get("ip") || url.searchParams.get("myip");

// Initialize API Handler
const cloudflare = new Cloudflare({
token: token,
});

const zone = await cloudflare.findZone(name);
for (const hostname of hostnames) {
const record = await cloudflare.findRecord(zone, hostname);
const result = await cloudflare.updateRecord(record, ip);
}

// Only returns this response when no exception is thrown.
return new Response(`good`, {
status: 200,
headers: {
"Content-Type": "text/plain;charset=UTF-8",
"Cache-Control": "no-store"
},
});
}

/**
* Throws exception on verification failure.
* @param {string} url
* @throws {UnauthorizedException}
*/
function verifyParameters(url) {
if (!url.searchParams) {
throw new BadRequestException("You must include proper query parameters");
}

if (!url.searchParams.get("hostname")) {
throw new BadRequestException("You must specify a hostname");
}

if (!(url.searchParams.get("ip") || url.searchParams.get("myip"))) {
throw new BadRequestException("You must specify an ip address");
}
}

/**
* Parse HTTP Basic Authorization value.
* @param {Request} request
* @throws {BadRequestException}
* @returns {{ user: string, pass: string }}
*/
function basicAuthentication(request) {
const Authorization = request.headers.get("Authorization");

const [scheme, encoded] = Authorization.split(" ");

// Decodes the base64 value and performs unicode normalization.
// @see https://dev.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String/normalize
const buffer = Uint8Array.from(atob(encoded), (character) =>
character.charCodeAt(0)
);
const decoded = new TextDecoder().decode(buffer).normalize();

// The username & password are split by the first colon.
//=> example: "username:password"
const index = decoded.indexOf(":");

// The user & password are split by the first colon and MUST NOT contain control characters.
// @see https://tools.ietf.org/html/rfc5234#appendix-B.1 (=> "CTL = %x00-1F / %x7F")
if (index === -1 || /[\0-\x1F\x7F]/.test(decoded)) {
throw new BadRequestException("Invalid authorization value.");
}

return {
username: decoded.substring(0, index),
password: decoded.substring(index + 1),
};
}

class UnauthorizedException {
constructor(reason) {
this.status = 401;
this.statusText = "Unauthorized";
this.reason = reason;
}
}

class BadRequestException {
constructor(reason) {
this.status = 400;
this.statusText = "Bad Request";
this.reason = reason;
}
}

class CloudflareApiException {
constructor(reason) {
this.status = 500;
this.statusText = "Internal Server Error";
this.reason = reason;
}
}

class Cloudflare {
constructor(options) {
this.cloudflare_url = "https://api.cloudflare.com/client/v4";

if (options.token) {
this.token = options.token;
}

this.findZone = async (name) => {
var response = await fetch(
`https://api.cloudflare.com/client/v4/zones?name=${name}`,
{
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${this.token}`,
},
}
);
var body = await response.json();
if(body.success !== true || body.result.length === 0) {
throw new CloudflareApiException("Failed to find zone '" + name + "'");
}
return body.result[0];
};

this.findRecord = async (zone, name) => {
var response = await fetch(
`https://api.cloudflare.com/client/v4/zones/${zone.id}/dns_records?name=${name}`,
{
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${this.token}`,
},
}
);
var body = await response.json();
if(body.success !== true || body.result.length === 0) {
throw new CloudflareApiException("Failed to find dns record '" + name + "'");
}
return body.result[0];
};

this.updateRecord = async (record, value) => {
record.content = value;
var response = await fetch(
`https://api.cloudflare.com/client/v4/zones/${record.zone_id}/dns_records/${record.id}`,
{
method: "PUT",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${this.token}`,
},
body: JSON.stringify(record),
}
);
var body = await response.json();
if(body.success !== true) {
throw new CloudflareApiException("Failed to update dns record");
}
return body.result[0];
};
}
}

addEventListener("fetch", (event) => {
event.respondWith(
handleRequest(event.request).catch((err) => {
console.error(err.constructor.name, err);
const message = err.reason || err.stack || "Unknown Error";

return new Response(message, {
status: err.status || 500,
statusText: err.statusText || null,
headers: {
"Content-Type": "text/plain;charset=UTF-8",
// Disables caching by default.
"Cache-Control": "no-store",
// Returns the "Content-Length" header for HTTP HEAD requests.
"Content-Length": message.length,
},
});
})
);
});
* Welcome to Cloudflare Workers! This is your first worker.
*
* - Run `npx wrangler dev src/index.js` in your terminal to start a development server
* - Open a browser tab at http://localhost:8787/ to see your worker in action
* - Run `npx wrangler publish src/index.js --name my-worker` to publish your worker
*
* Learn more at https://developers.cloudflare.com/workers/
*/

export default {
async fetch(request, env, ctx) {
return new Response("Hello World!");
},
};
23 changes: 23 additions & 0 deletions index.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
const { unstable_dev } = require("wrangler");

describe("Worker", () => {
let worker;

beforeAll(async () => {
worker = await unstable_dev("src/index.js", {
experimental: { disableExperimentalWarning: true },
});
});

afterAll(async () => {
await worker.stop();
});

it("should return Hello World", async () => {
const resp = await worker.fetch();
if (resp) {
const text = await resp.text();
expect(text).toMatchInlineSnapshot(`"Hello World!"`);
}
});
});
Loading

0 comments on commit b452d2d

Please sign in to comment.