mirror of
https://github.com/LittleChest/Dohna-NS.git
synced 2026-05-06 22:44:50 +08:00
common: Handler 2.0
This commit is contained in:
parent
c7c9885752
commit
26f610e00c
239
common.js
Normal file
239
common.js
Normal file
@ -0,0 +1,239 @@
|
||||
const dns = "https://dns.google/dns-query";
|
||||
const api = "https://dns.google/resolve";
|
||||
|
||||
export default async function handler(request) {
|
||||
const { method, headers, url } = request;
|
||||
const { search, searchParams, pathname } = new URL(url);
|
||||
const ip = headers.get("x-forwarded-for");
|
||||
|
||||
// Anti-GFW
|
||||
if (
|
||||
headers.get("user-agent") === "Go-http-client/1.1" &&
|
||||
pathname === "/dns-query" &&
|
||||
method === "POST"
|
||||
) {
|
||||
return new Response(null, { status: 403 });
|
||||
}
|
||||
|
||||
let res = new Response(null, { status: 404 });
|
||||
|
||||
// JSON API
|
||||
if (pathname === "/resolve") {
|
||||
res = new Response(null, { status: 400 });
|
||||
|
||||
if (method === "GET" && searchParams.has("name")) {
|
||||
res = fetch(api + search, {
|
||||
method: "GET",
|
||||
headers: {
|
||||
Accept: "application/dns-json",
|
||||
},
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// DNS Query
|
||||
if (pathname === "/dns-query") {
|
||||
res = new Response(null, { status: 400 });
|
||||
|
||||
let queryData;
|
||||
|
||||
// GET
|
||||
if (method === "GET" && searchParams.has("dns")) {
|
||||
// Decode the base64-encoded DNS query
|
||||
const decodedQuery = atob(searchParams.get("dns"));
|
||||
queryData = new Uint8Array(decodedQuery.length);
|
||||
for (let i = 0; i < decodedQuery.length; i++) {
|
||||
queryData[i] = decodedQuery.charCodeAt(i);
|
||||
}
|
||||
}
|
||||
|
||||
// POST
|
||||
if (
|
||||
method === "POST" &&
|
||||
headers.get("content-type") === "application/dns-message"
|
||||
) {
|
||||
const originalQuery = await request.arrayBuffer();
|
||||
queryData = new Uint8Array(originalQuery);
|
||||
}
|
||||
|
||||
if (queryData !== undefined) {
|
||||
res = await queryDns(queryData, ip);
|
||||
}
|
||||
}
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
async function queryDns(queryData, ip) {
|
||||
const hasOptRecord = checkForOptRecord(queryData);
|
||||
let newQueryData = queryData;
|
||||
if (!hasOptRecord && ip) {
|
||||
// Extract DNS Header and Question Section
|
||||
const [headerAndQuestion] = extractHeaderAndQuestion(queryData);
|
||||
|
||||
// Construct a new OPT record with ECS option
|
||||
const optRecord = createOptRecord(ip);
|
||||
|
||||
// Combine the header, question, and new OPT record to create a new query
|
||||
newQueryData = combineQueryData(headerAndQuestion, optRecord);
|
||||
}
|
||||
|
||||
const res = await fetch(dns, {
|
||||
method: "POST",
|
||||
headers: {
|
||||
"Content-Type": "application/dns-message",
|
||||
},
|
||||
body: newQueryData,
|
||||
});
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
function checkForOptRecord(data) {
|
||||
// Get the number of additional records (ARCOUNT)
|
||||
const arcount = (data[10] << 8) | data[11];
|
||||
if (arcount === 0) return false;
|
||||
|
||||
let offset = 12; // DNS header is 12 bytes
|
||||
|
||||
// Skip the Question Section
|
||||
const qdcount = (data[4] << 8) | data[5];
|
||||
for (let i = 0; i < qdcount; i++) {
|
||||
while (data[offset] !== 0) offset++; // Skip QNAME
|
||||
offset += 5; // Skip QNAME (0 byte) + QTYPE (2 bytes) + QCLASS (2 bytes)
|
||||
}
|
||||
|
||||
// Skip the Answer Section
|
||||
const ancount = (data[6] << 8) | data[7];
|
||||
for (let i = 0; i < ancount; i++) {
|
||||
// Skip each Answer record
|
||||
while (data[offset] !== 0) offset++;
|
||||
offset += 10; // TYPE(2) + CLASS(2) + TTL(4) + RDLENGTH(2)
|
||||
const rdlength = (data[offset - 2] << 8) | data[offset - 1];
|
||||
offset += rdlength;
|
||||
}
|
||||
|
||||
// Check Additional Section for OPT record
|
||||
for (let i = 0; i < arcount; i++) {
|
||||
if (data[offset] === 0) {
|
||||
// OPT record NAME must be root (0)
|
||||
const type = (data[offset + 1] << 8) | data[offset + 2];
|
||||
if (type === 41) {
|
||||
// 41 is the OPT record type
|
||||
return true;
|
||||
}
|
||||
}
|
||||
// Skip this additional record
|
||||
while (data[offset] !== 0) offset++;
|
||||
offset += 10;
|
||||
const rdlength = (data[offset - 2] << 8) | data[offset - 1];
|
||||
offset += rdlength;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
function extractHeaderAndQuestion(data) {
|
||||
let offset = 12; // DNS header is 12 bytes
|
||||
|
||||
// Get the number of questions
|
||||
const qdcount = (data[4] << 8) | data[5];
|
||||
|
||||
// Skip the Question Section
|
||||
for (let i = 0; i < qdcount; i++) {
|
||||
while (data[offset] !== 0) offset++; // Skip QNAME
|
||||
offset += 5; // Skip QNAME (0 byte) + QTYPE (2 bytes) + QCLASS (2 bytes)
|
||||
}
|
||||
|
||||
// Extract Header and Question Section
|
||||
const headerAndQuestion = data.subarray(0, offset);
|
||||
|
||||
return [headerAndQuestion, offset];
|
||||
}
|
||||
|
||||
function createOptRecord(ip) {
|
||||
let ecsData;
|
||||
let family;
|
||||
|
||||
if (isIPv4(ip)) {
|
||||
// Convert client IP to bytes
|
||||
const ipParts = ip.split(".").map((part) => parseInt(part, 10));
|
||||
family = 1; // IPv4
|
||||
const prefixLength = 32; // Adjust the prefix length as needed
|
||||
ecsData = [0, 8, 0, 8, 0, family, prefixLength, 0, ...ipParts];
|
||||
} else if (isIPv6(ip)) {
|
||||
// Convert client IP to bytes
|
||||
const ipParts = ipv6ToBytes(ip);
|
||||
family = 2; // IPv6
|
||||
const prefixLength = 128; // Adjust the prefix length as needed
|
||||
ecsData = [0, 8, 0, 20, 0, family, prefixLength, 0, ...ipParts];
|
||||
} else {
|
||||
throw new Error("Invalid IP address");
|
||||
}
|
||||
|
||||
// Construct the OPT record
|
||||
return new Uint8Array([
|
||||
0, // Name (root)
|
||||
0,
|
||||
41, // Type: OPT
|
||||
16,
|
||||
0, // UDP payload size (default 4096)
|
||||
0,
|
||||
0,
|
||||
0,
|
||||
0, // Extended RCODE and flags
|
||||
0,
|
||||
ecsData.length, // RD Length
|
||||
...ecsData,
|
||||
]);
|
||||
}
|
||||
|
||||
function isIPv4(ip) {
|
||||
return ip.split(".").length === 4;
|
||||
}
|
||||
|
||||
function isIPv6(ip) {
|
||||
return ip.split(":").length > 2; // At least 3 groups separated by colons
|
||||
}
|
||||
|
||||
function ipv6ToBytes(ipv6) {
|
||||
// Split the IPv6 address into segments
|
||||
let segments = ipv6.split(":");
|
||||
|
||||
// Expand shorthand notation (e.g., '::')
|
||||
let expandedSegments = [];
|
||||
for (let i = 0; i < segments.length; i++) {
|
||||
if (segments[i] === "") {
|
||||
// Insert zero segments for "::"
|
||||
let zeroSegments = 8 - (segments.length - 1);
|
||||
expandedSegments.push(...new Array(zeroSegments).fill("0000"));
|
||||
} else {
|
||||
expandedSegments.push(segments[i]);
|
||||
}
|
||||
}
|
||||
|
||||
// Convert each segment into a 16-bit number and then into 8-bit numbers
|
||||
let bytes = [];
|
||||
for (let segment of expandedSegments) {
|
||||
let segmentValue = parseInt(segment, 16);
|
||||
bytes.push((segmentValue >> 8) & 0xff); // High byte
|
||||
bytes.push(segmentValue & 0xff); // Low byte
|
||||
}
|
||||
|
||||
return bytes;
|
||||
}
|
||||
|
||||
function combineQueryData(headerAndQuestion, optRecord) {
|
||||
// Combine the header and question section with the new OPT record
|
||||
const newQueryData = new Uint8Array(
|
||||
headerAndQuestion.length + optRecord.length
|
||||
);
|
||||
newQueryData.set(headerAndQuestion, 0);
|
||||
newQueryData.set(optRecord, headerAndQuestion.length);
|
||||
// https://en.wikipedia.org/wiki/Domain_Name_System#DNS_message_format
|
||||
// Incrementing the QDCOUNT field (offset 3) to 32, signaling an additional record in the question section.
|
||||
// Setting the ARCOUNT field (offset 11) to 1, indicating one additional record in the message.
|
||||
newQueryData.set([32], 3);
|
||||
newQueryData.set([1], 11);
|
||||
return newQueryData;
|
||||
}
|
||||
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "dohna-ns",
|
||||
"version": "1.0.0",
|
||||
"version": "2.0.0",
|
||||
"description": "Another DNS over HTTPS recursive resolver.",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user