forked from mirrors/amnezia-wg-easy
		
	Version 10: Dark mode & sessionless auth (#691)
Version 10 release, including: Fixed issue when running on Synology NAS Added sessionless HTTP API auth (with thanks to @James-Stokes) Automatic dark mode (with thanks to @goodbyepavlyi) Version bumps Code refinement (that feels like it's only a small step up from saying "bug fixes")
This commit is contained in:
		
						commit
						032ef54798
					
				
					 36 changed files with 3875 additions and 703 deletions
				
			
		
							
								
								
									
										40
									
								
								.github/workflows/codeql.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.github/workflows/codeql.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,40 @@
 | 
			
		|||
name: "CodeQL"
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
    branches: [ "master" ]
 | 
			
		||||
  pull_request:
 | 
			
		||||
    branches: [ "master" ]
 | 
			
		||||
  schedule:
 | 
			
		||||
    - cron: "15 0 * * *"
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  analyze:
 | 
			
		||||
    name: Analyze
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    permissions:
 | 
			
		||||
      actions: read
 | 
			
		||||
      contents: read
 | 
			
		||||
      security-events: write
 | 
			
		||||
 | 
			
		||||
    strategy:
 | 
			
		||||
      fail-fast: false
 | 
			
		||||
      matrix:
 | 
			
		||||
        language: [ 'javascript-typescript' ]
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: Checkout repository
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
    - name: Initialize CodeQL
 | 
			
		||||
      uses: github/codeql-action/init@v3
 | 
			
		||||
      with:
 | 
			
		||||
        languages: ${{ matrix.language }}
 | 
			
		||||
 | 
			
		||||
    - name: Autobuild
 | 
			
		||||
      uses: github/codeql-action/autobuild@v3
 | 
			
		||||
 | 
			
		||||
    - name: Perform CodeQL Analysis
 | 
			
		||||
      uses: github/codeql-action/analyze@v3
 | 
			
		||||
      with:
 | 
			
		||||
        category: "/language:${{matrix.language}}"
 | 
			
		||||
							
								
								
									
										2
									
								
								.github/workflows/deploy-development.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/deploy-development.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -33,4 +33,4 @@ jobs:
 | 
			
		|||
      with:
 | 
			
		||||
        push: true
 | 
			
		||||
        platforms: linux/amd64,linux/arm/v6,linux/arm/v7,linux/arm64/v8
 | 
			
		||||
        tags: ghcr.io/wg-easy/wg-easy:development
 | 
			
		||||
        tags: ghcr.io/wg-easy/wg-easy:development
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										15
									
								
								.github/workflows/deploy-nightly.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/deploy-nightly.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -1,9 +1,9 @@
 | 
			
		|||
name: Build & Publish Nightly Docker Image to GitHub Container Registry
 | 
			
		||||
name: Build & Publish Nightly
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  workflow_dispatch:
 | 
			
		||||
  schedule:
 | 
			
		||||
    - cron: "0 12 * * *"
 | 
			
		||||
    - cron: "0 0 * * *"
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  deploy:
 | 
			
		||||
| 
						 | 
				
			
			@ -18,10 +18,10 @@ jobs:
 | 
			
		|||
        ref: production
 | 
			
		||||
 | 
			
		||||
    - name: Set up QEMU
 | 
			
		||||
      uses: docker/setup-qemu-action@v1
 | 
			
		||||
      uses: docker/setup-qemu-action@v3
 | 
			
		||||
 | 
			
		||||
    - name: Set up Docker Buildx
 | 
			
		||||
      uses: docker/setup-buildx-action@v1
 | 
			
		||||
      uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
    - name: Login to GitHub Container Registry
 | 
			
		||||
      uses: docker/login-action@v3
 | 
			
		||||
| 
						 | 
				
			
			@ -29,13 +29,10 @@ jobs:
 | 
			
		|||
        registry: ghcr.io
 | 
			
		||||
        username: ${{ github.actor }}
 | 
			
		||||
        password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          
 | 
			
		||||
    - name: Set environment variables
 | 
			
		||||
      run: echo RELEASE=$(cat ./src/package.json | jq -r .release) >> $GITHUB_ENV
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    - name: Build & Publish Docker Image
 | 
			
		||||
      uses: docker/build-push-action@v5
 | 
			
		||||
      with:
 | 
			
		||||
        push: true
 | 
			
		||||
        platforms: linux/amd64,linux/arm/v6,linux/arm/v7,linux/arm64/v8
 | 
			
		||||
        tags: ghcr.io/wg-easy/wg-easy:nightly, ghcr.io/wg-easy/wg-easy:${{ env.RELEASE }}-nightly
 | 
			
		||||
        tags: ghcr.io/wg-easy/wg-easy:nightly
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										10
									
								
								.github/workflows/deploy.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/deploy.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -1,4 +1,4 @@
 | 
			
		|||
name: Build & Publish Docker Image to GitHub Container Registry
 | 
			
		||||
name: Build & Publish Latest
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  workflow_dispatch:
 | 
			
		||||
| 
						 | 
				
			
			@ -19,10 +19,10 @@ jobs:
 | 
			
		|||
        ref: production
 | 
			
		||||
 | 
			
		||||
    - name: Set up QEMU
 | 
			
		||||
      uses: docker/setup-qemu-action@v1
 | 
			
		||||
      uses: docker/setup-qemu-action@v3
 | 
			
		||||
 | 
			
		||||
    - name: Set up Docker Buildx
 | 
			
		||||
      uses: docker/setup-buildx-action@v1
 | 
			
		||||
      uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
    - name: Login to GitHub Container Registry
 | 
			
		||||
      uses: docker/login-action@v3
 | 
			
		||||
| 
						 | 
				
			
			@ -30,10 +30,10 @@ jobs:
 | 
			
		|||
        registry: ghcr.io
 | 
			
		||||
        username: ${{ github.actor }}
 | 
			
		||||
        password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
    - name: Set environment variables
 | 
			
		||||
      run: echo RELEASE=$(cat ./src/package.json | jq -r .release) >> $GITHUB_ENV
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    - name: Build & Publish Docker Image
 | 
			
		||||
      uses: docker/build-push-action@v5
 | 
			
		||||
      with:
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										14
									
								
								.github/workflows/lint.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/lint.yml
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -12,12 +12,20 @@ jobs:
 | 
			
		|||
    name: Lint
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v2
 | 
			
		||||
      - uses: actions/setup-node@v1
 | 
			
		||||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
      - name: Setup Node
 | 
			
		||||
        uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: '18'
 | 
			
		||||
          check-latest: true
 | 
			
		||||
          cache: 'npm'
 | 
			
		||||
          cache-dependency-path: |
 | 
			
		||||
            package-lock.json
 | 
			
		||||
            src/package-lock.json
 | 
			
		||||
 | 
			
		||||
      - run: |
 | 
			
		||||
      - name: npm run lint
 | 
			
		||||
        run: |
 | 
			
		||||
          cd src
 | 
			
		||||
          npm ci
 | 
			
		||||
          npm run lint
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										41
									
								
								.github/workflows/npm-update-bot.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/npm-update-bot.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,41 @@
 | 
			
		|||
name: NPM Update Bot 🤖
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
    branches: [ "master" ]
 | 
			
		||||
  schedule:
 | 
			
		||||
    - cron: "0 0 * * *"
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  npmupbot:
 | 
			
		||||
    name: NPM Update Bot 🤖
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout repository
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
        with:
 | 
			
		||||
          repository: wg-easy/wg-easy
 | 
			
		||||
          ref: master
 | 
			
		||||
      - name: Setup Node
 | 
			
		||||
        uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: '18'
 | 
			
		||||
          check-latest: true
 | 
			
		||||
          cache: 'npm'
 | 
			
		||||
          cache-dependency-path: |
 | 
			
		||||
            package-lock.json
 | 
			
		||||
            src/package-lock.json
 | 
			
		||||
 | 
			
		||||
      - name: Bot 🤖 "Updating NPM Packages..."
 | 
			
		||||
        run: |
 | 
			
		||||
          npm install -g --silent npm-check-updates
 | 
			
		||||
          ncu -u
 | 
			
		||||
          npm update
 | 
			
		||||
          cd src
 | 
			
		||||
          ncu -u
 | 
			
		||||
          npm update
 | 
			
		||||
          git config --global user.name 'NPM Update Bot'
 | 
			
		||||
          git config --global user.email 'npmupbot@users.noreply.github.com'
 | 
			
		||||
          git add .
 | 
			
		||||
          git commit -am "npm: package updates" || true
 | 
			
		||||
          git push
 | 
			
		||||
							
								
								
									
										34
									
								
								.github/workflows/stale.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.github/workflows/stale.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,34 @@
 | 
			
		|||
# This workflow warns and then closes issues and PRs that have had no activity for a specified amount of time.
 | 
			
		||||
#
 | 
			
		||||
# You can adjust the behavior by modifying this file.
 | 
			
		||||
# For more information, see:
 | 
			
		||||
# https://github.com/actions/stale
 | 
			
		||||
name: Mark stale issues and pull requests
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  workflow_dispatch:
 | 
			
		||||
  schedule:
 | 
			
		||||
  - cron: '*/5 * * * *'
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  stale:
 | 
			
		||||
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    permissions:
 | 
			
		||||
      issues: write
 | 
			
		||||
      pull-requests: write
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
    - uses: actions/stale@v9
 | 
			
		||||
      with:
 | 
			
		||||
          days-before-issue-stale: 14
 | 
			
		||||
          days-before-issue-close: 7
 | 
			
		||||
          stale-issue-label: "stale"
 | 
			
		||||
          stale-issue-message: "This issue is stale because it has been open for 30 days with no activity."
 | 
			
		||||
          close-issue-message: "This issue was closed because it has been inactive for 14 days since being marked as stale."
 | 
			
		||||
          days-before-pr-stale: 30
 | 
			
		||||
          days-before-pr-close: 14
 | 
			
		||||
          stale-pr-message: "This PR is stale because it has been open for 30 days with no activity."
 | 
			
		||||
          close-pr-message: "This PR was closed because it has been inactive for 14 days since being marked as stale."
 | 
			
		||||
          repo-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          operations-per-run: 100
 | 
			
		||||
							
								
								
									
										3
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -1,3 +1,4 @@
 | 
			
		|||
/config
 | 
			
		||||
/wg0.conf
 | 
			
		||||
/wg0.json
 | 
			
		||||
/wg0.json
 | 
			
		||||
.DS_Store
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										10
									
								
								Dockerfile
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								Dockerfile
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -1,9 +1,12 @@
 | 
			
		|||
# There's an issue with node:20-alpine.
 | 
			
		||||
# Docker deployment is canceled after 25< minutes.
 | 
			
		||||
 | 
			
		||||
FROM docker.io/library/node:18-alpine AS build_node_modules
 | 
			
		||||
 | 
			
		||||
# Copy Web UI
 | 
			
		||||
COPY src/ /app/
 | 
			
		||||
WORKDIR /app
 | 
			
		||||
RUN npm ci --production
 | 
			
		||||
RUN npm ci --omit=dev
 | 
			
		||||
 | 
			
		||||
# Copy build result to a new image.
 | 
			
		||||
# This saves a lot of disk space.
 | 
			
		||||
| 
						 | 
				
			
			@ -19,8 +22,11 @@ COPY --from=build_node_modules /app /app
 | 
			
		|||
# than what runs inside of docker.
 | 
			
		||||
RUN mv /app/node_modules /node_modules
 | 
			
		||||
 | 
			
		||||
# Enable this to run `npm run serve`
 | 
			
		||||
RUN npm i -g nodemon
 | 
			
		||||
 | 
			
		||||
# Install Linux packages
 | 
			
		||||
RUN apk add -U --no-cache \
 | 
			
		||||
RUN apk add --no-cache \
 | 
			
		||||
    dpkg \
 | 
			
		||||
    dumb-init \
 | 
			
		||||
    iptables \
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										13
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										13
									
								
								README.md
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -22,6 +22,7 @@ You have found the easiest way to install & manage WireGuard on any Linux host!
 | 
			
		|||
* Statistics for which clients are connected.
 | 
			
		||||
* Tx/Rx charts for each connected client.
 | 
			
		||||
* Gravatar support.
 | 
			
		||||
* Automatic Light / Dark Mode (WIP: available in `nightly` build)
 | 
			
		||||
 | 
			
		||||
## Requirements
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -63,7 +64,7 @@ $ docker run -d \
 | 
			
		|||
</pre>
 | 
			
		||||
 | 
			
		||||
> 💡 Replace `YOUR_SERVER_IP` with your WAN IP, or a Dynamic DNS hostname.
 | 
			
		||||
> 
 | 
			
		||||
>
 | 
			
		||||
> 💡 Replace `YOUR_ADMIN_PASSWORD` with a password to log in on the Web UI.
 | 
			
		||||
 | 
			
		||||
The Web UI will now be available on `http://0.0.0.0:51821`.
 | 
			
		||||
| 
						 | 
				
			
			@ -72,7 +73,7 @@ The Web UI will now be available on `http://0.0.0.0:51821`.
 | 
			
		|||
 | 
			
		||||
### 3. Sponsor
 | 
			
		||||
 | 
			
		||||
Are you enjoying this project? [Buy me a beer!](https://github.com/sponsors/WeeJeWel) 🍻
 | 
			
		||||
Are you enjoying this project? [Buy Emile a beer!](https://github.com/sponsors/WeeJeWel) 🍻
 | 
			
		||||
 | 
			
		||||
## Options
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -80,14 +81,16 @@ These options can be configured by setting environment variables using `-e KEY="
 | 
			
		|||
 | 
			
		||||
| Env | Default | Example | Description |
 | 
			
		||||
| - | - | - | - |
 | 
			
		||||
| `PORT` | `51821` | `6789` | TCP port for Web UI. |
 | 
			
		||||
| `WEBUI_HOST` | `0.0.0.0` | `localhost` | IP address web UI binds to. |
 | 
			
		||||
| `PASSWORD` | - | `foobar123` | When set, requires a password when logging in to the Web UI. |
 | 
			
		||||
| `WG_HOST` | - | `vpn.myserver.com` | The public hostname of your VPN server. |
 | 
			
		||||
| `WG_DEVICE` | `eth0` | `ens6f0` | Ethernet device the wireguard traffic should be forwarded through. |
 | 
			
		||||
| `WG_PORT` | `51820` | `12345` | The public UDP port of your VPN server. WireGuard will always listen on `51820` inside the Docker container. |
 | 
			
		||||
| `WG_PORT` | `51820` | `12345` | The public UDP port of your VPN server. WireGuard will always listen on 51820 inside the Docker container. |
 | 
			
		||||
| `WG_MTU` | `null` | `1420` | The MTU the clients will use. Server uses default WG MTU. |
 | 
			
		||||
| `WG_PERSISTENT_KEEPALIVE` | `0` | `25` | Value in seconds to keep the "connection" open. If this value is 0, then connections won't be kept alive. |
 | 
			
		||||
| `WG_DEFAULT_ADDRESS` | `10.8.0.x` | `10.6.0.x` | Clients IP address range. |
 | 
			
		||||
| `WG_DEFAULT_DNS` | `1.1.1.1` | `8.8.8.8, 8.8.4.4` | DNS server clients will use. |
 | 
			
		||||
| `WG_DEFAULT_DNS` | `1.1.1.1` | `8.8.8.8, 8.8.4.4` | DNS server clients will use. If set to blank value, clients will not use any DNS. |
 | 
			
		||||
| `WG_ALLOWED_IPS` | `0.0.0.0/0, ::/0` | `192.168.15.0/24, 10.0.1.0/24` | Allowed IPs clients will use. |
 | 
			
		||||
| `WG_PRE_UP` | `...` | - | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L19) for the default value. |
 | 
			
		||||
| `WG_POST_UP` | `...` | `iptables ...` | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L20) for the default value. |
 | 
			
		||||
| 
						 | 
				
			
			@ -112,3 +115,5 @@ And then run the `docker run -d \ ...` command above again.
 | 
			
		|||
 | 
			
		||||
* [Using WireGuard-Easy with Pi-Hole](https://github.com/wg-easy/wg-easy/wiki/Using-WireGuard-Easy-with-Pi-Hole)
 | 
			
		||||
* [Using WireGuard-Easy with nginx/SSL](https://github.com/wg-easy/wg-easy/wiki/Using-WireGuard-Easy-with-nginx-SSL)
 | 
			
		||||
 | 
			
		||||
For less common or specific edge-case scenarios, please refer to the detailed information provided in the [Wiki](https://github.com/wg-easy/wg-easy/wiki).
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,6 +5,6 @@ services:
 | 
			
		|||
    command: npm run serve
 | 
			
		||||
    volumes:
 | 
			
		||||
      - ./src/:/app/
 | 
			
		||||
    environment: 
 | 
			
		||||
    environment:
 | 
			
		||||
      # - PASSWORD=p
 | 
			
		||||
      - WG_HOST=192.168.1.233
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,4 +1,7 @@
 | 
			
		|||
version: "3.8"
 | 
			
		||||
volumes:
 | 
			
		||||
  etc_wireguard:
 | 
			
		||||
 | 
			
		||||
services:
 | 
			
		||||
  wg-easy:
 | 
			
		||||
    environment:
 | 
			
		||||
| 
						 | 
				
			
			@ -13,15 +16,16 @@ services:
 | 
			
		|||
      # - WG_DEFAULT_DNS=1.1.1.1
 | 
			
		||||
      # - WG_MTU=1420
 | 
			
		||||
      # - WG_ALLOWED_IPS=192.168.15.0/24, 10.0.1.0/24
 | 
			
		||||
      # - WG_PERSISTENT_KEEPALIVE=25
 | 
			
		||||
      # - WG_PRE_UP=echo "Pre Up" > /etc/wireguard/pre-up.txt
 | 
			
		||||
      # - WG_POST_UP=echo "Post Up" > /etc/wireguard/post-up.txt
 | 
			
		||||
      # - WG_PRE_DOWN=echo "Pre Down" > /etc/wireguard/pre-down.txt
 | 
			
		||||
      # - WG_POST_DOWN=echo "Post Down" > /etc/wireguard/post-down.txt
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
    image: ghcr.io/wg-easy/wg-easy
 | 
			
		||||
    container_name: wg-easy
 | 
			
		||||
    volumes:
 | 
			
		||||
      - .:/etc/wireguard
 | 
			
		||||
      - etc_wireguard:/etc/wireguard
 | 
			
		||||
    ports:
 | 
			
		||||
      - "51820:51820/udp"
 | 
			
		||||
      - "51821:51821/tcp"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -7,5 +7,6 @@
 | 
			
		|||
  "6": "Many small performance improvements & bug fixes. Enjoy!",
 | 
			
		||||
  "7": "Improved the look & performance of the upload/download chart.",
 | 
			
		||||
  "8": "Updated to Node.js v18.",
 | 
			
		||||
  "9": "Fixed issue running on devices with older kernels."
 | 
			
		||||
  "9": "Fixed issue running on devices with older kernels.",
 | 
			
		||||
  "10": "Added sessionless HTTP API auth & automatic dark mode."
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										11
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										11
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							| 
						 | 
				
			
			@ -1,4 +1,11 @@
 | 
			
		|||
{
 | 
			
		||||
  "version": "1.0.0",
 | 
			
		||||
  "lockfileVersion": 1
 | 
			
		||||
  "name": "wg-easy",
 | 
			
		||||
  "version": "1.0.1",
 | 
			
		||||
  "lockfileVersion": 3,
 | 
			
		||||
  "requires": true,
 | 
			
		||||
  "packages": {
 | 
			
		||||
    "": {
 | 
			
		||||
      "version": "1.0.1"
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,8 +1,8 @@
 | 
			
		|||
{
 | 
			
		||||
  "version": "1.0.0",
 | 
			
		||||
  "version": "1.0.1",
 | 
			
		||||
  "scripts": {
 | 
			
		||||
    "build": "DOCKER_BUILDKIT=1 docker build --tag wg-easy .",
 | 
			
		||||
    "serve": "docker-compose -f docker-compose.yml -f docker-compose.dev.yml up",
 | 
			
		||||
    "start": "docker run --env WG_HOST=0.0.0.0 --name wg-easy --cap-add=NET_ADMIN --cap-add=SYS_MODULE --sysctl=\"net.ipv4.conf.all.src_valid_mark=1\" --mount type=bind,source=\"$(pwd)\"/config,target=/etc/wireguard -p 51820:51820/udp -p 51821:51821/tcp wg-easy"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -8,4 +8,4 @@
 | 
			
		|||
    "no-shadow": "off",
 | 
			
		||||
    "max-len": "off"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										2
									
								
								src/.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/.gitignore
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -1 +1 @@
 | 
			
		|||
/node_modules
 | 
			
		||||
/node_modules
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,6 +4,7 @@ const { release } = require('./package.json');
 | 
			
		|||
 | 
			
		||||
module.exports.RELEASE = release;
 | 
			
		||||
module.exports.PORT = process.env.PORT || 51821;
 | 
			
		||||
module.exports.WEBUI_HOST = process.env.WEBUI_HOST || '0.0.0.0';
 | 
			
		||||
module.exports.PASSWORD = process.env.PASSWORD;
 | 
			
		||||
module.exports.WG_PATH = process.env.WG_PATH || '/etc/wireguard/';
 | 
			
		||||
module.exports.WG_DEVICE = process.env.WG_DEVICE || 'eth0';
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,6 +1,8 @@
 | 
			
		|||
'use strict';
 | 
			
		||||
 | 
			
		||||
const path = require('path');
 | 
			
		||||
const bcrypt = require('bcryptjs');
 | 
			
		||||
const crypto = require('node:crypto');
 | 
			
		||||
 | 
			
		||||
const express = require('express');
 | 
			
		||||
const expressSession = require('express-session');
 | 
			
		||||
| 
						 | 
				
			
			@ -12,6 +14,7 @@ const WireGuard = require('../services/WireGuard');
 | 
			
		|||
 | 
			
		||||
const {
 | 
			
		||||
  PORT,
 | 
			
		||||
  WEBUI_HOST,
 | 
			
		||||
  RELEASE,
 | 
			
		||||
  PASSWORD,
 | 
			
		||||
} = require('../config');
 | 
			
		||||
| 
						 | 
				
			
			@ -25,17 +28,20 @@ module.exports = class Server {
 | 
			
		|||
      .use('/', express.static(path.join(__dirname, '..', 'www')))
 | 
			
		||||
      .use(express.json())
 | 
			
		||||
      .use(expressSession({
 | 
			
		||||
        secret: String(Math.random()),
 | 
			
		||||
        secret: crypto.randomBytes(256).toString('hex'),
 | 
			
		||||
        resave: true,
 | 
			
		||||
        saveUninitialized: true,
 | 
			
		||||
        cookie: {
 | 
			
		||||
          httpOnly: true,
 | 
			
		||||
        },
 | 
			
		||||
      }))
 | 
			
		||||
 | 
			
		||||
      .get('/api/release', (Util.promisify(async () => {
 | 
			
		||||
        return RELEASE;
 | 
			
		||||
      })))
 | 
			
		||||
 | 
			
		||||
      // Authentication
 | 
			
		||||
      .get('/api/session', Util.promisify(async req => {
 | 
			
		||||
    // Authentication
 | 
			
		||||
      .get('/api/session', Util.promisify(async (req) => {
 | 
			
		||||
        const requiresPassword = !!process.env.PASSWORD;
 | 
			
		||||
        const authenticated = requiresPassword
 | 
			
		||||
          ? !!(req.session && req.session.authenticated)
 | 
			
		||||
| 
						 | 
				
			
			@ -46,7 +52,7 @@ module.exports = class Server {
 | 
			
		|||
          authenticated,
 | 
			
		||||
        };
 | 
			
		||||
      }))
 | 
			
		||||
      .post('/api/session', Util.promisify(async req => {
 | 
			
		||||
      .post('/api/session', Util.promisify(async (req) => {
 | 
			
		||||
        const {
 | 
			
		||||
          password,
 | 
			
		||||
        } = req.body;
 | 
			
		||||
| 
						 | 
				
			
			@ -65,7 +71,7 @@ module.exports = class Server {
 | 
			
		|||
        debug(`New Session: ${req.session.id}`);
 | 
			
		||||
      }))
 | 
			
		||||
 | 
			
		||||
      // WireGuard
 | 
			
		||||
    // WireGuard
 | 
			
		||||
      .use((req, res, next) => {
 | 
			
		||||
        if (!PASSWORD) {
 | 
			
		||||
          return next();
 | 
			
		||||
| 
						 | 
				
			
			@ -75,18 +81,27 @@ module.exports = class Server {
 | 
			
		|||
          return next();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (req.path.startsWith('/api/') && req.headers['authorization']) {
 | 
			
		||||
          if (bcrypt.compareSync(req.headers['authorization'], bcrypt.hashSync(PASSWORD, 10))) {
 | 
			
		||||
            return next();
 | 
			
		||||
          }
 | 
			
		||||
          return res.status(401).json({
 | 
			
		||||
            error: 'Incorrect Password',
 | 
			
		||||
          });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return res.status(401).json({
 | 
			
		||||
          error: 'Not Logged In',
 | 
			
		||||
        });
 | 
			
		||||
      })
 | 
			
		||||
      .delete('/api/session', Util.promisify(async req => {
 | 
			
		||||
      .delete('/api/session', Util.promisify(async (req) => {
 | 
			
		||||
        const sessionId = req.session.id;
 | 
			
		||||
 | 
			
		||||
        req.session.destroy();
 | 
			
		||||
 | 
			
		||||
        debug(`Deleted Session: ${sessionId}`);
 | 
			
		||||
      }))
 | 
			
		||||
      .get('/api/wireguard/client', Util.promisify(async req => {
 | 
			
		||||
      .get('/api/wireguard/client', Util.promisify(async (req) => {
 | 
			
		||||
        return WireGuard.getClients();
 | 
			
		||||
      }))
 | 
			
		||||
      .get('/api/wireguard/client/:clientId/qrcode.svg', Util.promisify(async (req, res) => {
 | 
			
		||||
| 
						 | 
				
			
			@ -108,35 +123,47 @@ module.exports = class Server {
 | 
			
		|||
        res.header('Content-Type', 'text/plain');
 | 
			
		||||
        res.send(config);
 | 
			
		||||
      }))
 | 
			
		||||
      .post('/api/wireguard/client', Util.promisify(async req => {
 | 
			
		||||
      .post('/api/wireguard/client', Util.promisify(async (req) => {
 | 
			
		||||
        const { name } = req.body;
 | 
			
		||||
        return WireGuard.createClient({ name });
 | 
			
		||||
      }))
 | 
			
		||||
      .delete('/api/wireguard/client/:clientId', Util.promisify(async req => {
 | 
			
		||||
      .delete('/api/wireguard/client/:clientId', Util.promisify(async (req) => {
 | 
			
		||||
        const { clientId } = req.params;
 | 
			
		||||
        return WireGuard.deleteClient({ clientId });
 | 
			
		||||
      }))
 | 
			
		||||
      .post('/api/wireguard/client/:clientId/enable', Util.promisify(async req => {
 | 
			
		||||
      .post('/api/wireguard/client/:clientId/enable', Util.promisify(async (req, res) => {
 | 
			
		||||
        const { clientId } = req.params;
 | 
			
		||||
        if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
 | 
			
		||||
          res.end(403);
 | 
			
		||||
        }
 | 
			
		||||
        return WireGuard.enableClient({ clientId });
 | 
			
		||||
      }))
 | 
			
		||||
      .post('/api/wireguard/client/:clientId/disable', Util.promisify(async req => {
 | 
			
		||||
      .post('/api/wireguard/client/:clientId/disable', Util.promisify(async (req, res) => {
 | 
			
		||||
        const { clientId } = req.params;
 | 
			
		||||
        if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
 | 
			
		||||
          res.end(403);
 | 
			
		||||
        }
 | 
			
		||||
        return WireGuard.disableClient({ clientId });
 | 
			
		||||
      }))
 | 
			
		||||
      .put('/api/wireguard/client/:clientId/name', Util.promisify(async req => {
 | 
			
		||||
      .put('/api/wireguard/client/:clientId/name', Util.promisify(async (req, res) => {
 | 
			
		||||
        const { clientId } = req.params;
 | 
			
		||||
        if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
 | 
			
		||||
          res.end(403);
 | 
			
		||||
        }
 | 
			
		||||
        const { name } = req.body;
 | 
			
		||||
        return WireGuard.updateClientName({ clientId, name });
 | 
			
		||||
      }))
 | 
			
		||||
      .put('/api/wireguard/client/:clientId/address', Util.promisify(async req => {
 | 
			
		||||
      .put('/api/wireguard/client/:clientId/address', Util.promisify(async (req, res) => {
 | 
			
		||||
        const { clientId } = req.params;
 | 
			
		||||
        if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
 | 
			
		||||
          res.end(403);
 | 
			
		||||
        }
 | 
			
		||||
        const { address } = req.body;
 | 
			
		||||
        return WireGuard.updateClientAddress({ clientId, address });
 | 
			
		||||
      }))
 | 
			
		||||
 | 
			
		||||
      .listen(PORT, () => {
 | 
			
		||||
        debug(`Listening on http://0.0.0.0:${PORT}`);
 | 
			
		||||
      .listen(PORT, WEBUI_HOST, () => {
 | 
			
		||||
        debug(`Listening on http://${WEBUI_HOST}:${PORT}`);
 | 
			
		||||
      });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -21,7 +21,7 @@ module.exports = class Util {
 | 
			
		|||
    // eslint-disable-next-line func-names
 | 
			
		||||
    return function(req, res) {
 | 
			
		||||
      Promise.resolve().then(async () => fn(req, res))
 | 
			
		||||
        .then(result => {
 | 
			
		||||
        .then((result) => {
 | 
			
		||||
          if (res.headersSent) return;
 | 
			
		||||
 | 
			
		||||
          if (typeof result === 'undefined') {
 | 
			
		||||
| 
						 | 
				
			
			@ -34,7 +34,7 @@ module.exports = class Util {
 | 
			
		|||
            .status(200)
 | 
			
		||||
            .json(result);
 | 
			
		||||
        })
 | 
			
		||||
        .catch(error => {
 | 
			
		||||
        .catch((error) => {
 | 
			
		||||
          if (typeof error === 'string') {
 | 
			
		||||
            error = new Error(error);
 | 
			
		||||
          }
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -60,14 +60,14 @@ module.exports = class WireGuard {
 | 
			
		|||
 | 
			
		||||
        await this.__saveConfig(config);
 | 
			
		||||
        await Util.exec('wg-quick down wg0').catch(() => { });
 | 
			
		||||
        await Util.exec('wg-quick up wg0').catch(err => {
 | 
			
		||||
        await Util.exec('wg-quick up wg0').catch((err) => {
 | 
			
		||||
          if (err && err.message && err.message.includes('Cannot find device "wg0"')) {
 | 
			
		||||
            throw new Error('WireGuard exited with the error: Cannot find device "wg0"\nThis usually means that your host\'s kernel does not support WireGuard!');
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          throw err;
 | 
			
		||||
        });
 | 
			
		||||
        // await Util.exec(`iptables -t nat -A POSTROUTING -s ${WG_DEFAULT_ADDRESS.replace('x', '0')}/24 -o eth0 -j MASQUERADE`);
 | 
			
		||||
        // await Util.exec(`iptables -t nat -A POSTROUTING -s ${WG_DEFAULT_ADDRESS.replace('x', '0')}/24 -o ' + WG_DEVICE + ' -j MASQUERADE`);
 | 
			
		||||
        // await Util.exec('iptables -A INPUT -p udp -m udp --dport 51820 -j ACCEPT');
 | 
			
		||||
        // await Util.exec('iptables -A FORWARD -i wg0 -j ACCEPT');
 | 
			
		||||
        // await Util.exec('iptables -A FORWARD -o wg0 -j ACCEPT');
 | 
			
		||||
| 
						 | 
				
			
			@ -156,7 +156,7 @@ AllowedIPs = ${client.address}/32`;
 | 
			
		|||
      .trim()
 | 
			
		||||
      .split('\n')
 | 
			
		||||
      .slice(1)
 | 
			
		||||
      .forEach(line => {
 | 
			
		||||
      .forEach((line) => {
 | 
			
		||||
        const [
 | 
			
		||||
          publicKey,
 | 
			
		||||
          preSharedKey, // eslint-disable-line no-unused-vars
 | 
			
		||||
| 
						 | 
				
			
			@ -168,7 +168,7 @@ AllowedIPs = ${client.address}/32`;
 | 
			
		|||
          persistentKeepalive,
 | 
			
		||||
        ] = line.split('\t');
 | 
			
		||||
 | 
			
		||||
        const client = clients.find(client => client.publicKey === publicKey);
 | 
			
		||||
        const client = clients.find((client) => client.publicKey === publicKey);
 | 
			
		||||
        if (!client) return;
 | 
			
		||||
 | 
			
		||||
        client.latestHandshakeAt = latestHandshakeAt === '0'
 | 
			
		||||
| 
						 | 
				
			
			@ -233,7 +233,7 @@ Endpoint = ${WG_HOST}:${WG_PORT}`;
 | 
			
		|||
    // Calculate next IP
 | 
			
		||||
    let address;
 | 
			
		||||
    for (let i = 2; i < 255; i++) {
 | 
			
		||||
      const client = Object.values(config.clients).find(client => {
 | 
			
		||||
      const client = Object.values(config.clients).find((client) => {
 | 
			
		||||
        return client.address === WG_DEFAULT_ADDRESS.replace('x', i);
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -248,8 +248,9 @@ Endpoint = ${WG_HOST}:${WG_PORT}`;
 | 
			
		|||
    }
 | 
			
		||||
 | 
			
		||||
    // Create Client
 | 
			
		||||
    const clientId = uuid.v4();
 | 
			
		||||
    const id = uuid.v4();
 | 
			
		||||
    const client = {
 | 
			
		||||
      id,
 | 
			
		||||
      name,
 | 
			
		||||
      address,
 | 
			
		||||
      privateKey,
 | 
			
		||||
| 
						 | 
				
			
			@ -262,7 +263,7 @@ Endpoint = ${WG_HOST}:${WG_PORT}`;
 | 
			
		|||
      enabled: true,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    config.clients[clientId] = client;
 | 
			
		||||
    config.clients[id] = client;
 | 
			
		||||
 | 
			
		||||
    await this.saveConfig();
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										2076
									
								
								src/package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										2076
									
								
								src/package-lock.json
									
										
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,26 +1,33 @@
 | 
			
		|||
{
 | 
			
		||||
  "release": 9,
 | 
			
		||||
  "release": "10",
 | 
			
		||||
  "name": "wg-easy",
 | 
			
		||||
  "version": "1.0.0",
 | 
			
		||||
  "description": "",
 | 
			
		||||
  "version": "1.0.1",
 | 
			
		||||
  "description": "The easiest way to run WireGuard VPN + Web-based Admin UI.",
 | 
			
		||||
  "main": "server.js",
 | 
			
		||||
  "scripts": {
 | 
			
		||||
    "serve": "DEBUG=Server,WireGuard node --watch server.js",
 | 
			
		||||
    "serve": "DEBUG=Server,WireGuard nodemon server.js",
 | 
			
		||||
    "serve-with-password": "PASSWORD=wg npm run serve",
 | 
			
		||||
    "lint": "eslint ."
 | 
			
		||||
    "lint": "eslint .",
 | 
			
		||||
    "buildcss": "npx tailwindcss -i ./www/src/css/app.css -o ./www/css/app.css --watch"
 | 
			
		||||
  },
 | 
			
		||||
  "author": "Emile Nijssen",
 | 
			
		||||
  "license": "GPL",
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "debug": "^4.3.1",
 | 
			
		||||
    "express": "^4.17.1",
 | 
			
		||||
    "express-session": "^1.17.1",
 | 
			
		||||
    "qrcode": "^1.4.4",
 | 
			
		||||
    "uuid": "^8.3.2"
 | 
			
		||||
    "bcryptjs": "^2.4.3",
 | 
			
		||||
    "debug": "^4.3.4",
 | 
			
		||||
    "express": "^4.18.2",
 | 
			
		||||
    "express-session": "^1.17.3",
 | 
			
		||||
    "qrcode": "^1.5.3",
 | 
			
		||||
    "uuid": "^9.0.1"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "eslint": "^7.27.0",
 | 
			
		||||
    "eslint-config-athom": "^2.1.0"
 | 
			
		||||
    "eslint-config-athom": "^3.1.3",
 | 
			
		||||
    "tailwindcss": "^3.4.1"
 | 
			
		||||
  },
 | 
			
		||||
  "nodemonConfig": {
 | 
			
		||||
    "ignore": [
 | 
			
		||||
      "www/*"
 | 
			
		||||
    ]
 | 
			
		||||
  },
 | 
			
		||||
  "engines": {
 | 
			
		||||
    "node": "18"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,7 +5,7 @@ require('./services/Server');
 | 
			
		|||
const WireGuard = require('./services/WireGuard');
 | 
			
		||||
 | 
			
		||||
WireGuard.getConfig()
 | 
			
		||||
  .catch(err => {
 | 
			
		||||
  .catch((err) => {
 | 
			
		||||
  // eslint-disable-next-line no-console
 | 
			
		||||
    console.error(err);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										8
									
								
								src/tailwind.config.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/tailwind.config.js
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,8 @@
 | 
			
		|||
/** @type {import('tailwindcss').Config} */
 | 
			
		||||
 | 
			
		||||
'use strict';
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  darkMode: 'media',
 | 
			
		||||
  content: ['./www/**/*.{html,js}'],
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										1718
									
								
								src/www/css/app.css
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1718
									
								
								src/www/css/app.css
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										1
									
								
								src/www/css/vendor/tailwind.min.css
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								src/www/css/vendor/tailwind.min.css
									
										
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| 
						 | 
				
			
			@ -3,29 +3,35 @@
 | 
			
		|||
 | 
			
		||||
<head>
 | 
			
		||||
  <title>WireGuard</title>
 | 
			
		||||
  <link href="./css/vendor/tailwind.min.css" rel="stylesheet">
 | 
			
		||||
  <script src="https://cdn.tailwindcss.com"></script>
 | 
			
		||||
  <script>
 | 
			
		||||
    tailwind.config = {
 | 
			
		||||
      darkMode: 'media',
 | 
			
		||||
    }
 | 
			
		||||
  </script>
 | 
			
		||||
  <!-- <link href="./css/vendor/tailwind.min.css" rel="stylesheet"> -->
 | 
			
		||||
  <link rel="manifest" href="./manifest.json">
 | 
			
		||||
  <link rel="icon" type="image/png" href="./img/favicon.png">
 | 
			
		||||
  <link rel="apple-touch-icon" href="./img/apple-touch-icon.png">
 | 
			
		||||
  <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover">
 | 
			
		||||
  <meta name="apple-mobile-web-app-capable" content="yes">
 | 
			
		||||
</head>
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
  [v-cloak] {
 | 
			
		||||
    display: none;
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
 | 
			
		||||
<body class="bg-gray-50">
 | 
			
		||||
<body class="bg-gray-50 dark:bg-neutral-800">
 | 
			
		||||
 | 
			
		||||
  <div id="app">
 | 
			
		||||
 | 
			
		||||
    <div v-cloak class="container mx-auto max-w-3xl">
 | 
			
		||||
    <div v-cloak class="container mx-auto max-w-3xl px-5 md:px-0">
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === true">
 | 
			
		||||
        <span v-if="requiresPassword"
 | 
			
		||||
          class="text-sm text-gray-400 mb-10 mr-2 mt-3 cursor-pointer hover:underline float-right" @click="logout">
 | 
			
		||||
          class="text-sm text-gray-400 dark:text-neutral-400 mb-10 mr-2 mt-3 cursor-pointer hover:underline float-right"
 | 
			
		||||
          @click="logout">
 | 
			
		||||
          Logout
 | 
			
		||||
          <svg class="h-3 inline" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
            stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -33,13 +39,14 @@
 | 
			
		|||
              d="M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1" />
 | 
			
		||||
          </svg>
 | 
			
		||||
        </span>
 | 
			
		||||
        <h1 class="text-4xl font-medium mt-10 mb-2">
 | 
			
		||||
          <img src="./img/logo.png" width="32" class="inline align-middle" />
 | 
			
		||||
        <h1 class="text-4xl dark:text-neutral-200 font-medium mt-2 mb-2">
 | 
			
		||||
          <img src="./img/logo.png" width="32" class="inline align-middle dark:bg" />
 | 
			
		||||
          <span class="align-middle">WireGuard</span>
 | 
			
		||||
        </h1>
 | 
			
		||||
        <h2 class="text-sm text-gray-400 mb-10"></h2>
 | 
			
		||||
        <h2 class="text-sm text-gray-400 dark:text-neutral-400 mb-10"></h2>
 | 
			
		||||
 | 
			
		||||
        <div v-if="latestRelease" class="bg-red-800 p-4 text-white text-sm font-small mb-10 rounded-md shadow-lg"
 | 
			
		||||
        <div v-if="latestRelease"
 | 
			
		||||
          class="bg-red-800 dark:bg-red-100 p-4 text-white dark:text-red-600 text-sm font-small mb-10 rounded-md shadow-lg"
 | 
			
		||||
          :title="`v${currentRelease} → v${latestRelease.version}`">
 | 
			
		||||
          <div class="container mx-auto flex flex-row flex-auto items-center">
 | 
			
		||||
            <div class="flex-grow">
 | 
			
		||||
| 
						 | 
				
			
			@ -48,20 +55,20 @@
 | 
			
		|||
            </div>
 | 
			
		||||
 | 
			
		||||
            <a href="https://github.com/wg-easy/wg-easy#updating" target="_blank"
 | 
			
		||||
              class="p-3 rounded-md bg-white float-right font-sm font-semibold text-red-800 flex-shrink-0 border-2 border-red-800 hover:border-white hover:text-white hover:bg-red-800 transition-all">
 | 
			
		||||
              class="p-3 rounded-md bg-white dark:bg-red-100 float-right font-sm font-semibold text-red-800 dark:text-red-600 flex-shrink-0 border-2 border-red-800 dark:border-red-600 hover:border-white dark:hover:border-red-600 hover:text-white dark:hover:text-red-100 hover:bg-red-800 dark:hover:bg-red-600 transition-all">
 | 
			
		||||
              Update →
 | 
			
		||||
            </a>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div class="shadow-md rounded-lg bg-white overflow-hidden">
 | 
			
		||||
          <div class="flex flex-row flex-auto items-center p-3 px-5 border border-b-2 border-gray-100">
 | 
			
		||||
        <div class="shadow-md rounded-lg bg-white dark:bg-neutral-700 overflow-hidden">
 | 
			
		||||
          <div class="flex flex-row flex-auto items-center p-3 px-5 border-b-2 border-gray-100 dark:border-neutral-600">
 | 
			
		||||
            <div class="flex-grow">
 | 
			
		||||
              <p class="text-2xl font-medium">Clients</p>
 | 
			
		||||
              <p class="text-2xl font-medium dark:text-neutral-200">Clients</p>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="flex-shrink-0">
 | 
			
		||||
              <button @click="clientCreate = true; clientCreateName = '';"
 | 
			
		||||
                class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 border-2 border-gray-100 py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 dark:text-neutral-200 border-2 border-gray-100 dark:border-neutral-600 py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                <svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                  stroke="currentColor">
 | 
			
		||||
                  <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
| 
						 | 
				
			
			@ -75,113 +82,56 @@
 | 
			
		|||
          <div>
 | 
			
		||||
            <!-- Client -->
 | 
			
		||||
            <div v-if="clients && clients.length > 0" v-for="client in clients" :key="client.id"
 | 
			
		||||
              class="relative overflow-hidden border-b border-gray-100 border-solid">
 | 
			
		||||
              class="relative overflow-hidden border-b last:border-b-0 border-gray-100 dark:border-neutral-600 border-solid">
 | 
			
		||||
 | 
			
		||||
              <!-- Chart -->
 | 
			
		||||
              <div class="absolute z-0 bottom-0 left-0 right-0" style="width: 100%; height: 20%;">
 | 
			
		||||
                <!-- Bar -->
 | 
			
		||||
                <div v-for="(_, index) in client.transferTxHistory" :style="{
 | 
			
		||||
                    display: 'inline-flex',
 | 
			
		||||
                    alignItems: 'flex-end',
 | 
			
		||||
                    width: '2%', // 1/100th of client.transferTxHistory.length
 | 
			
		||||
                    height: '100%',
 | 
			
		||||
                    padding: '0 3px',
 | 
			
		||||
                    boxSizing: 'border-box',
 | 
			
		||||
                    fontSize: 0,
 | 
			
		||||
                  }">
 | 
			
		||||
 | 
			
		||||
                  <!-- TX -->
 | 
			
		||||
                  <div :style="{
 | 
			
		||||
                    minHeight: '0px',
 | 
			
		||||
                    minWidth: '2px',
 | 
			
		||||
                    maxWidth: '4px',
 | 
			
		||||
                    width: '50%',
 | 
			
		||||
                    marginRight: '1px',
 | 
			
		||||
                    height: Math.round((client.transferTxHistory[index]/client.transferMax)*100) + '%',
 | 
			
		||||
                    background: client.hoverTx
 | 
			
		||||
                      ? '#992922'
 | 
			
		||||
                      : '#F3F4F6',
 | 
			
		||||
                    transition: 'all 0.2s',
 | 
			
		||||
                    borderRadius: '2px 2px 0 0',
 | 
			
		||||
                  }"></div>
 | 
			
		||||
 | 
			
		||||
                  <!-- RX -->
 | 
			
		||||
                  <div :style="{
 | 
			
		||||
                    minHeight: '0px',
 | 
			
		||||
                    minWidth: '2px',
 | 
			
		||||
                    maxWidth: '4px',
 | 
			
		||||
                    width: '50%',
 | 
			
		||||
                    height: Math.round((client.transferRxHistory[index]/client.transferMax)*100) + '%',
 | 
			
		||||
                    background: client.hoverRx
 | 
			
		||||
                    ? '#992922'
 | 
			
		||||
                    : '#F0F1F3',
 | 
			
		||||
                    transition: 'all 0.2s',
 | 
			
		||||
                    borderRadius: '2px 2px 0 0',
 | 
			
		||||
                  }"></div>
 | 
			
		||||
                </div>
 | 
			
		||||
              <div class="absolute z-0 bottom-0 left-0 right-0" style="top: 60%;">
 | 
			
		||||
                <apexchart width="100%" height="100%" :options="client.chartOptions" :series="client.transferTxSeries">
 | 
			
		||||
                </apexchart>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="absolute z-0 top-0 left-0 right-0" style="bottom: 60%;">
 | 
			
		||||
                <apexchart width="100%" height="100%" :options="client.chartOptions" :series="client.transferRxSeries"
 | 
			
		||||
                  style="transform: scaleY(-1);">
 | 
			
		||||
                </apexchart>
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              <div class="relative p-5 z-10 flex flex-row">
 | 
			
		||||
                <div class="h-10 w-10 mr-5 rounded-full bg-gray-50 relative">
 | 
			
		||||
                  <svg class="w-6 m-2 text-gray-300" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                    fill="currentColor">
 | 
			
		||||
                    <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z"
 | 
			
		||||
                      clip-rule="evenodd" />
 | 
			
		||||
                  </svg>
 | 
			
		||||
                  <img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" />
 | 
			
		||||
              <div class="relative p-5 z-10 flex flex-col md:flex-row justify-between">
 | 
			
		||||
                <div class="flex items-center pb-2 md:pb-0">
 | 
			
		||||
                  <div class="h-10 w-10 mr-5 rounded-full bg-gray-50 relative">
 | 
			
		||||
                    <svg class="w-6 m-2 text-gray-300" xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                      viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
                      <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z"
 | 
			
		||||
                        clip-rule="evenodd" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                    <img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" />
 | 
			
		||||
 | 
			
		||||
                  <div
 | 
			
		||||
                    v-if="client.latestHandshakeAt && ((new Date() - new Date(client.latestHandshakeAt) < 1000 * 60 * 10))">
 | 
			
		||||
                    <div class="animate-ping w-4 h-4 p-1 bg-red-100 rounded-full absolute -bottom-1 -right-1"></div>
 | 
			
		||||
                    <div class="w-2 h-2 bg-red-800 rounded-full absolute bottom-0 right-0"></div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="flex-grow">
 | 
			
		||||
 | 
			
		||||
                  <!-- Name -->
 | 
			
		||||
                  <div class="text-gray-700 group" :title="'Created on ' + dateTime(new Date(client.createdAt))">
 | 
			
		||||
 | 
			
		||||
                    <!-- Show -->
 | 
			
		||||
                    <input v-show="clientEditNameId === client.id" v-model="clientEditName"
 | 
			
		||||
                      v-on:keyup.enter="updateClientName(client, clientEditName); clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                      v-on:keyup.escape="clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                      :ref="'client-' + client.id + '-name'"
 | 
			
		||||
                      class="rounded px-1 border-2 border-gray-100 focus:border-gray-200 outline-none w-30" />
 | 
			
		||||
                    <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                      class="inline-block border-t-2 border-b-2 border-transparent">{{client.name}}</span>
 | 
			
		||||
 | 
			
		||||
                    <!-- Edit -->
 | 
			
		||||
                    <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                      @click="clientEditName = client.name; clientEditNameId = client.id; setTimeout(() => $refs['client-' + client.id + '-name'][0].select(), 1);"
 | 
			
		||||
                      class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                      <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                        class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
                        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                          d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <div
 | 
			
		||||
                      v-if="client.latestHandshakeAt && ((new Date() - new Date(client.latestHandshakeAt) < 1000 * 60 * 10))">
 | 
			
		||||
                      <div
 | 
			
		||||
                        class="animate-ping w-4 h-4 p-1 bg-red-100 dark:bg-red-100 rounded-full absolute -bottom-1 -right-1">
 | 
			
		||||
                      </div>
 | 
			
		||||
                      <div class="w-2 h-2 bg-red-800 dark:bg-red-600 rounded-full absolute bottom-0 right-0"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                  <!-- Info -->
 | 
			
		||||
                  <div class="text-gray-400 text-xs">
 | 
			
		||||
                  <div class="flex-grow">
 | 
			
		||||
 | 
			
		||||
                    <!-- Address -->
 | 
			
		||||
                    <span class="group">
 | 
			
		||||
                    <!-- Name -->
 | 
			
		||||
                    <div class="text-gray-700 dark:text-neutral-200 group"
 | 
			
		||||
                      :title="'Created on ' + dateTime(new Date(client.createdAt))">
 | 
			
		||||
 | 
			
		||||
                      <!-- Show -->
 | 
			
		||||
                      <input v-show="clientEditAddressId === client.id" v-model="clientEditAddress"
 | 
			
		||||
                        v-on:keyup.enter="updateClientAddress(client, clientEditAddress); clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                        v-on:keyup.escape="clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                        :ref="'client-' + client.id + '-address'"
 | 
			
		||||
                        class="rounded border-2 border-gray-100 focus:border-gray-200 outline-none w-20 text-black" />
 | 
			
		||||
                      <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                        class="inline-block border-t-2 border-b-2 border-transparent">{{client.address}}</span>
 | 
			
		||||
                      <input v-show="clientEditNameId === client.id" v-model="clientEditName"
 | 
			
		||||
                        v-on:keyup.enter="updateClientName(client, clientEditName); clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                        v-on:keyup.escape="clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                        :ref="'client-' + client.id + '-name'"
 | 
			
		||||
                        class="rounded px-1 border-2 dark:bg-neutral-700 border-gray-100 dark:border-neutral-600 focus:border-gray-200 dark:focus:border-neutral-500 dark:placeholder:text-neutral-500 outline-none w-30" />
 | 
			
		||||
                      <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                        class="inline-block border-t-2 border-b-2 border-transparent">{{client.name}}</span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Edit -->
 | 
			
		||||
                      <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                        @click="clientEditAddress = client.address; clientEditAddressId = client.id; setTimeout(() => $refs['client-' + client.id + '-address'][0].select(), 1);"
 | 
			
		||||
                      <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                        @click="clientEditName = client.name; clientEditNameId = client.id; setTimeout(() => $refs['client-' + client.id + '-name'][0].select(), 1);"
 | 
			
		||||
                        class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                        <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                          class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none"
 | 
			
		||||
| 
						 | 
				
			
			@ -190,48 +140,71 @@
 | 
			
		|||
                            d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
 | 
			
		||||
                    <!-- Transfer TX -->
 | 
			
		||||
                    <span v-if="client.transferTx" :title="'Total Download: ' + bytes(client.transferTx)"
 | 
			
		||||
                      @mouseover="client.hoverTx = clientsPersist[client.id].hoverTx = true;"
 | 
			
		||||
                      @mouseleave="client.hoverTx = clientsPersist[client.id].hoverTx = false;"
 | 
			
		||||
                      style="cursor: default;">
 | 
			
		||||
                      ·
 | 
			
		||||
                      <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                        fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
                          d="M16.707 10.293a1 1 0 010 1.414l-6 6a1 1 0 01-1.414 0l-6-6a1 1 0 111.414-1.414L9 14.586V3a1 1 0 012 0v11.586l4.293-4.293a1 1 0 011.414 0z"
 | 
			
		||||
                          clip-rule="evenodd" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                      {{client.transferTxCurrent | bytes}}/s
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <!-- Info -->
 | 
			
		||||
                    <div class="text-gray-400 dark:text-neutral-400 text-xs">
 | 
			
		||||
 | 
			
		||||
                    <!-- Transfer RX -->
 | 
			
		||||
                    <span v-if="client.transferRx" :title="'Total Upload: ' + bytes(client.transferRx)"
 | 
			
		||||
                      @mouseover="client.hoverRx = clientsPersist[client.id].hoverRx = true;"
 | 
			
		||||
                      @mouseleave="client.hoverRx = clientsPersist[client.id].hoverRx = false;"
 | 
			
		||||
                      style="cursor: default;">
 | 
			
		||||
                      ·
 | 
			
		||||
                      <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                        fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
                          d="M3.293 9.707a1 1 0 010-1.414l6-6a1 1 0 011.414 0l6 6a1 1 0 01-1.414 1.414L11 5.414V17a1 1 0 11-2 0V5.414L4.707 9.707a1 1 0 01-1.414 0z"
 | 
			
		||||
                          clip-rule="evenodd" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                      {{client.transferRxCurrent | bytes}}/s
 | 
			
		||||
                    </span>
 | 
			
		||||
                      <!-- Address -->
 | 
			
		||||
                      <span class="group block md:inline-block pb-1 md:pb-0">
 | 
			
		||||
 | 
			
		||||
                    <!-- Last seen -->
 | 
			
		||||
                    <span v-if="client.latestHandshakeAt"
 | 
			
		||||
                      :title="'Last seen on ' + dateTime(new Date(client.latestHandshakeAt))">
 | 
			
		||||
                      · {{new Date(client.latestHandshakeAt) | timeago}}
 | 
			
		||||
                    </span>
 | 
			
		||||
                        <!-- Show -->
 | 
			
		||||
                        <input v-show="clientEditAddressId === client.id" v-model="clientEditAddress"
 | 
			
		||||
                          v-on:keyup.enter="updateClientAddress(client, clientEditAddress); clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                          v-on:keyup.escape="clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                          :ref="'client-' + client.id + '-address'"
 | 
			
		||||
                          class="rounded border-2 dark:bg-neutral-700 border-gray-100 dark:border-neutral-600 focus:border-gray-200 dark:focus:border-neutral-500 outline-none w-20 text-black dark:text-neutral-300 dark:placeholder:text-neutral-500" />
 | 
			
		||||
                        <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                          class="inline-block border-t-2 border-b-2 border-transparent">{{client.address}}</span>
 | 
			
		||||
 | 
			
		||||
                        <!-- Edit -->
 | 
			
		||||
                        <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                          @click="clientEditAddress = client.address; clientEditAddressId = client.id; setTimeout(() => $refs['client-' + client.id + '-address'][0].select(), 1);"
 | 
			
		||||
                          class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                          <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                            class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none"
 | 
			
		||||
                            viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                            <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                              d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                          </svg>
 | 
			
		||||
                        </span>
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Transfer TX -->
 | 
			
		||||
                      <span v-if="client.transferTx" :title="'Total Download: ' + bytes(client.transferTx)">
 | 
			
		||||
                        ·
 | 
			
		||||
                        <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                          fill="currentColor">
 | 
			
		||||
                          <path fill-rule="evenodd"
 | 
			
		||||
                            d="M16.707 10.293a1 1 0 010 1.414l-6 6a1 1 0 01-1.414 0l-6-6a1 1 0 111.414-1.414L9 14.586V3a1 1 0 012 0v11.586l4.293-4.293a1 1 0 011.414 0z"
 | 
			
		||||
                            clip-rule="evenodd" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                        {{client.transferTxCurrent | bytes}}/s
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Transfer RX -->
 | 
			
		||||
                      <span v-if="client.transferRx" :title="'Total Upload: ' + bytes(client.transferRx)">
 | 
			
		||||
                        ·
 | 
			
		||||
                        <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                          fill="currentColor">
 | 
			
		||||
                          <path fill-rule="evenodd"
 | 
			
		||||
                            d="M3.293 9.707a1 1 0 010-1.414l6-6a1 1 0 011.414 0l6 6a1 1 0 01-1.414 1.414L11 5.414V17a1 1 0 11-2 0V5.414L4.707 9.707a1 1 0 01-1.414 0z"
 | 
			
		||||
                            clip-rule="evenodd" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                        {{client.transferRxCurrent | bytes}}/s
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Last seen -->
 | 
			
		||||
                      <span v-if="client.latestHandshakeAt"
 | 
			
		||||
                        :title="'Last seen on ' + dateTime(new Date(client.latestHandshakeAt))">
 | 
			
		||||
                        · {{new Date(client.latestHandshakeAt) | timeago}}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="text-right">
 | 
			
		||||
                  <div class="text-gray-400">
 | 
			
		||||
                <div class="flex items-center justify-end">
 | 
			
		||||
                  <div class="text-gray-400 dark:text-neutral-400 flex gap-1 items-center justify-between">
 | 
			
		||||
 | 
			
		||||
                    <!-- Enable/Disable -->
 | 
			
		||||
                    <div @click="disableClient(client)" v-if="client.enabled === true" title="Disable Client"
 | 
			
		||||
| 
						 | 
				
			
			@ -239,12 +212,13 @@
 | 
			
		|||
                      <div class="rounded-full w-4 h-4 m-1 ml-5 bg-white"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div @click="enableClient(client)" v-if="client.enabled === false" title="Enable Client"
 | 
			
		||||
                      class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-gray-200 cursor-pointer hover:bg-gray-300 transition-all">
 | 
			
		||||
                      class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-gray-200 dark:bg-neutral-400 cursor-pointer hover:bg-gray-300 dark:hover:bg-neutral-500 transition-all">
 | 
			
		||||
                      <div class="rounded-full w-4 h-4 m-1 bg-white"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
 | 
			
		||||
                    <!-- Show QR-->
 | 
			
		||||
                    <button class="align-middle bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                    <button
 | 
			
		||||
                      class="align-middle bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Show QR Code" @click="qrcode = `./api/wireguard/client/${client.id}/qrcode.svg`">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -255,7 +229,7 @@
 | 
			
		|||
 | 
			
		||||
                    <!-- Download Config -->
 | 
			
		||||
                    <a :href="'./api/wireguard/client/' + client.id + '/configuration'" download
 | 
			
		||||
                      class="align-middle inline-block bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                      class="align-middle inline-block bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Download Configuration">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -265,7 +239,8 @@
 | 
			
		|||
                    </a>
 | 
			
		||||
 | 
			
		||||
                    <!-- Delete -->
 | 
			
		||||
                    <button class="align-middle bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                    <button
 | 
			
		||||
                      class="align-middle bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Delete Client" @click="clientDelete = client">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
| 
						 | 
				
			
			@ -280,9 +255,10 @@
 | 
			
		|||
 | 
			
		||||
            </div>
 | 
			
		||||
            <div v-if="clients && clients.length === 0">
 | 
			
		||||
              <p class="text-center m-10 text-gray-400 text-sm">There are no clients yet.<br /><br />
 | 
			
		||||
              <p class="text-center m-10 text-gray-400 dark:text-neutral-400 text-sm">
 | 
			
		||||
                There are no clients yet.<br /><br />
 | 
			
		||||
                <button @click="clientCreate = true; clientCreateName = '';"
 | 
			
		||||
                  class="bg-red-800 text-white hover:bg-red-700 border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                  class="bg-red-800 hover:bg-red-700 text-white border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                  <svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                    stroke="currentColor">
 | 
			
		||||
                    <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
| 
						 | 
				
			
			@ -292,7 +268,7 @@
 | 
			
		|||
                </button>
 | 
			
		||||
              </p>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div v-if="clients === null" class="text-gray-200 p-5">
 | 
			
		||||
            <div v-if="clients === null" class="text-gray-200 dark:text-red-300 p-5">
 | 
			
		||||
              <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
                fill="currentColor">
 | 
			
		||||
                <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
 | 
			
		||||
| 
						 | 
				
			
			@ -308,7 +284,8 @@
 | 
			
		|||
        <div v-if="qrcode">
 | 
			
		||||
          <div class="bg-black bg-opacity-50 fixed top-0 right-0 left-0 bottom-0 flex items-center justify-center z-20">
 | 
			
		||||
            <div class="bg-white rounded-md shadow-lg relative p-8">
 | 
			
		||||
              <button @click="qrcode = null" class="absolute right-4 top-4 text-gray-600 hover:text-gray-800">
 | 
			
		||||
              <button @click="qrcode = null"
 | 
			
		||||
                class="absolute right-4 top-4 text-gray-600 dark:text-neutral-500 hover:text-gray-800 dark:hover:text-neutral-700">
 | 
			
		||||
                <svg class="w-8" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                  stroke="currentColor">
 | 
			
		||||
                  <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
 | 
			
		||||
| 
						 | 
				
			
			@ -321,7 +298,7 @@
 | 
			
		|||
 | 
			
		||||
        <!-- Create Dialog -->
 | 
			
		||||
        <div v-if="clientCreate" class="fixed z-10 inset-0 overflow-y-auto">
 | 
			
		||||
          <div class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
          <div class="flex items-center justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
            <!--
 | 
			
		||||
        Background overlay, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -333,7 +310,7 @@
 | 
			
		|||
          To: "opacity-0"
 | 
			
		||||
      -->
 | 
			
		||||
            <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 opacity-75"></div>
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 dark:bg-black opacity-75 dark:opacity-50"></div>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <!-- This element is to trick the browser into centering the modal contents. -->
 | 
			
		||||
| 
						 | 
				
			
			@ -342,49 +319,50 @@
 | 
			
		|||
        Modal panel, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
        Entering: "ease-out duration-300"
 | 
			
		||||
          From: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          From: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
        Leaving: "ease-in duration-200"
 | 
			
		||||
          From: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          From: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
      -->
 | 
			
		||||
            <div
 | 
			
		||||
              class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full"
 | 
			
		||||
              class="inline-block align-bottom bg-white dark:bg-neutral-700 rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg w-full"
 | 
			
		||||
              role="dialog" aria-modal="true" aria-labelledby="modal-headline">
 | 
			
		||||
              <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
              <div class="bg-white dark:bg-neutral-700 px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
                <div class="sm:flex sm:items-start">
 | 
			
		||||
                  <div
 | 
			
		||||
                    class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-red-800 sm:mx-0 sm:h-10 sm:w-10">
 | 
			
		||||
                    <svg class="h-6 w-6 text-white" inline xmlns="http://www.w3.org/2000/svg" fill="none"
 | 
			
		||||
                      viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                    <svg class="h-6 w-6 text-white" inline xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                      fill="none" viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                      <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                        d="M12 6v6m0 0v6m0-6h6m-6 0H6" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="flex-grow mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900" id="modal-headline">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900 dark:text-neutral-200" id="modal-headline">
 | 
			
		||||
                      New Client
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    <div class="mt-2">
 | 
			
		||||
                      <p class="text-sm text-gray-500">
 | 
			
		||||
                        <input class="rounded p-2 border-2 border-gray-100 focus:border-gray-200 outline-none w-full"
 | 
			
		||||
                        <input
 | 
			
		||||
                          class="rounded p-2 border-2 dark:bg-neutral-700 dark:text-neutral-200 border-gray-100 dark:border-neutral-600 focus:border-gray-200 focus:dark:border-neutral-500 dark:placeholder:text-neutral-400 outline-none w-full"
 | 
			
		||||
                          type="text" v-model.trim="clientCreateName" placeholder="Name" />
 | 
			
		||||
                      </p>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
              <div class="bg-gray-50 dark:bg-neutral-700 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
                <button v-if="clientCreateName.length" type="button" @click="createClient(); clientCreate = null"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-800 text-base font-medium text-white hover:bg-red-700 focus:outline-none sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Create
 | 
			
		||||
                </button>
 | 
			
		||||
                <button v-else type="button"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-200 text-base font-medium text-white sm:ml-3 sm:w-auto sm:text-sm cursor-not-allowed">
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-200 dark:bg-neutral-400 text-base font-medium text-white dark:text-neutral-300 sm:ml-3 sm:w-auto sm:text-sm cursor-not-allowed">
 | 
			
		||||
                  Create
 | 
			
		||||
                </button>
 | 
			
		||||
                <button type="button" @click="clientCreate = null"
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 dark:border-neutral-500 shadow-sm px-4 py-2 bg-white dark:bg-neutral-500 text-base font-medium text-gray-700 dark:text-neutral-50 hover:bg-gray-50 dark:hover:bg-neutral-600 dark:hover:border-neutral-600 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Cancel
 | 
			
		||||
                </button>
 | 
			
		||||
              </div>
 | 
			
		||||
| 
						 | 
				
			
			@ -394,7 +372,7 @@
 | 
			
		|||
 | 
			
		||||
        <!-- Delete Dialog -->
 | 
			
		||||
        <div v-if="clientDelete" class="fixed z-10 inset-0 overflow-y-auto">
 | 
			
		||||
          <div class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
          <div class="flex items-center justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
            <!--
 | 
			
		||||
        Background overlay, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -406,7 +384,7 @@
 | 
			
		|||
          To: "opacity-0"
 | 
			
		||||
      -->
 | 
			
		||||
            <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 opacity-75"></div>
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 dark:bg-black opacity-75 dark:opacity-50"></div>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <!-- This element is to trick the browser into centering the modal contents. -->
 | 
			
		||||
| 
						 | 
				
			
			@ -415,16 +393,16 @@
 | 
			
		|||
        Modal panel, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
        Entering: "ease-out duration-300"
 | 
			
		||||
          From: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          From: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
        Leaving: "ease-in duration-200"
 | 
			
		||||
          From: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          From: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
      -->
 | 
			
		||||
            <div
 | 
			
		||||
              class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full"
 | 
			
		||||
              class="inline-block align-bottom bg-white dark:bg-neutral-700 rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg w-full"
 | 
			
		||||
              role="dialog" aria-modal="true" aria-labelledby="modal-headline">
 | 
			
		||||
              <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
              <div class="bg-white dark:bg-neutral-700 px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
                <div class="sm:flex sm:items-start">
 | 
			
		||||
                  <div
 | 
			
		||||
                    class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-red-100 sm:mx-0 sm:h-10 sm:w-10">
 | 
			
		||||
| 
						 | 
				
			
			@ -436,11 +414,11 @@
 | 
			
		|||
                    </svg>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900" id="modal-headline">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900 dark:text-neutral-200" id="modal-headline">
 | 
			
		||||
                      Delete Client
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    <div class="mt-2">
 | 
			
		||||
                      <p class="text-sm text-gray-500">
 | 
			
		||||
                      <p class="text-sm text-gray-500 dark:text-neutral-300">
 | 
			
		||||
                        Are you sure you want to delete <strong>{{clientDelete.name}}</strong>?
 | 
			
		||||
                        This action cannot be undone.
 | 
			
		||||
                      </p>
 | 
			
		||||
| 
						 | 
				
			
			@ -448,13 +426,13 @@
 | 
			
		|||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
              <div class="bg-gray-50 dark:bg-neutral-600 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
                <button type="button" @click="deleteClient(clientDelete); clientDelete = null"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 dark:bg-red-600 text-base font-medium text-white dark:text-white hover:bg-red-700 dark:hover:bg-red-700 focus:outline-none sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Delete
 | 
			
		||||
                </button>
 | 
			
		||||
                <button type="button" @click="clientDelete = null"
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 dark:border-neutral-500 shadow-sm px-4 py-2 bg-white dark:bg-neutral-500 text-base font-medium text-gray-700 dark:text-neutral-50 hover:bg-gray-50 dark:hover:bg-neutral-600 dark:hover:border-neutral-600 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Cancel
 | 
			
		||||
                </button>
 | 
			
		||||
              </div>
 | 
			
		||||
| 
						 | 
				
			
			@ -464,22 +442,23 @@
 | 
			
		|||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === false">
 | 
			
		||||
        <h1 class="text-4xl font-medium my-16 text-gray-700 text-center">WireGuard</h1>
 | 
			
		||||
        <h1 class="text-4xl font-medium my-16 text-gray-700 dark:text-neutral-200 text-center">WireGuard</h1>
 | 
			
		||||
 | 
			
		||||
        <form @submit="login" class="shadow rounded-md bg-white mx-auto w-64 p-5 overflow-hidden mt-10">
 | 
			
		||||
        <form @submit="login"
 | 
			
		||||
          class="shadow rounded-md bg-white dark:bg-neutral-700 mx-auto w-64 p-5 overflow-hidden mt-10">
 | 
			
		||||
          <!-- Avatar -->
 | 
			
		||||
          <div class="h-20 w-20 mb-10 mt-5 mx-auto rounded-full bg-red-800 relative overflow-hidden">
 | 
			
		||||
            <svg class="w-10 h-10 m-5 text-white" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
              fill="currentColor">
 | 
			
		||||
          <div class="h-20 w-20 mb-10 mt-5 mx-auto rounded-full bg-red-800 dark:bg-red-800 relative overflow-hidden">
 | 
			
		||||
            <svg class="w-10 h-10 m-5 text-white dark:text-white" xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
              <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z" clip-rule="evenodd" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <input type="password" name="password" placeholder="Password" v-model="password"
 | 
			
		||||
            class="px-3 py-2 text-sm text-gray-500 mb-5 border-2 border-gray-100 rounded-lg w-full focus:border-red-800 outline-none" />
 | 
			
		||||
            class="px-3 py-2 text-sm dark:bg-neutral-700 text-gray-500 dark:text-gray-500 mb-5 border-2 border-gray-100 dark:border-neutral-800 rounded-lg w-full focus:border-red-800 dark:focus:border-red-800 dark:placeholder:text-neutral-400 outline-none" />
 | 
			
		||||
 | 
			
		||||
          <button v-if="authenticating"
 | 
			
		||||
            class="bg-red-800 w-full rounded shadow py-2 text-sm text-white cursor-not-allowed">
 | 
			
		||||
            class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed">
 | 
			
		||||
            <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
              fill="currentColor">
 | 
			
		||||
              <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
 | 
			
		||||
| 
						 | 
				
			
			@ -489,14 +468,15 @@
 | 
			
		|||
            </svg>
 | 
			
		||||
          </button>
 | 
			
		||||
          <input v-if="!authenticating && password" type="submit"
 | 
			
		||||
            class="bg-red-800 w-full rounded shadow py-2 text-sm text-white hover:bg-red-700 transition cursor-pointer"
 | 
			
		||||
            class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white hover:bg-red-700 dark:hover:bg-red-700 transition cursor-pointer"
 | 
			
		||||
            value="Sign In">
 | 
			
		||||
          <input v-if="!authenticating && !password" type="submit"
 | 
			
		||||
            class="bg-gray-200 w-full rounded shadow py-2 text-sm text-white cursor-not-allowed" value="Sign In">
 | 
			
		||||
            class="bg-gray-200 dark:bg-neutral-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed"
 | 
			
		||||
            value="Sign In">
 | 
			
		||||
        </form>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === null" class="text-gray-300 pt-24 pb-12">
 | 
			
		||||
      <div v-if="authenticated === null" class="text-gray-300 dark:text-red-300 pt-24 pb-12">
 | 
			
		||||
 | 
			
		||||
        <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
          fill="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -510,7 +490,7 @@
 | 
			
		|||
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <p v-cloak class="text-center m-10 text-gray-300 text-xs">Made by <a target="_blank" class="hover:underline"
 | 
			
		||||
    <p v-cloak class="text-center m-10 text-gray-300 dark:text-neutral-600 text-xs">Made by <a target="_blank" class="hover:underline"
 | 
			
		||||
        href="https://emilenijssen.nl/?ref=wg-easy">Emile Nijssen</a> · <a class="hover:underline"
 | 
			
		||||
        href="https://github.com/sponsors/WeeJeWel" target="_blank">Donate</a> · <a class="hover:underline"
 | 
			
		||||
        href="https://github.com/wg-easy/wg-easy" target="_blank">GitHub</a></p>
 | 
			
		||||
| 
						 | 
				
			
			@ -519,10 +499,12 @@
 | 
			
		|||
  </div>
 | 
			
		||||
 | 
			
		||||
  <script src="./js/vendor/vue.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/md5.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/apexcharts.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/vue-apexcharts.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/sha512.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/timeago.min.js"></script>
 | 
			
		||||
  <script src="./js/api.js"></script>
 | 
			
		||||
  <script src="./js/app.js"></script>
 | 
			
		||||
</body>
 | 
			
		||||
 | 
			
		||||
</html>
 | 
			
		||||
</html>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -62,7 +62,7 @@ class API {
 | 
			
		|||
    return this.call({
 | 
			
		||||
      method: 'get',
 | 
			
		||||
      path: '/wireguard/client',
 | 
			
		||||
    }).then(clients => clients.map(client => ({
 | 
			
		||||
    }).then((clients) => clients.map((client) => ({
 | 
			
		||||
      ...client,
 | 
			
		||||
      createdAt: new Date(client.createdAt),
 | 
			
		||||
      updatedAt: new Date(client.updatedAt),
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -45,6 +45,8 @@ new Vue({
 | 
			
		|||
    currentRelease: null,
 | 
			
		||||
    latestRelease: null,
 | 
			
		||||
 | 
			
		||||
    isDark: null,
 | 
			
		||||
 | 
			
		||||
    chartOptions: {
 | 
			
		||||
      chart: {
 | 
			
		||||
        background: 'transparent',
 | 
			
		||||
| 
						 | 
				
			
			@ -112,7 +114,7 @@ new Vue({
 | 
			
		|||
    },
 | 
			
		||||
  },
 | 
			
		||||
  methods: {
 | 
			
		||||
    dateTime: value => {
 | 
			
		||||
    dateTime: (value) => {
 | 
			
		||||
      return new Intl.DateTimeFormat(undefined, {
 | 
			
		||||
        year: 'numeric',
 | 
			
		||||
        month: 'short',
 | 
			
		||||
| 
						 | 
				
			
			@ -127,9 +129,9 @@ new Vue({
 | 
			
		|||
      if (!this.authenticated) return;
 | 
			
		||||
 | 
			
		||||
      const clients = await this.api.getClients();
 | 
			
		||||
      this.clients = clients.map(client => {
 | 
			
		||||
      this.clients = clients.map((client) => {
 | 
			
		||||
        if (client.name.includes('@') && client.name.includes('.')) {
 | 
			
		||||
          client.avatar = `https://www.gravatar.com/avatar/${md5(client.name)}?d=blank`;
 | 
			
		||||
          client.avatar = `https://www.gravatar.com/avatar/${sha512(client.name)}?d=blank`;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!this.clientsPersist[client.id]) {
 | 
			
		||||
| 
						 | 
				
			
			@ -186,7 +188,7 @@ new Vue({
 | 
			
		|||
          this.requiresPassword = session.requiresPassword;
 | 
			
		||||
          return this.refresh();
 | 
			
		||||
        })
 | 
			
		||||
        .catch(err => {
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          alert(err.message || err.toString());
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
| 
						 | 
				
			
			@ -202,7 +204,7 @@ new Vue({
 | 
			
		|||
          this.authenticated = false;
 | 
			
		||||
          this.clients = null;
 | 
			
		||||
        })
 | 
			
		||||
        .catch(err => {
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          alert(err.message || err.toString());
 | 
			
		||||
        });
 | 
			
		||||
    },
 | 
			
		||||
| 
						 | 
				
			
			@ -211,54 +213,69 @@ new Vue({
 | 
			
		|||
      if (!name) return;
 | 
			
		||||
 | 
			
		||||
      this.api.createClient({ name })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    deleteClient(client) {
 | 
			
		||||
      this.api.deleteClient({ clientId: client.id })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    enableClient(client) {
 | 
			
		||||
      this.api.enableClient({ clientId: client.id })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    disableClient(client) {
 | 
			
		||||
      this.api.disableClient({ clientId: client.id })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    updateClientName(client, name) {
 | 
			
		||||
      this.api.updateClientName({ clientId: client.id, name })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    updateClientAddress(client, address) {
 | 
			
		||||
      this.api.updateClientAddress({ clientId: client.id, address })
 | 
			
		||||
        .catch(err => alert(err.message || err.toString()))
 | 
			
		||||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    toggleTheme() {
 | 
			
		||||
      if (this.isDark) {
 | 
			
		||||
        localStorage.theme = 'light';
 | 
			
		||||
        document.documentElement.classList.remove('dark');
 | 
			
		||||
      } else {
 | 
			
		||||
        localStorage.theme = 'dark';
 | 
			
		||||
        document.documentElement.classList.add('dark');
 | 
			
		||||
      }
 | 
			
		||||
      this.isDark = !this.isDark;
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  filters: {
 | 
			
		||||
    bytes,
 | 
			
		||||
    timeago: value => {
 | 
			
		||||
    timeago: (value) => {
 | 
			
		||||
      return timeago().format(value);
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  mounted() {
 | 
			
		||||
    this.isDark = false;
 | 
			
		||||
    if (localStorage.theme === 'dark') {
 | 
			
		||||
      this.isDark = true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.api = new API();
 | 
			
		||||
    this.api.getSession()
 | 
			
		||||
      .then(session => {
 | 
			
		||||
      .then((session) => {
 | 
			
		||||
        this.authenticated = session.authenticated;
 | 
			
		||||
        this.requiresPassword = session.requiresPassword;
 | 
			
		||||
        this.refresh({
 | 
			
		||||
          updateCharts: true,
 | 
			
		||||
        }).catch(err => {
 | 
			
		||||
        }).catch((err) => {
 | 
			
		||||
          alert(err.message || err.toString());
 | 
			
		||||
        });
 | 
			
		||||
      })
 | 
			
		||||
      .catch(err => {
 | 
			
		||||
      .catch((err) => {
 | 
			
		||||
        alert(err.message || err.toString());
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -271,8 +288,8 @@ new Vue({
 | 
			
		|||
    Promise.resolve().then(async () => {
 | 
			
		||||
      const currentRelease = await this.api.getRelease();
 | 
			
		||||
      const latestRelease = await fetch('https://wg-easy.github.io/wg-easy/changelog.json')
 | 
			
		||||
        .then(res => res.json())
 | 
			
		||||
        .then(releases => {
 | 
			
		||||
        .then((res) => res.json())
 | 
			
		||||
        .then((releases) => {
 | 
			
		||||
          const releasesArray = Object.entries(releases).map(([version, changelog]) => ({
 | 
			
		||||
            version: parseInt(version, 10),
 | 
			
		||||
            changelog,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										14
									
								
								src/www/js/vendor/apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/www/js/vendor/apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1
									
								
								src/www/js/vendor/md5.min.js
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								src/www/js/vendor/md5.min.js
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -1 +0,0 @@
 | 
			
		|||
!function(n){"use strict";function d(n,t){var r=(65535&n)+(65535&t);return(n>>16)+(t>>16)+(r>>16)<<16|65535&r}function f(n,t,r,e,o,u){return d((c=d(d(t,n),d(e,u)))<<(f=o)|c>>>32-f,r);var c,f}function l(n,t,r,e,o,u,c){return f(t&r|~t&e,n,t,o,u,c)}function v(n,t,r,e,o,u,c){return f(t&e|r&~e,n,t,o,u,c)}function g(n,t,r,e,o,u,c){return f(t^r^e,n,t,o,u,c)}function m(n,t,r,e,o,u,c){return f(r^(t|~e),n,t,o,u,c)}function i(n,t){var r,e,o,u;n[t>>5]|=128<<t%32,n[14+(t+64>>>9<<4)]=t;for(var c=1732584193,f=-271733879,i=-1732584194,a=271733878,h=0;h<n.length;h+=16)c=l(r=c,e=f,o=i,u=a,n[h],7,-680876936),a=l(a,c,f,i,n[h+1],12,-389564586),i=l(i,a,c,f,n[h+2],17,606105819),f=l(f,i,a,c,n[h+3],22,-1044525330),c=l(c,f,i,a,n[h+4],7,-176418897),a=l(a,c,f,i,n[h+5],12,1200080426),i=l(i,a,c,f,n[h+6],17,-1473231341),f=l(f,i,a,c,n[h+7],22,-45705983),c=l(c,f,i,a,n[h+8],7,1770035416),a=l(a,c,f,i,n[h+9],12,-1958414417),i=l(i,a,c,f,n[h+10],17,-42063),f=l(f,i,a,c,n[h+11],22,-1990404162),c=l(c,f,i,a,n[h+12],7,1804603682),a=l(a,c,f,i,n[h+13],12,-40341101),i=l(i,a,c,f,n[h+14],17,-1502002290),c=v(c,f=l(f,i,a,c,n[h+15],22,1236535329),i,a,n[h+1],5,-165796510),a=v(a,c,f,i,n[h+6],9,-1069501632),i=v(i,a,c,f,n[h+11],14,643717713),f=v(f,i,a,c,n[h],20,-373897302),c=v(c,f,i,a,n[h+5],5,-701558691),a=v(a,c,f,i,n[h+10],9,38016083),i=v(i,a,c,f,n[h+15],14,-660478335),f=v(f,i,a,c,n[h+4],20,-405537848),c=v(c,f,i,a,n[h+9],5,568446438),a=v(a,c,f,i,n[h+14],9,-1019803690),i=v(i,a,c,f,n[h+3],14,-187363961),f=v(f,i,a,c,n[h+8],20,1163531501),c=v(c,f,i,a,n[h+13],5,-1444681467),a=v(a,c,f,i,n[h+2],9,-51403784),i=v(i,a,c,f,n[h+7],14,1735328473),c=g(c,f=v(f,i,a,c,n[h+12],20,-1926607734),i,a,n[h+5],4,-378558),a=g(a,c,f,i,n[h+8],11,-2022574463),i=g(i,a,c,f,n[h+11],16,1839030562),f=g(f,i,a,c,n[h+14],23,-35309556),c=g(c,f,i,a,n[h+1],4,-1530992060),a=g(a,c,f,i,n[h+4],11,1272893353),i=g(i,a,c,f,n[h+7],16,-155497632),f=g(f,i,a,c,n[h+10],23,-1094730640),c=g(c,f,i,a,n[h+13],4,681279174),a=g(a,c,f,i,n[h],11,-358537222),i=g(i,a,c,f,n[h+3],16,-722521979),f=g(f,i,a,c,n[h+6],23,76029189),c=g(c,f,i,a,n[h+9],4,-640364487),a=g(a,c,f,i,n[h+12],11,-421815835),i=g(i,a,c,f,n[h+15],16,530742520),c=m(c,f=g(f,i,a,c,n[h+2],23,-995338651),i,a,n[h],6,-198630844),a=m(a,c,f,i,n[h+7],10,1126891415),i=m(i,a,c,f,n[h+14],15,-1416354905),f=m(f,i,a,c,n[h+5],21,-57434055),c=m(c,f,i,a,n[h+12],6,1700485571),a=m(a,c,f,i,n[h+3],10,-1894986606),i=m(i,a,c,f,n[h+10],15,-1051523),f=m(f,i,a,c,n[h+1],21,-2054922799),c=m(c,f,i,a,n[h+8],6,1873313359),a=m(a,c,f,i,n[h+15],10,-30611744),i=m(i,a,c,f,n[h+6],15,-1560198380),f=m(f,i,a,c,n[h+13],21,1309151649),c=m(c,f,i,a,n[h+4],6,-145523070),a=m(a,c,f,i,n[h+11],10,-1120210379),i=m(i,a,c,f,n[h+2],15,718787259),f=m(f,i,a,c,n[h+9],21,-343485551),c=d(c,r),f=d(f,e),i=d(i,o),a=d(a,u);return[c,f,i,a]}function a(n){for(var t="",r=32*n.length,e=0;e<r;e+=8)t+=String.fromCharCode(n[e>>5]>>>e%32&255);return t}function h(n){var t=[];for(t[(n.length>>2)-1]=void 0,e=0;e<t.length;e+=1)t[e]=0;for(var r=8*n.length,e=0;e<r;e+=8)t[e>>5]|=(255&n.charCodeAt(e/8))<<e%32;return t}function e(n){for(var t,r="0123456789abcdef",e="",o=0;o<n.length;o+=1)t=n.charCodeAt(o),e+=r.charAt(t>>>4&15)+r.charAt(15&t);return e}function r(n){return unescape(encodeURIComponent(n))}function o(n){return a(i(h(t=r(n)),8*t.length));var t}function u(n,t){return function(n,t){var r,e,o=h(n),u=[],c=[];for(u[15]=c[15]=void 0,16<o.length&&(o=i(o,8*n.length)),r=0;r<16;r+=1)u[r]=909522486^o[r],c[r]=1549556828^o[r];return e=i(u.concat(h(t)),512+8*t.length),a(i(c.concat(e),640))}(r(n),r(t))}function t(n,t,r){return t?r?u(t,n):e(u(t,n)):r?o(n):e(o(n))}"function"==typeof define&&define.amd?define(function(){return t}):"object"==typeof module&&module.exports?module.exports=t:n.md5=t}(this);
 | 
			
		||||
							
								
								
									
										1
									
								
								src/www/js/vendor/sha512.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/www/js/vendor/sha512.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										7
									
								
								src/www/js/vendor/vue-apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/www/js/vendor/vue-apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,7 @@
 | 
			
		|||
/**
 | 
			
		||||
 * Minified by jsDelivr using Terser v5.7.1.
 | 
			
		||||
 * Original file: /npm/vue-apexcharts@1.6.2/dist/vue-apexcharts.js
 | 
			
		||||
 *
 | 
			
		||||
 * Do NOT use SRI with dynamically generated files! More information: https://www.jsdelivr.com/using-sri-with-dynamic-files
 | 
			
		||||
 */
 | 
			
		||||
!function (t, e) { "object" == typeof exports && "undefined" != typeof module ? module.exports = e(require("apexcharts/dist/apexcharts.min")) : "function" == typeof define && define.amd ? define(["apexcharts/dist/apexcharts.min"], e) : t.VueApexCharts = e(t.ApexCharts) }(this, (function (t) { "use strict"; function e(t) { return (e = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (t) { return typeof t } : function (t) { return t && "function" == typeof Symbol && t.constructor === Symbol && t !== Symbol.prototype ? "symbol" : typeof t })(t) } function n(t, e, n) { return e in t ? Object.defineProperty(t, e, { value: n, enumerable: !0, configurable: !0, writable: !0 }) : t[e] = n, t } t = t && t.hasOwnProperty("default") ? t.default : t; var i = { props: { options: { type: Object }, type: { type: String }, series: { type: Array, required: !0, default: function () { return [] } }, width: { default: "100%" }, height: { default: "auto" } }, data: function () { return { chart: null } }, beforeMount: function () { window.ApexCharts = t }, mounted: function () { this.init() }, created: function () { var t = this; this.$watch("options", (function (e) { !t.chart && e ? t.init() : t.chart.updateOptions(t.options) })), this.$watch("series", (function (e) { !t.chart && e ? t.init() : t.chart.updateSeries(t.series) }));["type", "width", "height"].forEach((function (e) { t.$watch(e, (function () { t.refresh() })) })) }, beforeDestroy: function () { this.chart && this.destroy() }, render: function (t) { return t("div") }, methods: { init: function () { var e = this, n = { chart: { type: this.type || this.options.chart.type || "line", height: this.height, width: this.width, events: {} }, series: this.series }; Object.keys(this.$listeners).forEach((function (t) { n.chart.events[t] = e.$listeners[t] })); var i = this.extend(this.options, n); return this.chart = new t(this.$el, i), this.chart.render() }, isObject: function (t) { return t && "object" === e(t) && !Array.isArray(t) && null != t }, extend: function (t, e) { var i = this; "function" != typeof Object.assign && (Object.assign = function (t) { if (null == t) throw new TypeError("Cannot convert undefined or null to object"); for (var e = Object(t), n = 1; n < arguments.length; n++) { var i = arguments[n]; if (null != i) for (var r in i) i.hasOwnProperty(r) && (e[r] = i[r]) } return e }); var r = Object.assign({}, t); return this.isObject(t) && this.isObject(e) && Object.keys(e).forEach((function (o) { i.isObject(e[o]) && o in t ? r[o] = i.extend(t[o], e[o]) : Object.assign(r, n({}, o, e[o])) })), r }, refresh: function () { return this.destroy(), this.init() }, destroy: function () { this.chart.destroy() }, updateSeries: function (t, e) { return this.chart.updateSeries(t, e) }, updateOptions: function (t, e, n, i) { return this.chart.updateOptions(t, e, n, i) }, toggleSeries: function (t) { return this.chart.toggleSeries(t) }, showSeries: function (t) { this.chart.showSeries(t) }, hideSeries: function (t) { this.chart.hideSeries(t) }, appendSeries: function (t, e) { return this.chart.appendSeries(t, e) }, resetSeries: function () { this.chart.resetSeries() }, zoomX: function (t, e) { this.chart.zoomX(t, e) }, toggleDataPointSelection: function (t, e) { this.chart.toggleDataPointSelection(t, e) }, appendData: function (t) { return this.chart.appendData(t) }, addText: function (t) { this.chart.addText(t) }, addImage: function (t) { this.chart.addImage(t) }, addShape: function (t) { this.chart.addShape(t) }, dataURI: function () { return this.chart.dataURI() }, setLocale: function (t) { return this.chart.setLocale(t) }, addXaxisAnnotation: function (t, e) { this.chart.addXaxisAnnotation(t, e) }, addYaxisAnnotation: function (t, e) { this.chart.addYaxisAnnotation(t, e) }, addPointAnnotation: function (t, e) { this.chart.addPointAnnotation(t, e) }, removeAnnotation: function (t, e) { this.chart.removeAnnotation(t, e) }, clearAnnotations: function () { this.chart.clearAnnotations() } } }; return window.ApexCharts = t, i.install = function (e) { e.ApexCharts = t, window.ApexCharts = t, Object.defineProperty(e.prototype, "$apexcharts", { get: function () { return t } }) }, i }));
 | 
			
		||||
| 
						 | 
				
			
			@ -8,4 +8,4 @@
 | 
			
		|||
      "type": "image/png"
 | 
			
		||||
    }
 | 
			
		||||
  ]
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										3
									
								
								src/www/src/css/app.css
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/www/src/css/app.css
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,3 @@
 | 
			
		|||
@tailwind base;
 | 
			
		||||
@tailwind components;
 | 
			
		||||
@tailwind utilities;
 | 
			
		||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue