tasmota-esp-web-tools
Version:
Web tools for ESP devices
1,018 lines (948 loc) • 30.5 kB
text/typescript
import { LitElement, html, PropertyValues, css, TemplateResult } from "lit";
import { state } from "lit/decorators.js";
import "./components/ewt-button";
import "./components/ewt-checkbox";
import "./components/ewt-console";
import "./components/ewt-dialog";
import "./components/ewt-formfield";
import "./components/ewt-icon-button";
import "./components/ewt-textfield";
import type { EwtTextfield } from "./components/ewt-textfield";
import "./components/ewt-select";
import "./components/ewt-list-item";
import "./pages/ewt-page-progress";
import "./pages/ewt-page-message";
import { chipIcon, closeIcon, firmwareIcon } from "./components/svg";
import { Logger, Manifest, FlashStateType, FlashState } from "./const.js";
import { ImprovSerial, Ssid } from "improv-wifi-serial-sdk/dist/serial";
import {
ImprovSerialCurrentState,
ImprovSerialErrorState,
PortNotReady,
} from "improv-wifi-serial-sdk/dist/const";
import { flash } from "./flash";
import { textDownload } from "./util/file-download";
import { fireEvent } from "./util/fire-event";
import { sleep } from "./util/sleep";
import { downloadManifest } from "./util/manifest";
import { dialogStyles } from "./styles";
const ERROR_ICON = "⚠️";
const OK_ICON = "🎉";
export class EwtInstallDialog extends LitElement {
public port!: SerialPort;
public manifestPath!: string;
public firmwareFile?: File;
public logger: Logger = console;
public overrides?: {
checkSameFirmware?: (
manifest: Manifest,
deviceImprov: ImprovSerial["info"],
) => boolean;
};
private _manifest!: Manifest;
private _info?: ImprovSerial["info"];
// null = NOT_SUPPORTED
private _client?: ImprovSerial | null;
private _state:
| "ERROR"
| "DASHBOARD"
| "PROVISION"
| "INSTALL"
| "ASK_ERASE"
| "LOGS" = "DASHBOARD";
private _installErase = false;
private _installConfirmed = false;
private _installState?: FlashState;
private _provisionForce = false;
private _wasProvisioned = false;
private _error?: string;
private _busy = false;
// undefined = not loaded
// null = not available
private _ssids?: Ssid[] | null;
// -1 = custom
private _selectedSsid = -1;
protected render() {
if (!this.port) {
return html``;
}
let heading: string | undefined;
let content: TemplateResult;
let hideActions = false;
let allowClosing = false;
// During installation phase we temporarily remove the client
if (
this._client === undefined &&
this._state !== "INSTALL" &&
this._state !== "LOGS"
) {
if (this._error) {
[heading, content, hideActions] = this._renderError(this._error);
} else {
content = this._renderProgress("Connecting");
hideActions = true;
}
} else if (this._state === "INSTALL") {
[heading, content, hideActions, allowClosing] = this._renderInstall();
} else if (this._state === "ASK_ERASE") {
[heading, content] = this._renderAskErase();
} else if (this._state === "ERROR") {
[heading, content, hideActions] = this._renderError(this._error!);
} else if (this._state === "DASHBOARD") {
[heading, content, hideActions, allowClosing] = this._client
? this._renderDashboard()
: this._renderDashboardNoImprov();
} else if (this._state === "PROVISION") {
[heading, content, hideActions] = this._renderProvision();
} else if (this._state === "LOGS") {
[heading, content, hideActions] = this._renderLogs();
}
return html`
<ewt-dialog
open
.heading=${heading!}
scrimClickAction
=${this._handleClose}
.hideActions=${hideActions}
>
${heading && allowClosing
? html`
<ewt-icon-button dialogAction="close">
${closeIcon}
</ewt-icon-button>
`
: ""}
${content!}
</ewt-dialog>
`;
}
_renderProgress(label: string | TemplateResult, progress?: number) {
return html`
<ewt-page-progress
.label=${label}
.progress=${progress}
></ewt-page-progress>
`;
}
_renderError(label: string): [string, TemplateResult, boolean] {
const heading = "Error";
const content = html`
<ewt-page-message .icon=${ERROR_ICON} .label=${label}></ewt-page-message>
<ewt-button
slot="primaryAction"
dialogAction="ok"
label="Close"
></ewt-button>
`;
const hideActions = false;
return [heading, content, hideActions];
}
_renderDashboard(): [string, TemplateResult, boolean, boolean] {
const heading = this._info!.name;
let content: TemplateResult;
let hideActions = true;
let allowClosing = true;
content = html`
<div class="table-row">
${firmwareIcon}
<div>${this._info!.firmware} ${this._info!.version}</div>
</div>
<div class="table-row last">
${chipIcon}
<div>${this._info!.chipFamily}</div>
</div>
<div class="dashboard-buttons">
${!this._isSameVersion
? html`
<div>
<ewt-button
text-left
.label=${!this._isSameFirmware
? `Install ${this._manifest.name}`
: `Update ${this._manifest.name}`}
=${() => {
if (this._isSameFirmware) {
this._startInstall(false);
} else if (this._manifest.new_install_prompt_erase) {
this._state = "ASK_ERASE";
} else {
this._startInstall(true);
}
}}
></ewt-button>
</div>
`
: ""}
${this._client!.nextUrl === undefined
? ""
: html`
<div>
<a
href=${this._client!.nextUrl}
class="has-button"
target="_blank"
>
<ewt-button label="Visit Device"></ewt-button>
</a>
</div>
`}
${!this._manifest.home_assistant_domain ||
this._client!.state !== ImprovSerialCurrentState.PROVISIONED
? ""
: html`
<div>
<a
href=${`https://my.home-assistant.io/redirect/config_flow_start/?domain=${this._manifest.home_assistant_domain}`}
class="has-button"
target="_blank"
>
<ewt-button label="Add to Home Assistant"></ewt-button>
</a>
</div>
`}
<div>
<ewt-button
.label=${this._client!.state === ImprovSerialCurrentState.READY
? "Connect to Wi-Fi"
: "Change Wi-Fi"}
=${() => {
this._state = "PROVISION";
if (
this._client!.state === ImprovSerialCurrentState.PROVISIONED
) {
this._provisionForce = true;
}
}}
></ewt-button>
</div>
<div>
<ewt-button
label="Logs & Console"
=${async () => {
const client = this._client;
if (client) {
await this._closeClientWithoutEvents(client);
await sleep(100);
}
// Also set `null` back to undefined.
this._client = undefined;
this._state = "LOGS";
}}
></ewt-button>
</div>
${this._isSameFirmware && this._manifest.funding_url
? html`
<div>
<a
class="button"
href=${this._manifest.funding_url}
target="_blank"
>
<ewt-button label="Fund Development"></ewt-button>
</a>
</div>
`
: ""}
${this._isSameVersion
? html`
<div>
<ewt-button
class="danger"
label="Erase User Data"
=${() => this._startInstall(true)}
></ewt-button>
</div>
`
: ""}
</div>
`;
return [heading, content, hideActions, allowClosing];
}
_renderDashboardNoImprov(): [string, TemplateResult, boolean, boolean] {
const heading = "Device Dashboard";
let content: TemplateResult;
let hideActions = true;
let allowClosing = true;
content = html`
<div class="dashboard-buttons">
<div>
<ewt-button
text-left
.label=${`Install ${this._manifest.name}`}
=${() => {
if (this._manifest.new_install_prompt_erase) {
this._state = "ASK_ERASE";
} else {
// Default is to erase a device that does not support Improv Serial
this._startInstall(true);
}
}}
></ewt-button>
</div>
<div>
<ewt-button
label="Logs & Console"
=${async () => {
// Also set `null` back to undefined.
this._client = undefined;
this._state = "LOGS";
}}
></ewt-button>
</div>
</div>
`;
return [heading, content, hideActions, allowClosing];
}
_renderProvision(): [string | undefined, TemplateResult, boolean] {
let heading: string | undefined = "Configure Wi-Fi";
let content: TemplateResult;
let hideActions = false;
if (this._busy) {
return [
heading,
this._renderProgress(
this._ssids === undefined
? "Scanning for networks"
: "Trying to connect",
),
true,
];
}
if (
!this._provisionForce &&
this._client!.state === ImprovSerialCurrentState.PROVISIONED
) {
heading = undefined;
const showSetupLinks =
!this._wasProvisioned &&
(this._client!.nextUrl !== undefined ||
"home_assistant_domain" in this._manifest);
hideActions = showSetupLinks;
content = html`
<ewt-page-message
.icon=${OK_ICON}
label="Device connected to the network!"
></ewt-page-message>
${showSetupLinks
? html`
<div class="dashboard-buttons">
${this._client!.nextUrl === undefined
? ""
: html`
<div>
<a
href=${this._client!.nextUrl}
class="has-button"
target="_blank"
=${() => {
this._state = "DASHBOARD";
}}
>
<ewt-button label="Visit Device"></ewt-button>
</a>
</div>
`}
${!this._manifest.home_assistant_domain
? ""
: html`
<div>
<a
href=${`https://my.home-assistant.io/redirect/config_flow_start/?domain=${this._manifest.home_assistant_domain}`}
class="has-button"
target="_blank"
=${() => {
this._state = "DASHBOARD";
}}
>
<ewt-button
label="Add to Home Assistant"
></ewt-button>
</a>
</div>
`}
<div>
<ewt-button
label="Skip"
=${() => {
this._state = "DASHBOARD";
}}
></ewt-button>
</div>
</div>
`
: html`
<ewt-button
slot="primaryAction"
label="Continue"
=${() => {
this._state = "DASHBOARD";
}}
></ewt-button>
`}
`;
} else {
let error: string | undefined;
switch (this._client!.error) {
case ImprovSerialErrorState.UNABLE_TO_CONNECT:
error = "Unable to connect";
break;
case ImprovSerialErrorState.NO_ERROR:
// Happens when list SSIDs not supported.
case ImprovSerialErrorState.UNKNOWN_RPC_COMMAND:
break;
default:
error = `Unknown error (${this._client!.error})`;
}
content = html`
<div>
Enter the credentials of the Wi-Fi network that you want your device
to connect to.
</div>
${error ? html`<p class="error">${error}</p>` : ""}
${this._ssids !== null
? html`
<ewt-select
fixedMenuPosition
label="Network"
=${(ev: { detail: { index: number } }) => {
const index = ev.detail.index;
// The "Join Other" item is always the last item.
this._selectedSsid =
index === this._ssids!.length ? -1 : index;
}}
=${(ev: Event) => ev.stopPropagation()}
>
${this._ssids!.map(
(info, idx) => html`
<ewt-list-item
.selected=${this._selectedSsid === idx}
value=${idx}
>
${info.name}
</ewt-list-item>
`,
)}
<ewt-list-item
.selected=${this._selectedSsid === -1}
value="-1"
>
Join other…
</ewt-list-item>
</ewt-select>
`
: ""}
${
// Show input box if command not supported or "Join Other" selected
this._selectedSsid === -1
? html`
<ewt-textfield label="Network Name" name="ssid"></ewt-textfield>
`
: ""
}
<ewt-textfield
label="Password"
name="password"
type="password"
></ewt-textfield>
<ewt-button
slot="primaryAction"
label="Connect"
=${this._doProvision}
></ewt-button>
<ewt-button
slot="secondaryAction"
.label=${this._installState && this._installErase ? "Skip" : "Back"}
=${() => {
this._state = "DASHBOARD";
}}
></ewt-button>
`;
}
return [heading, content, hideActions];
}
_renderAskErase(): [string | undefined, TemplateResult] {
const heading = "Erase device";
const content = html`
<div>
Do you want to erase the device before installing
${this._manifest.name}? All data on the device will be lost.
</div>
<ewt-formfield label="Erase device" class="danger">
<ewt-checkbox></ewt-checkbox>
</ewt-formfield>
<ewt-button
slot="primaryAction"
label="Next"
=${() => {
const checkbox = this.shadowRoot!.querySelector("ewt-checkbox")!;
this._startInstall(checkbox.checked);
}}
></ewt-button>
<ewt-button
slot="secondaryAction"
label="Back"
=${() => {
this._state = "DASHBOARD";
}}
></ewt-button>
`;
return [heading, content];
}
_renderInstall(): [string | undefined, TemplateResult, boolean, boolean] {
let heading: string | undefined;
let content: TemplateResult;
let hideActions = false;
const allowClosing = false;
const isUpdate = !this._installErase && this._isSameFirmware;
if (!this._installConfirmed && this._isSameVersion) {
heading = "Erase User Data";
content = html`
Do you want to reset your device and erase all user data from your
device?
<ewt-button
class="danger"
slot="primaryAction"
label="Erase User Data"
=${this._confirmInstall}
></ewt-button>
`;
} else if (!this._installConfirmed) {
heading = "Confirm Installation";
const action = isUpdate ? "update to" : "install";
content = html`
${isUpdate
? html`Your device is running
${this._info!.firmware} ${this._info!.version}.<br /><br />`
: ""}
Do you want to ${action}
${this._manifest.name} ${this._manifest.version}?
${this._installErase
? html`<br /><br />All data on the device will be erased.`
: ""}
<ewt-button
slot="primaryAction"
label="Install"
=${this._confirmInstall}
></ewt-button>
<ewt-button
slot="secondaryAction"
label="Back"
=${() => {
this._state = "DASHBOARD";
}}
></ewt-button>
`;
} else if (
!this._installState ||
this._installState.state === FlashStateType.INITIALIZING ||
this._installState.state === FlashStateType.MANIFEST ||
this._installState.state === FlashStateType.PREPARING
) {
heading = "Installing";
content = this._renderProgress("Preparing installation");
hideActions = true;
} else if (this._installState.state === FlashStateType.ERASING) {
heading = "Installing";
content = this._renderProgress("Erasing");
hideActions = true;
} else if (
this._installState.state === FlashStateType.WRITING ||
// When we're finished, keep showing this screen with 100% written
// until Improv is initialized / not detected.
(this._installState.state === FlashStateType.FINISHED &&
this._client === undefined)
) {
heading = "Installing";
let percentage: number | undefined;
let undeterminateLabel: string | undefined;
if (this._installState.state === FlashStateType.FINISHED) {
// We're done writing and detecting improv, show spinner
undeterminateLabel = "Wrapping up";
} else if (this._installState.details.percentage < 4) {
// We're writing the firmware under 4%, show spinner or else we don't show any pixels
undeterminateLabel = "Installing";
} else {
// We're writing the firmware over 4%, show progress bar
percentage = this._installState.details.percentage;
}
content = this._renderProgress(
html`
${undeterminateLabel ? html`${undeterminateLabel}<br />` : ""}
<br />
This will take
${this._installState.chipFamily === "ESP8266"
? "a minute"
: "2 minutes"}.<br />
Keep this page visible to prevent slow down
`,
percentage,
);
hideActions = true;
} else if (this._installState.state === FlashStateType.FINISHED) {
heading = undefined;
const supportsImprov = this._client !== null;
content = html`
<ewt-page-message
.icon=${OK_ICON}
label="Installation complete!"
></ewt-page-message>
<ewt-button
slot="primaryAction"
label="Next"
=${() => {
this._state =
supportsImprov && this._installErase ? "PROVISION" : "DASHBOARD";
}}
></ewt-button>
`;
} else if (this._installState.state === FlashStateType.ERROR) {
heading = "Installation failed";
content = html`
<ewt-page-message
.icon=${ERROR_ICON}
.label=${this._installState.message}
></ewt-page-message>
<ewt-button
slot="primaryAction"
label="Back"
=${async () => {
this._initialize();
this._state = "DASHBOARD";
}}
></ewt-button>
`;
}
return [heading, content!, hideActions, allowClosing];
}
_renderLogs(): [string | undefined, TemplateResult, boolean] {
let heading: string | undefined = `Logs`;
let content: TemplateResult;
let hideActions = false;
content = html`
<ewt-console .port=${this.port} .logger=${this.logger}></ewt-console>
<ewt-button
slot="primaryAction"
label="Back"
=${async () => {
await this.shadowRoot!.querySelector("ewt-console")!.disconnect();
this._state = "DASHBOARD";
this._initialize();
}}
></ewt-button>
<ewt-button
slot="secondaryAction"
label="Download Logs"
=${() => {
textDownload(
this.shadowRoot!.querySelector("ewt-console")!.logs(),
`esp-web-tools-logs.txt`,
);
this.shadowRoot!.querySelector("ewt-console")!.reset();
}}
></ewt-button>
<ewt-button
slot="secondaryAction"
label="Reset Device"
=${async () => {
await this.shadowRoot!.querySelector("ewt-console")!.reset();
}}
></ewt-button>
`;
return [heading, content!, hideActions];
}
public override willUpdate(changedProps: PropertyValues) {
if (!changedProps.has("_state")) {
return;
}
// Clear errors when changing between pages unless we change
// to the error page.
if (this._state !== "ERROR") {
this._error = undefined;
}
// Scan for SSIDs on provision
if (this._state === "PROVISION") {
this._ssids = undefined;
this._busy = true;
this._client!.scan().then(
(ssids) => {
this._busy = false;
this._ssids = ssids;
this._selectedSsid = ssids.length ? 0 : -1;
},
() => {
this._busy = false;
this._ssids = null;
this._selectedSsid = -1;
},
);
} else {
// Reset this value if we leave provisioning.
this._provisionForce = false;
}
if (this._state === "INSTALL") {
this._installConfirmed = false;
this._installState = undefined;
}
}
protected override firstUpdated(changedProps: PropertyValues) {
super.firstUpdated(changedProps);
this._initialize();
}
protected override updated(changedProps: PropertyValues) {
super.updated(changedProps);
if (changedProps.has("_state")) {
this.setAttribute("state", this._state);
}
if (this._state !== "PROVISION") {
return;
}
if (changedProps.has("_selectedSsid") && this._selectedSsid === -1) {
// If we pick "Join other", select SSID input.
this._focusFormElement("ewt-textfield[name=ssid]");
} else if (changedProps.has("_ssids")) {
// Form is shown when SSIDs are loaded/marked not supported
this._focusFormElement();
}
}
private _focusFormElement(selector = "ewt-textfield, ewt-select") {
const formEl = this.shadowRoot!.querySelector(
selector,
) as LitElement | null;
if (formEl) {
formEl.updateComplete.then(() => setTimeout(() => formEl.focus(), 100));
}
}
private async _initialize(justInstalled = false) {
if (this.port.readable === null || this.port.writable === null) {
this._state = "ERROR";
this._error =
"Serial port is not readable/writable. Close any other application using it and try again.";
return;
}
try {
// If local file upload via browser is used, we already provide a manifest as a JSON string and not a URL to it
this._manifest = JSON.parse(this.manifestPath);
} catch {
// Standard procedure - download manifest.json with provided URL
try {
this._manifest = await downloadManifest(this.manifestPath);
} catch (err: any) {
this._state = "ERROR";
this._error = "Failed to download manifest";
return;
}
}
if (this._manifest.new_install_improv_wait_time === 0) {
this._client = null;
return;
}
const client = new ImprovSerial(this.port!, this.logger);
client.addEventListener("state-changed", () => {
this.requestUpdate();
});
client.addEventListener("error-changed", () => this.requestUpdate());
try {
// If a device was just installed, give new firmware 10 seconds (overridable) to
// format the rest of the flash and do other stuff.
const timeout = !justInstalled
? 1000
: this._manifest.new_install_improv_wait_time !== undefined
? this._manifest.new_install_improv_wait_time * 1000
: 10000;
this._info = await client.initialize(timeout);
this._client = client;
client.addEventListener("disconnect", this._handleDisconnect);
} catch (err: any) {
// Clear old value
this._info = undefined;
if (err instanceof PortNotReady) {
this._state = "ERROR";
this._error =
"Serial port is not ready. Close any other application using it and try again.";
} else {
this._client = null; // not supported
this.logger.error("Improv initialization failed.", err);
}
}
}
private _startInstall(erase: boolean) {
this._state = "INSTALL";
this._installErase = erase;
this._installConfirmed = false;
}
private async _confirmInstall() {
this._installConfirmed = true;
this._installState = undefined;
if (this._client) {
await this._closeClientWithoutEvents(this._client);
}
this._client = undefined;
if (this.firmwareFile != undefined) {
// If a uploaded File was provided -> create Uint8Array of content
new Blob([this.firmwareFile])
.arrayBuffer()
.then((b) => this._flashFilebuffer(b as Uint8Array));
} else {
// Use "standard way" with URL to manifest and firmware binary
flash(
(state) => {
this._installState = state;
if (state.state === FlashStateType.FINISHED) {
sleep(100)
.then(() => this._initialize(true))
.then(() => this.requestUpdate());
}
},
this.port,
this.logger,
this.manifestPath,
this._installErase,
new Uint8Array(0),
);
}
}
async _flashFilebuffer(fileBuffer: Uint8Array) {
flash(
(state) => {
this._installState = state;
if (state.state === FlashStateType.FINISHED) {
sleep(100)
.then(() => this._initialize(true))
.then(() => this.requestUpdate());
}
},
this.port,
this.logger,
this.manifestPath,
this._installErase,
fileBuffer,
);
}
private async _doProvision() {
this._busy = true;
this._wasProvisioned =
this._client!.state === ImprovSerialCurrentState.PROVISIONED;
const ssid =
this._selectedSsid === -1
? (
this.shadowRoot!.querySelector(
"ewt-textfield[name=ssid]",
) as EwtTextfield
).value
: this._ssids![this._selectedSsid].name;
const password = (
this.shadowRoot!.querySelector(
"ewt-textfield[name=password]",
) as EwtTextfield
).value;
try {
await this._client!.provision(ssid, password);
} catch (err: any) {
return;
} finally {
this._busy = false;
this._provisionForce = false;
}
}
private _handleDisconnect = () => {
this._state = "ERROR";
this._error = "Disconnected";
};
private async _handleClose() {
if (this._client) {
await this._closeClientWithoutEvents(this._client);
}
fireEvent(this, "closed" as any);
this.parentNode!.removeChild(this);
}
/**
* Return if the device runs same firmware as manifest.
*/
private get _isSameFirmware() {
return !this._info
? false
: this.overrides?.checkSameFirmware
? this.overrides.checkSameFirmware(this._manifest, this._info)
: this._info.firmware === this._manifest.name;
}
/**
* Return if the device runs same firmware and version as manifest.
*/
private get _isSameVersion() {
return (
this._isSameFirmware && this._info!.version === this._manifest.version
);
}
private async _closeClientWithoutEvents(client: ImprovSerial) {
client.removeEventListener("disconnect", this._handleDisconnect);
await client.close();
}
static styles = [
dialogStyles,
css`
:host {
--mdc-dialog-max-width: 390px;
}
ewt-icon-button {
position: absolute;
right: 4px;
top: 10px;
}
.table-row {
display: flex;
}
.table-row.last {
margin-bottom: 16px;
}
.table-row svg {
width: 20px;
margin-right: 8px;
}
ewt-textfield,
ewt-select {
display: block;
margin-top: 16px;
}
.dashboard-buttons {
margin: 0 0 -16px -8px;
}
.dashboard-buttons div {
display: block;
margin: 4px 0;
}
a.has-button {
text-decoration: none;
}
.error {
color: var(--improv-danger-color);
}
.danger {
--mdc-theme-primary: var(--improv-danger-color);
--mdc-theme-secondary: var(--improv-danger-color);
}
button.link {
background: none;
color: inherit;
border: none;
padding: 0;
font: inherit;
text-align: left;
text-decoration: underline;
cursor: pointer;
}
:host([state="LOGS"]) ewt-dialog {
--mdc-dialog-max-width: 90vw;
}
ewt-console {
width: calc(80vw - 48px);
height: 80vh;
}
`,
];
}
customElements.define("ewt-install-dialog", EwtInstallDialog);
declare global {
interface HTMLElementTagNameMap {
"ewt-install-dialog": EwtInstallDialog;
}
}