mirror of
https://github.com/discourse/discourse.git
synced 2025-04-03 05:39:41 +08:00
DEV: Add --forward-host
option to bin/ember-cli
(#17244)
This allows to e.g. test multisite setup in a local dev environment. Also fixes some minor proxy issues.
This commit is contained in:
parent
39a025c7af
commit
16f22e3c36
@ -8,6 +8,8 @@ const path = require("path");
|
|||||||
const { promises: fs } = require("fs");
|
const { promises: fs } = require("fs");
|
||||||
const { JSDOM } = require("jsdom");
|
const { JSDOM } = require("jsdom");
|
||||||
const { shouldLoadPluginTestJs } = require("discourse/lib/plugin-js");
|
const { shouldLoadPluginTestJs } = require("discourse/lib/plugin-js");
|
||||||
|
const { Buffer } = require("node:buffer");
|
||||||
|
const { cwd, env } = require("node:process");
|
||||||
|
|
||||||
// via https://stackoverflow.com/a/6248722/165668
|
// via https://stackoverflow.com/a/6248722/165668
|
||||||
function generateUID() {
|
function generateUID() {
|
||||||
@ -202,7 +204,7 @@ async function applyBootstrap(bootstrap, template, response, baseURL, preload) {
|
|||||||
async function buildFromBootstrap(proxy, baseURL, req, response, preload) {
|
async function buildFromBootstrap(proxy, baseURL, req, response, preload) {
|
||||||
try {
|
try {
|
||||||
const template = await fs.readFile(
|
const template = await fs.readFile(
|
||||||
path.join(process.cwd(), "dist", "index.html"),
|
path.join(cwd(), "dist", "index.html"),
|
||||||
"utf8"
|
"utf8"
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -221,8 +223,23 @@ async function buildFromBootstrap(proxy, baseURL, req, response, preload) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function handleRequest(proxy, baseURL, req, res) {
|
async function handleRequest(proxy, baseURL, req, res) {
|
||||||
const originalHost = req.headers["x-forwarded-host"] || req.headers.host;
|
// x-forwarded-host is used in e.g. GitHub CodeSpaces
|
||||||
|
let originalHost = req.headers["x-forwarded-host"] || req.headers.host;
|
||||||
|
|
||||||
|
if (env["FORWARD_HOST"] === "true") {
|
||||||
|
if (/^localhost(\:|$)/.test(originalHost)) {
|
||||||
|
// Can't access default site in multisite via "localhost", redirect to 127.0.0.1
|
||||||
|
res.redirect(
|
||||||
|
307,
|
||||||
|
`http://${originalHost.replace("localhost", "127.0.0.1")}${req.path}`
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
req.headers.host = originalHost;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
req.headers.host = new URL(proxy).host;
|
req.headers.host = new URL(proxy).host;
|
||||||
|
}
|
||||||
|
|
||||||
if (req.headers["Origin"]) {
|
if (req.headers["Origin"]) {
|
||||||
req.headers["Origin"] = req.headers["Origin"]
|
req.headers["Origin"] = req.headers["Origin"]
|
||||||
@ -270,18 +287,21 @@ async function handleRequest(proxy, baseURL, req, res) {
|
|||||||
`http://${originalHost}/assets/`,
|
`http://${originalHost}/assets/`,
|
||||||
`http://${originalHost}/ember-cli-live-reload.js`,
|
`http://${originalHost}/ember-cli-live-reload.js`,
|
||||||
`http://${originalHost}/_lr/`,
|
`http://${originalHost}/_lr/`,
|
||||||
];
|
].join(" ");
|
||||||
|
|
||||||
const newCSP = csp
|
const newCSP = csp
|
||||||
.replace(new RegExp(proxy, "g"), `http://${originalHost}`)
|
.replaceAll(proxy, `http://${originalHost}`)
|
||||||
.replace(
|
.replaceAll("script-src ", `script-src ${emberCliAdditions}`);
|
||||||
new RegExp("script-src ", "g"),
|
|
||||||
`script-src ${emberCliAdditions.join(" ")} `
|
|
||||||
);
|
|
||||||
res.set("content-security-policy", newCSP);
|
res.set("content-security-policy", newCSP);
|
||||||
}
|
}
|
||||||
|
|
||||||
const contentType = response.headers.get("content-type");
|
const contentType = response.headers.get("content-type");
|
||||||
const isHTML = contentType && contentType.startsWith("text/html");
|
const isHTML = contentType?.startsWith("text/html");
|
||||||
|
|
||||||
|
res.status(response.status);
|
||||||
|
|
||||||
|
if (isHTML) {
|
||||||
const responseText = await response.text();
|
const responseText = await response.text();
|
||||||
const preloadJson = isHTML ? extractPreloadJson(responseText) : null;
|
const preloadJson = isHTML ? extractPreloadJson(responseText) : null;
|
||||||
|
|
||||||
@ -296,9 +316,11 @@ async function handleRequest(proxy, baseURL, req, res) {
|
|||||||
res.set("content-type", "text/html");
|
res.set("content-type", "text/html");
|
||||||
res.send(html);
|
res.send(html);
|
||||||
} else {
|
} else {
|
||||||
res.status(response.status);
|
|
||||||
res.send(responseText);
|
res.send(responseText);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
res.send(Buffer.from(await response.arrayBuffer()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
@ -308,7 +330,7 @@ module.exports = {
|
|||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
contentFor: function (type, config) {
|
contentFor(type, config) {
|
||||||
if (shouldLoadPluginTestJs() && type === "test-plugin-js") {
|
if (shouldLoadPluginTestJs() && type === "test-plugin-js") {
|
||||||
return `
|
return `
|
||||||
<script src="${config.rootURL}assets/discourse/tests/active-plugins.js"></script>
|
<script src="${config.rootURL}assets/discourse/tests/active-plugins.js"></script>
|
||||||
@ -339,8 +361,11 @@ to serve API requests. For example:
|
|||||||
|
|
||||||
app.use(rawMiddleware, async (req, res, next) => {
|
app.use(rawMiddleware, async (req, res, next) => {
|
||||||
try {
|
try {
|
||||||
if (this.shouldHandleRequest(req)) {
|
if (this.shouldForwardRequest(req)) {
|
||||||
await handleRequest(proxy, baseURL, req, res);
|
await handleRequest(proxy, baseURL, req, res);
|
||||||
|
} else {
|
||||||
|
// Fixes issues when using e.g. "localhost" instead of loopback IP address
|
||||||
|
req.headers.host = "127.0.0.1";
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
res.send(`
|
res.send(`
|
||||||
@ -357,28 +382,19 @@ to serve API requests. For example:
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
shouldHandleRequest(request) {
|
shouldForwardRequest(request) {
|
||||||
if (request.path === "/tests/index.html") {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (request.get("Accept") && request.get("Accept").includes("text/html")) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const contentType = request.get("Content-Type");
|
|
||||||
if (!contentType) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
contentType.includes("application/x-www-form-urlencoded") ||
|
["/tests/index.html", "/ember-cli-live-reload.js", "/testem.js"].includes(
|
||||||
contentType.includes("multipart/form-data") ||
|
request.path
|
||||||
contentType.includes("application/json")
|
)
|
||||||
) {
|
) {
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (request.path.startsWith("/_lr/")) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -6,8 +6,8 @@ require 'pathname'
|
|||||||
RAILS_ROOT = File.expand_path("../../", Pathname.new(__FILE__).realpath)
|
RAILS_ROOT = File.expand_path("../../", Pathname.new(__FILE__).realpath)
|
||||||
PORT = ENV["UNICORN_PORT"] ||= "3000"
|
PORT = ENV["UNICORN_PORT"] ||= "3000"
|
||||||
HOSTNAME = ENV["DISCOURSE_HOSTNAME"] ||= "127.0.0.1"
|
HOSTNAME = ENV["DISCOURSE_HOSTNAME"] ||= "127.0.0.1"
|
||||||
yarn_dir = File.join(RAILS_ROOT, "app/assets/javascripts/discourse")
|
YARN_DIR = File.join(RAILS_ROOT, "app/assets/javascripts/discourse")
|
||||||
|
CUSTOM_ARGS = ["--try", "--test", "--unicorn", "-u", "--forward-host"]
|
||||||
PROXY =
|
PROXY =
|
||||||
if ARGV.include?("--try")
|
if ARGV.include?("--try")
|
||||||
"https://try.discourse.org"
|
"https://try.discourse.org"
|
||||||
@ -38,19 +38,22 @@ if ARGV.include?("-h") || ARGV.include?("--help")
|
|||||||
puts "#{"--test".cyan} To run the test suite"
|
puts "#{"--test".cyan} To run the test suite"
|
||||||
puts "#{"--unicorn, -u".cyan} To run a unicorn server as well"
|
puts "#{"--unicorn, -u".cyan} To run a unicorn server as well"
|
||||||
puts "The rest of the arguments are passed to ember server per:", ""
|
puts "The rest of the arguments are passed to ember server per:", ""
|
||||||
exec "yarn -s --cwd #{yarn_dir} run ember #{command} --help"
|
exec "yarn -s --cwd #{YARN_DIR} run ember #{command} --help"
|
||||||
end
|
end
|
||||||
|
|
||||||
args = ["-s", "--cwd", yarn_dir, "run", "ember", command] + ARGV.reject do |a|
|
args = ["-s", "--cwd", YARN_DIR, "run", "ember", command] + (ARGV - CUSTOM_ARGS)
|
||||||
["--try", "--test", "--unicorn", "-u"].include?(a)
|
|
||||||
end
|
|
||||||
|
|
||||||
if !args.include?("test") && !args.include?("--proxy")
|
if !args.include?("test") && !args.include?("--proxy")
|
||||||
args << "--proxy"
|
args << "--proxy"
|
||||||
args << PROXY
|
args << PROXY
|
||||||
end
|
end
|
||||||
|
|
||||||
exit 1 if !system "yarn -s install --cwd #{yarn_dir}"
|
exit 1 if !system "yarn -s install --cwd #{YARN_DIR}"
|
||||||
|
|
||||||
|
yarn_env = {}
|
||||||
|
if ARGV.include?("--forward-host")
|
||||||
|
yarn_env["FORWARD_HOST"] = "true"
|
||||||
|
end
|
||||||
|
|
||||||
if ARGV.include?("-u") || ARGV.include?("--unicorn")
|
if ARGV.include?("-u") || ARGV.include?("--unicorn")
|
||||||
unicorn_env = { "DISCOURSE_PORT" => ENV["DISCOURSE_PORT"] || "4200" }
|
unicorn_env = { "DISCOURSE_PORT" => ENV["DISCOURSE_PORT"] || "4200" }
|
||||||
@ -58,7 +61,7 @@ if ARGV.include?("-u") || ARGV.include?("--unicorn")
|
|||||||
|
|
||||||
Thread.new do
|
Thread.new do
|
||||||
require 'open3'
|
require 'open3'
|
||||||
Open3.popen2e("yarn", *args.to_a.flatten) do |i, oe, t|
|
Open3.popen2e(yarn_env, "yarn", *args.to_a.flatten) do |i, oe, t|
|
||||||
puts "Ember CLI running on PID: #{t.pid}"
|
puts "Ember CLI running on PID: #{t.pid}"
|
||||||
oe.each do |line|
|
oe.each do |line|
|
||||||
if line.include?("\e[32m200\e") || line.include?("\e[36m304\e[0m") || line.include?("POST /message-bus")
|
if line.include?("\e[32m200\e") || line.include?("\e[36m304\e[0m") || line.include?("POST /message-bus")
|
||||||
@ -77,5 +80,5 @@ if ARGV.include?("-u") || ARGV.include?("--unicorn")
|
|||||||
|
|
||||||
Process.wait(unicorn_pid)
|
Process.wait(unicorn_pid)
|
||||||
else
|
else
|
||||||
exec "yarn", *args.to_a.flatten
|
exec(yarn_env, "yarn", *args.to_a.flatten)
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user