Compare commits
33 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| fe21222d33 | |||
| f1d33f3f63 | |||
| 1205705555 | |||
| 4259af79bd | |||
| 05ebb904b2 | |||
| 73200e8c27 | |||
| 27ff8f19c7 | |||
| 8124698775 | |||
| 58890a7c9d | |||
| 521b2adf67 | |||
| 84db9ecc9e | |||
| b75ca64d22 | |||
| 8fee384fc0 | |||
| 8c2db8fdc5 | |||
| bda105ceeb | |||
| 4e4e034db2 | |||
| b8047a7988 | |||
| ef85f6782d | |||
| 46a14f0e46 | |||
| ed1230eee8 | |||
| 334393b9de | |||
| 684f3f70dd | |||
| 70b5a4979d | |||
| 1898808f79 | |||
| df2c426d76 | |||
| f04e9484e7 | |||
| 3fa1ad7269 | |||
| 66cc550949 | |||
| 6dd9a0213c | |||
| d993157cfa | |||
| 06237b1b8b | |||
| 50a25dd04c | |||
| 22ad0e9289 |
+8
-6
@@ -1,18 +1,20 @@
|
|||||||
# Stage 1: Build
|
# Stage 1: Build
|
||||||
FROM node:20-alpine AS build
|
FROM node:20.2-alpine AS build
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY package.json yarn.lock ./
|
COPY package.json yarn.lock ./
|
||||||
RUN yarn install --frozen-lockfile && yarn cache clean
|
RUN yarn install
|
||||||
COPY . .
|
COPY . .
|
||||||
|
RUN yarn build
|
||||||
|
|
||||||
# Stage 2: Production
|
# Stage 2: Production
|
||||||
FROM nginx:alpine
|
FROM nginx:alpine
|
||||||
COPY --from=build /app/build /usr/share/nginx/html
|
WORKDIR /etc/nginx
|
||||||
COPY nginx.conf /etc/nginx/nginx.conf
|
COPY ./nginx.conf /etc/nginx/nginx.conf
|
||||||
|
WORKDIR /usr/share/nginx/html
|
||||||
|
COPY --from=build /app/build .
|
||||||
|
|
||||||
# Expose ports 80 and 443
|
# Expose port 80
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
EXPOSE 443
|
|
||||||
|
|
||||||
# Healthcheck
|
# Healthcheck
|
||||||
HEALTHCHECK CMD curl --fail http://localhost:80 || exit 1
|
HEALTHCHECK CMD curl --fail http://localhost:80 || exit 1
|
||||||
|
|||||||
@@ -94,11 +94,10 @@ version: '3.8'
|
|||||||
|
|
||||||
services:
|
services:
|
||||||
matrix-to:
|
matrix-to:
|
||||||
container_name: Matrix-to
|
container_name: Matrix-to
|
||||||
image: domoel/matrix-to:latest
|
image: domoel/matrix-to:latest
|
||||||
ports:
|
ports:
|
||||||
- "5000:5000"
|
- "1336:80"
|
||||||
restart: unless-stopped
|
|
||||||
environment:
|
environment:
|
||||||
- PORT=5000
|
- NODE_ENV=production
|
||||||
```
|
```
|
||||||
|
|||||||
+3
-9
@@ -1,16 +1,10 @@
|
|||||||
version: '3.8'
|
version: '3.8'
|
||||||
|
|
||||||
services:
|
services:
|
||||||
web:
|
matrix-to:
|
||||||
|
container_name: Matrix-to
|
||||||
image: domoel/matrix-to:latest
|
image: domoel/matrix-to:latest
|
||||||
ports:
|
ports:
|
||||||
- "80:80"
|
- "1336:80"
|
||||||
environment:
|
environment:
|
||||||
- NODE_ENV=production
|
- NODE_ENV=production
|
||||||
volumes:
|
|
||||||
- ./nginx.conf:/etc/nginx/nginx.conf
|
|
||||||
healthcheck:
|
|
||||||
test: ["CMD", "curl", "--fail", "http://localhost:80"]
|
|
||||||
interval: 30s
|
|
||||||
timeout: 10s
|
|
||||||
retries: 3
|
|
||||||
|
|||||||
+34
@@ -0,0 +1,34 @@
|
|||||||
|
worker_processes 1;
|
||||||
|
|
||||||
|
events {
|
||||||
|
worker_connections 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
http {
|
||||||
|
include mime.types;
|
||||||
|
default_type application/octet-stream;
|
||||||
|
|
||||||
|
sendfile on;
|
||||||
|
keepalive_timeout 65;
|
||||||
|
|
||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
server_name localhost;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
index index.html index.htm;
|
||||||
|
try_files $uri $uri/ /index.html;
|
||||||
|
}
|
||||||
|
|
||||||
|
error_page 404 /404.html;
|
||||||
|
location = /404.html {
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
}
|
||||||
|
|
||||||
|
error_page 500 502 503 504 /50x.html;
|
||||||
|
location = /50x.html {
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
+18
-1
@@ -25,6 +25,7 @@ export class Preferences extends EventEmitter {
|
|||||||
// used to differentiate web from native if a client supports both
|
// used to differentiate web from native if a client supports both
|
||||||
this.platform = null;
|
this.platform = null;
|
||||||
this.homeservers = null;
|
this.homeservers = null;
|
||||||
|
this.customWebInstances = {};
|
||||||
|
|
||||||
const prefsStr = localStorage.getItem("preferred_client");
|
const prefsStr = localStorage.getItem("preferred_client");
|
||||||
if (prefsStr) {
|
if (prefsStr) {
|
||||||
@@ -36,6 +37,10 @@ export class Preferences extends EventEmitter {
|
|||||||
if (serversStr) {
|
if (serversStr) {
|
||||||
this.homeservers = JSON.parse(serversStr);
|
this.homeservers = JSON.parse(serversStr);
|
||||||
}
|
}
|
||||||
|
const customWebInstancesStr = localStorage.getItem("custom_web_instances");
|
||||||
|
if (customWebInstancesStr) {
|
||||||
|
this.customWebInstances = JSON.parse(customWebInstancesStr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setClient(id, platform) {
|
setClient(id, platform) {
|
||||||
@@ -54,15 +59,27 @@ export class Preferences extends EventEmitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setCustomWebInstance(client_id, instance_url) {
|
||||||
|
this.customWebInstances[client_id] = instance_url;
|
||||||
|
this._localStorage.setItem("custom_web_instances", JSON.stringify(this.customWebInstances));
|
||||||
|
this.emit("canClear");
|
||||||
|
}
|
||||||
|
|
||||||
|
getCustomWebInstance(client_id) {
|
||||||
|
return this.customWebInstances[client_id];
|
||||||
|
}
|
||||||
|
|
||||||
clear() {
|
clear() {
|
||||||
this._localStorage.removeItem("preferred_client");
|
this._localStorage.removeItem("preferred_client");
|
||||||
this._localStorage.removeItem("consented_servers");
|
this._localStorage.removeItem("consented_servers");
|
||||||
|
this._localStorage.removeItem("custom_web_instances");
|
||||||
this.clientId = null;
|
this.clientId = null;
|
||||||
this.platform = null;
|
this.platform = null;
|
||||||
this.homeservers = null;
|
this.homeservers = null;
|
||||||
|
this.customWebInstances = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
get canClear() {
|
get canClear() {
|
||||||
return !!this.clientId || !!this.platform || !!this.homeservers;
|
return !!this.clientId || !!this.platform || !!this.homeservers || !!this.customWebInstances;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,6 +39,14 @@ function renderInstructions(parts) {
|
|||||||
export class ClientView extends TemplateView {
|
export class ClientView extends TemplateView {
|
||||||
|
|
||||||
render(t, vm) {
|
render(t, vm) {
|
||||||
|
return t.mapView(vm => vm.customWebInstanceFormOpen, open => {
|
||||||
|
switch (open) {
|
||||||
|
case true: return new SetCustomWebInstanceView(vm);
|
||||||
|
case false: return new TemplateView(vm, t => this.renderContent(t, vm));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
renderContent(t, vm) {
|
||||||
return t.div({className: {"ClientView": true, "isPreferred": vm => vm.hasPreferredWebInstance}}, [
|
return t.div({className: {"ClientView": true, "isPreferred": vm => vm.hasPreferredWebInstance}}, [
|
||||||
... vm.hasPreferredWebInstance ? [t.div({className: "hostedBanner"}, vm.hostedByBannerLabel)] : [],
|
... vm.hasPreferredWebInstance ? [t.div({className: "hostedBanner"}, vm.hostedByBannerLabel)] : [],
|
||||||
t.div({className: "header"}, [
|
t.div({className: "header"}, [
|
||||||
@@ -112,10 +120,49 @@ class InstallClientView extends TemplateView {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class SetCustomWebInstanceView extends TemplateView {
|
||||||
|
render(t, vm) {
|
||||||
|
return t.div({className: "SetCustomWebInstanceView"}, [
|
||||||
|
t.p([
|
||||||
|
"Use a custom web instance for the ", t.strong(vm.name), " client:",
|
||||||
|
]),
|
||||||
|
t.form({action: "#", id: "setCustomWebInstanceForm", onSubmit: evt => this._onSubmit(evt)}, [
|
||||||
|
t.input({
|
||||||
|
type: "text",
|
||||||
|
className: "fullwidth large",
|
||||||
|
placeholder: "chat.example.org",
|
||||||
|
name: "instanceHostname",
|
||||||
|
value: vm.preferredWebInstance || "",
|
||||||
|
}),
|
||||||
|
t.input({type: "submit", value: "Save", className: "primary fullwidth"}),
|
||||||
|
t.input({type: "button", value: "Use Default Instance", className: "secondary fullwidth", onClick: evt => this._onReset(evt)}),
|
||||||
|
])
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onSubmit(evt) {
|
||||||
|
evt.preventDefault();
|
||||||
|
const form = evt.target;
|
||||||
|
const {instanceHostname} = form.elements;
|
||||||
|
this.value.setCustomWebInstance(instanceHostname.value);
|
||||||
|
this.value.closeCustomWebInstanceForm();
|
||||||
|
}
|
||||||
|
|
||||||
|
_onReset(evt) {
|
||||||
|
this.value.setCustomWebInstance(undefined);
|
||||||
|
this.value.closeCustomWebInstanceForm();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function showBack(t, vm) {
|
function showBack(t, vm) {
|
||||||
return t.p({className: {caption: true, "back": true, hidden: vm => !vm.showBack}}, [
|
return t.p({className: {caption: true, "back": true, hidden: vm => !vm.showBack}}, [
|
||||||
`Continue with ${vm.name} · `,
|
`Continue with ${vm.name} · `,
|
||||||
t.button({className: "text", onClick: () => vm.back()}, "Change"),
|
t.button({className: "text", onClick: () => vm.back()}, "Change"),
|
||||||
|
t.span({hidden: vm => !vm.supportsCustomWebInstances}, [
|
||||||
|
' · ',
|
||||||
|
t.button({className: "text", onClick: () => vm.configureCustomWebInstance()}, "Use Custom Web Instance"),
|
||||||
|
])
|
||||||
|
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+46
-13
@@ -35,6 +35,7 @@ export class ClientViewModel extends ViewModel {
|
|||||||
this._pickClient = pickClient;
|
this._pickClient = pickClient;
|
||||||
// to provide "choose other client" button after calling pick()
|
// to provide "choose other client" button after calling pick()
|
||||||
this._clientListViewModel = null;
|
this._clientListViewModel = null;
|
||||||
|
this.customWebInstanceFormOpen = false;
|
||||||
this._update();
|
this._update();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,11 +60,11 @@ export class ClientViewModel extends ViewModel {
|
|||||||
if (this._proposedPlatform === this._nativePlatform) {
|
if (this._proposedPlatform === this._nativePlatform) {
|
||||||
deepLinkLabel = "Open in app";
|
deepLinkLabel = "Open in app";
|
||||||
} else {
|
} else {
|
||||||
deepLinkLabel = `Open on ${this._client.getPreferredWebInstance(this._link)}`;
|
deepLinkLabel = `Open on ${this.preferredWebInstance}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const actions = [];
|
const actions = [];
|
||||||
const proposedDeepLink = this._client.getDeepLink(this._proposedPlatform, this._link);
|
const proposedDeepLink = this._client.getDeepLink(this._proposedPlatform, this._link, this.preferredWebInstance);
|
||||||
if (proposedDeepLink) {
|
if (proposedDeepLink) {
|
||||||
actions.push({
|
actions.push({
|
||||||
label: deepLinkLabel,
|
label: deepLinkLabel,
|
||||||
@@ -83,8 +84,8 @@ export class ClientViewModel extends ViewModel {
|
|||||||
// show only if there is a preferred instance, and if we don't already link to it in the first button
|
// show only if there is a preferred instance, and if we don't already link to it in the first button
|
||||||
if (hasPreferredWebInstance && this._webPlatform && this._proposedPlatform !== this._webPlatform) {
|
if (hasPreferredWebInstance && this._webPlatform && this._proposedPlatform !== this._webPlatform) {
|
||||||
actions.push({
|
actions.push({
|
||||||
label: `Open on ${this._client.getPreferredWebInstance(this._link)}`,
|
label: `Open on ${this.preferredWebInstance}`,
|
||||||
url: this._client.getDeepLink(this._webPlatform, this._link),
|
url: this._client.getDeepLink(this._webPlatform, this._link, this.preferredWebInstance),
|
||||||
kind: "open-in-web",
|
kind: "open-in-web",
|
||||||
activated: () => {} // don't persist this choice as we don't persist the preferred web instance, it's in the url
|
activated: () => {} // don't persist this choice as we don't persist the preferred web instance, it's in the url
|
||||||
});
|
});
|
||||||
@@ -108,10 +109,10 @@ export class ClientViewModel extends ViewModel {
|
|||||||
actions.push(...nativeActions);
|
actions.push(...nativeActions);
|
||||||
}
|
}
|
||||||
if (this._webPlatform) {
|
if (this._webPlatform) {
|
||||||
const webDeepLink = this._client.getDeepLink(this._webPlatform, this._link);
|
const webDeepLink = this._client.getDeepLink(this._webPlatform, this._link, this.preferredWebInstance);
|
||||||
if (webDeepLink) {
|
if (webDeepLink) {
|
||||||
const webLabel = this.hasPreferredWebInstance ?
|
const webLabel = this.hasPreferredWebInstance ?
|
||||||
`Open on ${this._client.getPreferredWebInstance(this._link)}` :
|
`Open on ${this.preferredWebInstance}` :
|
||||||
`Continue in your browser`;
|
`Continue in your browser`;
|
||||||
actions.push({
|
actions.push({
|
||||||
label: webLabel,
|
label: webLabel,
|
||||||
@@ -128,18 +129,26 @@ export class ClientViewModel extends ViewModel {
|
|||||||
return actions;
|
return actions;
|
||||||
}
|
}
|
||||||
|
|
||||||
get hasPreferredWebInstance() {
|
get preferredWebInstance() {
|
||||||
// also check there is a web platform that matches the platforms the user is on (mobile or desktop web)
|
// also check there is a web platform that matches the platforms the user is on (mobile or desktop web)
|
||||||
return this._webPlatform && typeof this._client.getPreferredWebInstance(this._link) === "string";
|
if (!this._webPlatform) return undefined;
|
||||||
|
return (
|
||||||
|
this.preferences.getCustomWebInstance(this._client.id)
|
||||||
|
|| this._client.getPreferredWebInstance(this._link)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
get hasPreferredWebInstance() {
|
||||||
|
return typeof this.preferredWebInstance === "string";
|
||||||
}
|
}
|
||||||
|
|
||||||
get hostedByBannerLabel() {
|
get hostedByBannerLabel() {
|
||||||
const preferredWebInstance = this._client.getPreferredWebInstance(this._link);
|
if (this.hasPreferredWebInstance) {
|
||||||
if (this._webPlatform && preferredWebInstance) {
|
const preferredWebInstance = this.preferredWebInstance;
|
||||||
let label = preferredWebInstance;
|
let label = preferredWebInstance;
|
||||||
const subDomainIdx = preferredWebInstance.lastIndexOf(".", preferredWebInstance.lastIndexOf("."));
|
const subDomainIdx = preferredWebInstance.lastIndexOf(".", preferredWebInstance.lastIndexOf(".") - 1);
|
||||||
if (subDomainIdx !== -1) {
|
if (subDomainIdx !== -1) {
|
||||||
label = preferredWebInstance.slice(preferredWebInstance.length - subDomainIdx + 1);
|
label = preferredWebInstance.slice(subDomainIdx + 1);
|
||||||
}
|
}
|
||||||
return `Hosted by ${label}`;
|
return `Hosted by ${label}`;
|
||||||
}
|
}
|
||||||
@@ -188,7 +197,7 @@ export class ClientViewModel extends ViewModel {
|
|||||||
|
|
||||||
get showDeepLinkInInstall() {
|
get showDeepLinkInInstall() {
|
||||||
// we can assume this._nativePlatform as this._clientCanIntercept already checks it
|
// we can assume this._nativePlatform as this._clientCanIntercept already checks it
|
||||||
return this._clientCanIntercept && !!this._client.getDeepLink(this._nativePlatform, this._link);
|
return this._clientCanIntercept && !!this._client.getDeepLink(this._nativePlatform, this._link, this.preferredWebInstance);
|
||||||
}
|
}
|
||||||
|
|
||||||
get availableOnPlatformNames() {
|
get availableOnPlatformNames() {
|
||||||
@@ -223,6 +232,10 @@ export class ClientViewModel extends ViewModel {
|
|||||||
return !!this._clientListViewModel;
|
return !!this._clientListViewModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get supportsCustomWebInstances() {
|
||||||
|
return !!this._client.supportsCustomInstances;
|
||||||
|
}
|
||||||
|
|
||||||
back() {
|
back() {
|
||||||
if (this._clientListViewModel) {
|
if (this._clientListViewModel) {
|
||||||
const vm = this._clientListViewModel;
|
const vm = this._clientListViewModel;
|
||||||
@@ -231,9 +244,29 @@ export class ClientViewModel extends ViewModel {
|
|||||||
// in the list with all clients, and also if we refresh, we get the list with
|
// in the list with all clients, and also if we refresh, we get the list with
|
||||||
// all clients rather than having our "change client" click reverted.
|
// all clients rather than having our "change client" click reverted.
|
||||||
this.preferences.setClient(undefined, undefined);
|
this.preferences.setClient(undefined, undefined);
|
||||||
|
this.preferences.setCustomWebInstance(this._client.id, undefined);
|
||||||
this._update();
|
this._update();
|
||||||
this.emitChange();
|
this.emitChange();
|
||||||
vm.showAll();
|
vm.showAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
configureCustomWebInstance() {
|
||||||
|
this.customWebInstanceFormOpen = true;
|
||||||
|
this.emitChange();
|
||||||
|
}
|
||||||
|
|
||||||
|
closeCustomWebInstanceForm() {
|
||||||
|
this.customWebInstanceFormOpen = false;
|
||||||
|
this.emitChange();
|
||||||
|
}
|
||||||
|
|
||||||
|
setCustomWebInstance(hostname) {
|
||||||
|
if (hostname) {
|
||||||
|
hostname = hostname.trim().replace(/^https:\/\//, '').replace(/\/.*$/, '');
|
||||||
|
}
|
||||||
|
this.preferences.setClient(this._client.id, hostname ? this._webPlatform : (this._nativePlatform || this._webPlatform));
|
||||||
|
this.preferences.setCustomWebInstance(this._client.id, hostname || undefined);
|
||||||
|
this._update();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -56,8 +56,9 @@ export class Element {
|
|||||||
get homepage() { return "https://element.io"; }
|
get homepage() { return "https://element.io"; }
|
||||||
get author() { return "Element"; }
|
get author() { return "Element"; }
|
||||||
getMaturity(platform) { return Maturity.Stable; }
|
getMaturity(platform) { return Maturity.Stable; }
|
||||||
|
get supportsCustomInstances() { return true; }
|
||||||
|
|
||||||
getDeepLink(platform, link) {
|
getDeepLink(platform, link, preferredWebInstance) {
|
||||||
let fragmentPath;
|
let fragmentPath;
|
||||||
switch (link.kind) {
|
switch (link.kind) {
|
||||||
case LinkKind.User:
|
case LinkKind.User:
|
||||||
@@ -83,8 +84,8 @@ export class Element {
|
|||||||
let instanceHost = trustedWebInstances[0];
|
let instanceHost = trustedWebInstances[0];
|
||||||
// we use app.element.io which iOS will intercept, but it likely won't intercept any other trusted instances
|
// we use app.element.io which iOS will intercept, but it likely won't intercept any other trusted instances
|
||||||
// so only use a preferred web instance for true web links.
|
// so only use a preferred web instance for true web links.
|
||||||
if (isWebPlatform && trustedWebInstances.includes(link.webInstances[this.id])) {
|
if (isWebPlatform && preferredWebInstance) {
|
||||||
instanceHost = link.webInstances[this.id];
|
instanceHost = preferredWebInstance;
|
||||||
}
|
}
|
||||||
return `https://${instanceHost}/#/${fragmentPath}`;
|
return `https://${instanceHost}/#/${fragmentPath}`;
|
||||||
} else if (platform === Platform.Linux || platform === Platform.Windows || platform === Platform.macOS) {
|
} else if (platform === Platform.Linux || platform === Platform.Windows || platform === Platform.macOS) {
|
||||||
|
|||||||
Reference in New Issue
Block a user