Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
538974e0fe | ||
|
|
0d997a0cc2 | ||
|
|
6c7cdc6421 | ||
|
|
e73e85c5a9 | ||
|
|
209e01deb6 | ||
|
|
e8d2eb0aa1 | ||
|
|
1e471affd0 | ||
|
|
49fdd9e49b | ||
|
|
d52e379d5f | ||
|
|
cfb1ddb4e5 | ||
|
|
cad1af18c1 | ||
|
|
c93072b1c6 | ||
| 7b87fdcc3f | |||
| a2275f4444 | |||
| 798d3fcdb5 | |||
|
|
9aa8378599 | ||
|
|
f0c28a4961 |
44
.github/workflows/publish-release.yml
vendored
44
.github/workflows/publish-release.yml
vendored
@@ -1,19 +1,41 @@
|
||||
---
|
||||
name: Publish Release
|
||||
# .github/workflows/semantic-versioning-release.yml
|
||||
name: Automated Semantic Versioning Release
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- main
|
||||
|
||||
jobs:
|
||||
publish-release:
|
||||
release:
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: write # for creating tags and releases
|
||||
issues: write # for commenting on issues
|
||||
pull-requests: write # for commenting on PRs
|
||||
steps:
|
||||
- name: Publish Release
|
||||
uses: ncipollo/release-action@v1
|
||||
- name: Checkout code
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
token: ${{secrets.GITHUB_TOKEN}}
|
||||
draft: false
|
||||
prerelease: false
|
||||
name: "0.0.${{ github.run_number }}"
|
||||
tag: "0.0.${{ github.run_number }}"
|
||||
body: "Release 0.0.${{ github.run_number }}"
|
||||
# This is required to analyze the full commit history
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Automated Semantic Release
|
||||
# This action wraps the popular semantic-release tool
|
||||
uses: cycjimmy/semantic-release-action@v4
|
||||
with:
|
||||
# You can specify the branches to release from
|
||||
branch: main
|
||||
extra_plugins: |
|
||||
@semantic-release/changelog
|
||||
@semantic-release/git
|
||||
env:
|
||||
# GITHUB_TOKEN is required for authentication
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
||||
# To make this work, you must follow the Conventional Commits specification.
|
||||
# Examples:
|
||||
# - fix: correct a typo in the documentation
|
||||
# - feat: add a new user authentication endpoint
|
||||
# - feat(api): add rate limiting
|
||||
# BREAKING CHANGE: The API now returns 429 when rate limit is exceeded.
|
||||
|
||||
13
.releaserc.json
Normal file
13
.releaserc.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"branches": ["main"],
|
||||
"plugins": [
|
||||
"@semantic-release/commit-analyzer",
|
||||
"@semantic-release/release-notes-generator",
|
||||
["@semantic-release/changelog", { "changelogFile": "CHANGELOG.md" }],
|
||||
"@semantic-release/github",
|
||||
["@semantic-release/git", {
|
||||
"assets": ["CHANGELOG.md"],
|
||||
"message": "chore(release): ${nextRelease.version} [skip ci]\n\n${nextRelease.notes}"
|
||||
}]
|
||||
]
|
||||
}
|
||||
79
CHANGELOG.md
Normal file
79
CHANGELOG.md
Normal file
@@ -0,0 +1,79 @@
|
||||
# 1.0.0 (2026-02-07)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* add null/false guards, proper error handling, and VNC popup fix ([49fdd9e](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/49fdd9e49ba87bfb4b72dd741e15f790c1050033))
|
||||
* TestConnection for unsaved servers, traffic display, and cache-busting ([e8d2eb0](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/e8d2eb0aa1f173f13bb0b8d7dfca0acebb821ac7))
|
||||
* XSS escaping, null guards, JS bug fixes, and documentation updates ([6c7cdc6](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/6c7cdc6421678390746adcee4877a7ade8f2a061))
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* add client-side SSH Ed25519 key generator on order page ([209e01d](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/209e01deb6832dce76a307410fbab28b1e420093))
|
||||
* add VNC check, SSH key paste, resources panel, sliders, and self-service billing ([1e471af](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/1e471affd0ae9a68358afa5704523bce9bb413d0))
|
||||
* streamline network panel, conditional self-service, remove IP add endpoints ([e73e85c](https://github.com/EZSCALE/virtfusion-whmcs-module/commit/e73e85c5a9faa79b50e4949328c1d2a3cbc49ddf))
|
||||
|
||||
# Changelog
|
||||
|
||||
All notable changes to the VirtFusion Direct Provisioning Module for WHMCS.
|
||||
|
||||
## [0.0.18] - 2025-10-01
|
||||
|
||||
### Changed
|
||||
- Updated GitHub Actions publish workflow
|
||||
- Moved custom field SQL to `modify.sql` file
|
||||
- Minor code tweaks
|
||||
|
||||
## [0.0.17] - 2024-01-16
|
||||
|
||||
### Fixed
|
||||
- Fix in hooks.php (PR #2 by Prophet731)
|
||||
|
||||
## [0.0.16] - 2023-09-11
|
||||
|
||||
### Added
|
||||
- GitHub issue templates
|
||||
|
||||
## [0.0.15] - 2023-09-10
|
||||
|
||||
### Fixed
|
||||
- Typo fixes in module code
|
||||
|
||||
## [0.0.14] - 2023-09-10
|
||||
|
||||
### Fixed
|
||||
- Fix hook event registration placement
|
||||
|
||||
## [0.0.13] - 2023-09-10
|
||||
|
||||
### Added
|
||||
- Contributions from BlinkohHost
|
||||
- Database-first package ID lookup with API fallback by product name
|
||||
- Server build initialization on successful server creation
|
||||
|
||||
### Changed
|
||||
- Custom fields changed to not required
|
||||
- Removed linter workflow (not needed for this project)
|
||||
- Code cleanup
|
||||
|
||||
## [0.0.9] - 2023-09-10
|
||||
|
||||
### Changed
|
||||
- Refactored codebase to object-oriented architecture (OOP)
|
||||
- Updated README with badges and documentation
|
||||
|
||||
## [0.0.6] - 2023-09-10
|
||||
|
||||
### Added
|
||||
- Initial release
|
||||
- Core provisioning: server create, suspend, unsuspend, terminate
|
||||
- WHMCS hooks for dynamic OS template and SSH key dropdowns
|
||||
- Checkout validation for OS selection
|
||||
- Client area overview template with server information
|
||||
- Admin services tab with server ID management
|
||||
- Package change (upgrade/downgrade) support
|
||||
- Configurable option mapping for dynamic resource allocation
|
||||
- GitHub Actions CI/CD with semantic-release
|
||||
- Security policy (SECURITY.md)
|
||||
- License (GPL v3)
|
||||
114
CLAUDE.md
Normal file
114
CLAUDE.md
Normal file
@@ -0,0 +1,114 @@
|
||||
# CLAUDE.md
|
||||
|
||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||
|
||||
## Project Overview
|
||||
|
||||
VirtFusion Direct Provisioning Module for WHMCS — a PHP module that integrates WHMCS with the VirtFusion control panel API for automated VPS provisioning, management, and client self-service. No build system or package manager; the module is pure PHP installed by copying `modules/servers/VirtFusionDirect/` into a WHMCS installation.
|
||||
|
||||
## Development & Testing
|
||||
|
||||
There is no automated test suite, linter, or build step. Testing is manual:
|
||||
|
||||
- **Test connection:** WHMCS Admin → System Settings → Servers → Test Connection button
|
||||
- **Dry run validation:** `VirtFusionDirect_validateServerConfig()` tests configuration without creating a server
|
||||
- **Module logging:** WHMCS Admin → Utilities → Logs → Module Log captures all API calls and responses
|
||||
- **Server object viewer:** Admin services tab shows full JSON response from VirtFusion API
|
||||
|
||||
## Release Process
|
||||
|
||||
Releases are automated via GitHub Actions using semantic-release on pushes to `main`. Use **conventional commits**:
|
||||
- `fix:` → patch release
|
||||
- `feat:` → minor release
|
||||
- `BREAKING CHANGE:` in commit body → major release
|
||||
|
||||
## Architecture
|
||||
|
||||
**Namespace:** `WHMCS\Module\Server\VirtFusionDirect`
|
||||
|
||||
### Entry Points
|
||||
|
||||
| File | Purpose |
|
||||
|------|---------|
|
||||
| `VirtFusionDirect.php` | WHMCS module interface — non-namespaced functions (`VirtFusionDirect_CreateAccount()`, etc.) that delegate to library classes |
|
||||
| `client.php` | Client-facing AJAX API — authenticated by WHMCS session + service ownership validation |
|
||||
| `admin.php` | Admin-facing AJAX API — requires WHMCS admin authentication |
|
||||
| `hooks.php` | WHMCS hooks — checkout validation (OS selection), dynamic dropdown/slider injection, SSH key paste |
|
||||
|
||||
### Core Classes (in `lib/`)
|
||||
|
||||
| Class | Role |
|
||||
|-------|------|
|
||||
| `Module` | Base class with API integration, auth checks, power/network/VNC/backup/resource/self-service methods. All client/admin actions route through here. |
|
||||
| `ModuleFunctions` | Extends `Module`. Service lifecycle: create, suspend, unsuspend, terminate, change package, usage updates, client area rendering. |
|
||||
| `ConfigureService` | Extends `Module`. Order-time operations: package discovery, OS template fetching, server build initialization, SSH key retrieval and creation. |
|
||||
| `Database` | Static methods for `mod_virtfusion_direct` table operations and WHMCS DB queries. Auto-creates/migrates schema on first use. |
|
||||
| `Curl` | HTTP client wrapper with Bearer token auth, SSL verification, 30s timeout. Methods: `get`, `post`, `put`, `patch`, `delete`. |
|
||||
| `ServerResource` | Transforms VirtFusion API response into flat key-value format for Smarty templates. |
|
||||
| `AdminHTML` | Static methods generating admin services tab HTML (server ID editor, JSON viewer, action buttons). |
|
||||
| `Log` | Thin wrapper around WHMCS module logging. |
|
||||
|
||||
### Class Hierarchy
|
||||
|
||||
`ModuleFunctions` and `ConfigureService` both extend `Module`. Most business logic lives in `Module` — it handles API calls, auth, validation, and all feature-specific operations (power, network, VNC, backup, resource modification). `ModuleFunctions` orchestrates the WHMCS service lifecycle (provisioning flow, suspension, termination).
|
||||
|
||||
### Client-Side
|
||||
|
||||
- **`templates/overview.tpl`** — Smarty template for client area (server info, power, network, rebuild, resources, VNC, self-service billing, billing overview)
|
||||
- **`templates/js/module.js`** — Vanilla JS (1000+ lines) handling AJAX calls to `client.php`, DOM updates, status badges, power actions, all management UIs
|
||||
- **`templates/js/keygen.js`** — Client-side SSH Ed25519 key generator using Web Crypto API (loaded on checkout page)
|
||||
- **`templates/css/module.css`** — Cross-theme styles with Bootstrap 3/4/5 dual class support (`panel card`, `panel-body card-body`)
|
||||
|
||||
### Removed Features
|
||||
|
||||
- **Firewall** — Removed (non-functional; rulesets must be created in VirtFusion admin panel)
|
||||
- **IP add buttons** — Removed (`addIPv4`, `addIPv6` endpoints and UI); IPs are managed by VirtFusion during provisioning
|
||||
- **Upgrade/Downgrade link** — Removed from resources panel
|
||||
|
||||
### Data Flow: Server Creation
|
||||
|
||||
1. WHMCS calls `VirtFusionDirect_CreateAccount()` → `ModuleFunctions::createAccount()`
|
||||
2. Checks/creates VirtFusion user via external relation ID (WHMCS client ID)
|
||||
3. Reads configurable options (Package, Location, IPv4, Memory, CPU, Bandwidth, etc.)
|
||||
4. Dry-run validation → actual API POST to `/servers`
|
||||
5. Stores server ID in `mod_virtfusion_direct` table
|
||||
6. Updates WHMCS hosting record (IP, username, password, domain)
|
||||
7. Calls `ConfigureService::initServerBuild()` with selected OS + SSH key
|
||||
|
||||
### Configurable Option Mapping
|
||||
|
||||
Custom option names can be mapped in `config/ConfigOptionMapping.php` (copy from `-example.php`). Default mapping keys: `packageId`, `hypervisorId`, `ipv4`, `storage`, `memory`, `traffic`, `cpuCores`, `networkSpeedInbound`, `networkSpeedOutbound`, `networkProfile`, `storageProfile`.
|
||||
|
||||
## Security Patterns
|
||||
|
||||
- All PHP files start with `if (!defined("WHMCS")) die()` to prevent direct access
|
||||
- Client endpoints validate WHMCS session AND service ownership before any operation
|
||||
- API tokens stored encrypted in WHMCS server password field (decrypted via `localAPI('DecryptPassword')`)
|
||||
- Input validation: type casting, regex filtering, `filter_var()` for IP addresses
|
||||
- Output escaping: `htmlspecialchars()` in Smarty, `encodeURIComponent()` / `.text()` in JS
|
||||
- SSL verification enabled on all API calls (`CURLOPT_SSL_VERIFYPEER` + `CURLOPT_SSL_VERIFYHOST = 2`)
|
||||
|
||||
## VirtFusion API Compatibility
|
||||
|
||||
- **API reference (OpenAPI spec):** https://docs.virtfusion.com/api/openapi.yaml
|
||||
- **Base features:** VirtFusion v1.7.3+
|
||||
- **VNC console:** v6.1.0+
|
||||
- **Resource modification:** v6.2.0+
|
||||
- **Self-service billing:** Requires self-service feature enabled in VirtFusion
|
||||
|
||||
## Product Config Options
|
||||
|
||||
| Option | Name | Description | Default |
|
||||
|--------|------|-------------|---------|
|
||||
| configoption1 | Hypervisor Group ID | VirtFusion hypervisor group for server placement | 1 |
|
||||
| configoption2 | Package ID | VirtFusion package defining server resources | 1 |
|
||||
| configoption3 | Default IPv4 | Number of IPv4 addresses to assign (0-10) | 1 |
|
||||
| configoption4 | Self-Service Mode | 0=Disabled, 1=Hourly, 2=Resource Packs, 3=Both | 0 |
|
||||
| configoption5 | Auto Top-Off Threshold | Credit balance below which auto top-off triggers | 0 |
|
||||
| configoption6 | Auto Top-Off Amount | Credit amount to add on auto top-off | 100 |
|
||||
|
||||
## WHMCS Compatibility
|
||||
|
||||
- WHMCS 8.x+ (tested 8.0–8.10)
|
||||
- PHP 8.0+ with cURL extension
|
||||
- All WHMCS themes supported (Six, Twenty-One, Lagom, custom) via Bootstrap 3/4/5 dual classes
|
||||
661
README.md
661
README.md
@@ -1,90 +1,607 @@
|
||||
# VirtFusion Direct Provisioning Module for WHMCS
|
||||
|
||||
[](https://github.com/EZSCALE/virtfusion-whmcs-module/actions)
|
||||
[](https://github.com/EZSCALE/virtfusion-whmcs-module/actions)
|
||||

|
||||

|
||||

|
||||
|
||||
This module requires VirtFusion v1.7.3 or higher as this is what it's based on. Please refer to the
|
||||
official [documentation](https://docs.virtfusion.com/integrations/whmcs).
|
||||
A comprehensive WHMCS provisioning module for [VirtFusion](https://virtfusion.com) that enables automated VPS server provisioning, management, and client self-service directly from WHMCS.
|
||||
|
||||
## Table of Contents
|
||||
|
||||
- [Requirements](#requirements)
|
||||
- [Features](#features)
|
||||
- [Installation](#installation)
|
||||
- [Upgrading](#upgrading)
|
||||
- [Configuration](#configuration)
|
||||
- [Server Setup](#server-setup)
|
||||
- [Product Setup](#product-setup)
|
||||
- [Custom Fields](#custom-fields)
|
||||
- [Module Configuration Options](#module-configuration-options)
|
||||
- [Configurable Options (Dynamic Pricing)](#configurable-options-dynamic-pricing)
|
||||
- [Custom Option Name Mapping](#custom-option-name-mapping)
|
||||
- [Client Area Features](#client-area-features)
|
||||
- [Admin Area Features](#admin-area-features)
|
||||
- [Theme Compatibility](#theme-compatibility)
|
||||
- [API Endpoints Used](#api-endpoints-used)
|
||||
- [Usage Update (Cron)](#usage-update-cron)
|
||||
- [Troubleshooting](#troubleshooting)
|
||||
- [Known Issues](#known-issues)
|
||||
- [Security](#security)
|
||||
- [Contributing](#contributing)
|
||||
- [License](#license)
|
||||
|
||||
## Requirements
|
||||
|
||||
| Requirement | Minimum Version | Notes |
|
||||
|---|---|---|
|
||||
| **VirtFusion** | v1.7.3+ | v6.1.0+ required for VNC console |
|
||||
| **WHMCS** | 8.x+ | Tested with 8.0 through 8.10 |
|
||||
| **PHP** | 8.0+ | With cURL extension enabled |
|
||||
| **SSL** | Valid certificate | Required on VirtFusion panel |
|
||||
|
||||
You also need a VirtFusion API token with the following permissions:
|
||||
- Server management (create, read, update, delete, power, build)
|
||||
- User management (create, read, reset password, authentication tokens)
|
||||
- Package and template read access
|
||||
- Network management (if using IP management features)
|
||||
|
||||
## Features
|
||||
|
||||
### Server Provisioning
|
||||
- Automatic server creation with VirtFusion user account linking
|
||||
- Server suspension, unsuspension, and termination
|
||||
- Package/plan upgrades and downgrades
|
||||
- Configurable options mapping for dynamic resource allocation (CPU, RAM, disk, bandwidth, network speed)
|
||||
- **Dry run validation** - Test server creation parameters before provisioning
|
||||
- Automatic memory unit conversion (GB to MB for values < 1024)
|
||||
|
||||
### Client Area - Server Management
|
||||
- **Server Overview** - Real-time server info (hostname, IPs, resources) with status badge
|
||||
- **Power Management** - Start, restart, graceful shutdown, and force power off
|
||||
- **Control Panel SSO** - One-click login to VirtFusion panel
|
||||
- **Server Rebuild** - Reinstall with any available OS template
|
||||
- **Password Reset** - Reset VirtFusion panel login credentials
|
||||
- **Network Management** - View and remove IPv4 addresses; view IPv6 subnets
|
||||
- **Resources Panel** - Current memory, CPU, storage, traffic allocation with usage bars
|
||||
- **VNC Console** - Browser-based console access (panel auto-hides when VNC is disabled on the server)
|
||||
- **Self-Service Billing** - Credit balance display, usage breakdown, and credit top-up (when enabled)
|
||||
- **Bandwidth Usage** - Traffic usage display with allocation limits
|
||||
- **Billing Overview** - Product, billing cycle, dates, and payment information
|
||||
|
||||
### Admin Area
|
||||
- **Test Connection** - Verify API connectivity from WHMCS
|
||||
- **Server Data Display** - Live server information from VirtFusion
|
||||
- **Admin Impersonation** - Log into VirtFusion panel as server owner
|
||||
- **Server ID Management** - Editable Server ID for manual adjustments
|
||||
- **Server Object Viewer** - Full JSON response from VirtFusion API
|
||||
- **Validate Server Config** - Dry run server creation to check configuration
|
||||
- **Update Server Object** - Refresh cached server data from VirtFusion
|
||||
|
||||
### Ordering Process
|
||||
- Dynamic OS template dropdown populated from VirtFusion API
|
||||
- SSH key selection dropdown for users with saved keys, with option to paste a new public key
|
||||
- **SSH Ed25519 key generator** — Client-side keypair generation using Web Crypto API
|
||||
- Checkout validation ensuring OS selection before order placement
|
||||
- **Resource sliders** - Configurable option dropdowns are replaced with interactive range sliders
|
||||
- Compatible with all WHMCS order form templates
|
||||
|
||||
### Usage Tracking
|
||||
- **Automated bandwidth sync** - WHMCS daily cron pulls traffic usage from VirtFusion
|
||||
- **Disk usage sync** - Storage usage updated automatically
|
||||
- Visible in WHMCS client area and admin product details
|
||||
|
||||
### Backup Management
|
||||
- Assign backup plans to servers via the VirtFusion API
|
||||
- Remove backup plans from servers
|
||||
|
||||
### Resource Modification
|
||||
- In-place modification of server resources (memory, CPU cores, traffic)
|
||||
- No server rebuild required for resource changes
|
||||
- **Package change** now also applies individual resource modifications from configurable options
|
||||
|
||||
### Self-Service Billing
|
||||
- Credit balance display and top-up from client area
|
||||
- Usage breakdown reporting
|
||||
- Auto top-off via WHMCS cron when credit falls below threshold
|
||||
- Self-service mode configurable per product (Hourly, Resource Packs, or Both)
|
||||
|
||||
## Installation
|
||||
|
||||
1. Download the latest release from the [releases](https://github.com/EZSCALE/virtfusion-whmcs-module/releases) page.
|
||||
2. Extract the contents of the archive and upload the modules folder to your WHMCS installation directory.
|
||||
### Step 1: Download & Install
|
||||
|
||||
## :heavy_exclamation_mark: Important Notes :heavy_exclamation_mark:
|
||||
Download the latest release from the [releases](https://github.com/EZSCALE/virtfusion-whmcs-module/releases) page, or install directly via the command line:
|
||||
|
||||
You must create two custom fields in WHMCS for this module to work. You need to configure the following custom fields on
|
||||
each product you want to use this module with.
|
||||
|
||||
| Field Name | Field Type | Description | Validation | Select Options | Admin Only | Required Field | Show on Order Form | Show on Invoice |
|
||||
|--------------------------|------------|--------------------------|-------------|----------------|------------|----------------|--------------------|-----------------|
|
||||
| Initial Operating System | Text Box | Set to whatever you want | Leave Blank | Leave Blank | :x: | :x: | :white_check_mark: | :x: |
|
||||
| Initial SSH Key | Text Box | Set to whatever you want | Leave Blank | Leave Blank | :x: | :x: | :white_check_mark: | :x: |
|
||||
|
||||
You can run this SQL query to create the custom fields:
|
||||
|
||||
```sql
|
||||
-- Insert records for Initial Operating System if they don't already exist
|
||||
INSERT INTO tblcustomfields
|
||||
(type, relid, fieldname, fieldtype, description, fieldoptions, regexpr, adminonly, required, showorder, showinvoice,
|
||||
sortorder, created_at, updated_at)
|
||||
SELECT 'product',
|
||||
id,
|
||||
'Initial Operating System',
|
||||
'text',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'on',
|
||||
'',
|
||||
0,
|
||||
UTC_TIMESTAMP(),
|
||||
UTC_TIMESTAMP()
|
||||
FROM tblproducts
|
||||
WHERE servertype = 'VirtFusionDirect'
|
||||
AND NOT EXISTS (SELECT 1
|
||||
FROM tblcustomfields
|
||||
WHERE fieldname = 'Initial Operating System'
|
||||
AND relid = tblproducts.id);
|
||||
|
||||
-- Insert records for Initial SSH Key if they don't already exist
|
||||
INSERT INTO tblcustomfields
|
||||
(type, relid, fieldname, fieldtype, description, fieldoptions, regexpr, adminonly, required, showorder, showinvoice,
|
||||
sortorder, created_at, updated_at)
|
||||
SELECT 'product',
|
||||
id,
|
||||
'Initial SSH Key',
|
||||
'text',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'on',
|
||||
'',
|
||||
0,
|
||||
UTC_TIMESTAMP(),
|
||||
UTC_TIMESTAMP()
|
||||
FROM tblproducts
|
||||
WHERE servertype = 'VirtFusionDirect'
|
||||
AND NOT EXISTS (SELECT 1
|
||||
FROM tblcustomfields
|
||||
WHERE fieldname = 'Initial SSH Key'
|
||||
AND relid = tblproducts.id);
|
||||
```bash
|
||||
cd /tmp
|
||||
git clone https://github.com/EZSCALE/virtfusion-whmcs-module.git
|
||||
rsync -ahP --delete /tmp/virtfusion-whmcs-module/modules/servers/VirtFusionDirect/ /path/to/whmcs/modules/servers/VirtFusionDirect/
|
||||
rm -rf /tmp/virtfusion-whmcs-module
|
||||
```
|
||||
|
||||
Replace `/path/to/whmcs` with your actual WHMCS installation root.
|
||||
|
||||
## What does this module change?
|
||||
The resulting file structure should be:
|
||||
|
||||
This module changes the following things:
|
||||
```
|
||||
modules/servers/VirtFusionDirect/
|
||||
VirtFusionDirect.php # Main module file
|
||||
client.php # Client AJAX API
|
||||
admin.php # Admin AJAX API
|
||||
hooks.php # WHMCS hooks
|
||||
modify.sql # Custom field setup SQL
|
||||
lib/
|
||||
Module.php # Core module class
|
||||
ModuleFunctions.php # Provisioning functions
|
||||
ConfigureService.php # OS/SSH config service
|
||||
Database.php # Database operations
|
||||
Curl.php # HTTP client
|
||||
ServerResource.php # Data transformer
|
||||
AdminHTML.php # Admin interface HTML
|
||||
Log.php # Logging
|
||||
templates/
|
||||
overview.tpl # Client area template
|
||||
error.tpl # Error template
|
||||
css/module.css # Styles
|
||||
js/module.js # Client JavaScript
|
||||
js/keygen.js # SSH Ed25519 key generator
|
||||
config/
|
||||
ConfigOptionMapping-example.php # Config mapping example
|
||||
```
|
||||
|
||||
- Adds configurable options to the product configuration page to allow the user to select the operating system and add
|
||||
an ssh key to the initial deployment.
|
||||
### Step 2: Set Up Server in WHMCS
|
||||
|
||||
## TODO
|
||||
1. Go to **Configuration > System Settings > Servers**
|
||||
2. Click **Add New Server**
|
||||
3. Fill in:
|
||||
- **Name**: Anything descriptive (e.g., "VirtFusion Production")
|
||||
- **Hostname**: Your VirtFusion panel hostname (e.g., `cp.example.com`)
|
||||
- **Type**: VirtFusion Direct Provisioning
|
||||
- **Password/Access Hash**: Your VirtFusion API token
|
||||
4. Click **Test Connection** to verify
|
||||
5. Click **Save Changes**
|
||||
|
||||
- [ ] Add post checkout checks to ensure the user has selected an operating system and added a ssh key.
|
||||
### Step 3: Create Product
|
||||
|
||||
1. Go to **Configuration > System Settings > Products/Services**
|
||||
2. Create a new product or edit an existing one
|
||||
3. On the **Module Settings** tab:
|
||||
- Set **Module Name** to "VirtFusion Direct Provisioning"
|
||||
- Select your VirtFusion server
|
||||
- Set **Hypervisor Group ID**, **Package ID**, and **Default IPv4** count
|
||||
4. Save the product
|
||||
|
||||
### Step 4: Set Up Custom Fields
|
||||
|
||||
See [Custom Fields](#custom-fields) section below.
|
||||
|
||||
### Step 5: Activate Hooks
|
||||
|
||||
The hooks file (`hooks.php`) is automatically detected by WHMCS when the module is active. If you add the module files to an existing installation, you may need to re-save the product settings or clear the WHMCS template cache for hooks to take effect.
|
||||
|
||||
## Upgrading
|
||||
|
||||
1. Back up your existing `modules/servers/VirtFusionDirect/` directory
|
||||
2. Back up `config/ConfigOptionMapping.php` if you have a custom mapping
|
||||
3. Download and deploy the new version:
|
||||
|
||||
```bash
|
||||
cd /tmp
|
||||
git clone https://github.com/EZSCALE/virtfusion-whmcs-module.git
|
||||
rsync -ahP --delete /tmp/virtfusion-whmcs-module/modules/servers/VirtFusionDirect/ /path/to/whmcs/modules/servers/VirtFusionDirect/
|
||||
rm -rf /tmp/virtfusion-whmcs-module
|
||||
```
|
||||
|
||||
4. Restore your custom `config/ConfigOptionMapping.php` if applicable
|
||||
5. If you have theme-overridden templates, review them for any new template variables
|
||||
6. Clear the WHMCS template cache: **Configuration > System Settings > General Settings > clear template cache**
|
||||
|
||||
The module database table (`mod_virtfusion_direct`) is automatically migrated on first load.
|
||||
|
||||
## Configuration
|
||||
|
||||
### Server Setup
|
||||
|
||||
In WHMCS Admin under **Configuration > System Settings > Servers**:
|
||||
|
||||
| Field | Value |
|
||||
|---|---|
|
||||
| Hostname | Your VirtFusion panel domain (e.g., `cp.example.com`) |
|
||||
| Password | Your VirtFusion API token |
|
||||
| Type | VirtFusion Direct Provisioning |
|
||||
|
||||
**Important**: Do not include `https://` or `/api/v1` in the hostname. The module constructs the full URL automatically.
|
||||
|
||||
### Product Setup
|
||||
|
||||
Each WHMCS product using this module needs:
|
||||
1. Module set to "VirtFusion Direct Provisioning"
|
||||
2. A linked server (or the module will use any available VirtFusion server)
|
||||
3. The three configuration options set (Hypervisor Group ID, Package ID, Default IPv4)
|
||||
4. Custom fields created (see below)
|
||||
|
||||
### Custom Fields
|
||||
|
||||
You **must** create two custom fields on each product that uses this module:
|
||||
|
||||
| Field Name | Field Type | Show on Order Form | Admin Only | Required |
|
||||
|---|---|---|---|---|
|
||||
| Initial Operating System | Text Box | Yes | No | No |
|
||||
| Initial SSH Key | Text Box | Yes | No | No |
|
||||
|
||||
These fields are hidden text boxes that are dynamically replaced by dropdown selects via JavaScript hooks on the order form.
|
||||
|
||||
**Automated setup**: Run the SQL from [modify.sql](modify.sql) to auto-create these fields for all VirtFusion products:
|
||||
|
||||
```bash
|
||||
mysql -u whmcs_user -p whmcs_database < modules/servers/VirtFusionDirect/modify.sql
|
||||
```
|
||||
|
||||
### Module Configuration Options
|
||||
|
||||
Each product has three module-specific settings:
|
||||
|
||||
| Option | Name | Description | Default |
|
||||
|---|---|---|---|
|
||||
| Config Option 1 | Hypervisor Group ID | VirtFusion hypervisor group for server placement | 1 |
|
||||
| Config Option 2 | Package ID | VirtFusion package defining server resources | 1 |
|
||||
| Config Option 3 | Default IPv4 | Number of IPv4 addresses to assign (0-10) | 1 |
|
||||
| Config Option 4 | Self-Service Mode | Enable VirtFusion self-service billing (0=Disabled, 1=Hourly, 2=Resource Packs, 3=Both) | 0 |
|
||||
| Config Option 5 | Auto Top-Off Threshold | Credit balance below which auto top-off triggers during cron (0=disabled) | 0 |
|
||||
| Config Option 6 | Auto Top-Off Amount | Credit amount to add when auto top-off triggers | 100 |
|
||||
|
||||
You can find your Hypervisor Group IDs and Package IDs in the VirtFusion admin panel.
|
||||
|
||||
### Configurable Options (Dynamic Pricing)
|
||||
|
||||
To allow customers to select different resource levels with pricing tiers, create WHMCS Configurable Options groups with these option names:
|
||||
|
||||
| VirtFusion Parameter | Default Option Name | Description | Unit |
|
||||
|---|---|---|---|
|
||||
| `packageId` | Package | VirtFusion package ID | ID |
|
||||
| `hypervisorId` | Location | Hypervisor group for placement | ID |
|
||||
| `ipv4` | IPv4 | Number of IPv4 addresses | Count |
|
||||
| `storage` | Storage | Disk space | GB |
|
||||
| `memory` | Memory | RAM (values < 1024 auto-converted from GB) | MB |
|
||||
| `traffic` | Bandwidth | Monthly traffic allowance | GB |
|
||||
| `cpuCores` | CPU Cores | Number of CPU cores | Count |
|
||||
| `networkSpeedInbound` | Inbound Network Speed | Inbound speed | Mbps |
|
||||
| `networkSpeedOutbound` | Outbound Network Speed | Outbound speed | Mbps |
|
||||
| `networkProfile` | Network Type | VirtFusion network profile | ID |
|
||||
| `storageProfile` | Storage Type | VirtFusion storage profile | ID |
|
||||
|
||||
### Custom Option Name Mapping
|
||||
|
||||
If your configurable option names differ from the defaults above:
|
||||
|
||||
1. Copy `config/ConfigOptionMapping-example.php` to `config/ConfigOptionMapping.php`
|
||||
2. Edit the mapping array:
|
||||
|
||||
```php
|
||||
return [
|
||||
'memory' => 'RAM', // Your option name for memory
|
||||
'cpuCores' => 'vCPU Count', // Your option name for CPU
|
||||
'traffic' => 'Data Transfer', // Your option name for bandwidth
|
||||
// ... add only the options that differ from defaults
|
||||
];
|
||||
```
|
||||
|
||||
## Client Area Features
|
||||
|
||||
### Server Overview
|
||||
Displays real-time server information fetched from VirtFusion:
|
||||
- Server name and hostname
|
||||
- Memory, CPU cores, storage allocation
|
||||
- IPv4 and IPv6 addresses
|
||||
- Traffic usage vs. allocation
|
||||
- Server status badge (Active, Suspended, etc.)
|
||||
|
||||
### Power Management
|
||||
Four power control buttons:
|
||||
- **Start** - Boot the server
|
||||
- **Restart** - Graceful restart
|
||||
- **Shutdown** - Graceful ACPI shutdown
|
||||
- **Force Off** - Immediate power cut (use with caution)
|
||||
|
||||
### Network Management
|
||||
- View all IPv4 addresses and IPv6 subnets assigned to the server
|
||||
- Remove secondary IPv4 addresses (primary cannot be removed)
|
||||
|
||||
### VNC Console
|
||||
- Opens a browser-based VNC console to the server
|
||||
- Requires VirtFusion v6.1.0+ and the server must be running
|
||||
- Opens in a new browser window/tab
|
||||
|
||||
### Server Rebuild
|
||||
- Select from available OS templates (filtered by server package)
|
||||
- Includes a confirmation dialog warning about data loss
|
||||
- Triggers email notification on completion
|
||||
|
||||
### Control Panel SSO
|
||||
- One-click login to the VirtFusion panel
|
||||
- Opens in a new window (with fallback to same-window navigation)
|
||||
- Password reset option for direct VirtFusion panel access
|
||||
|
||||
### Billing Overview
|
||||
- Product name and group
|
||||
- Recurring amount and billing cycle
|
||||
- Registration and next due dates
|
||||
- Payment method
|
||||
|
||||
## Admin Area Features
|
||||
|
||||
### Admin Services Tab
|
||||
When viewing a service in WHMCS admin, the module adds:
|
||||
- **Server ID** - Editable field showing the VirtFusion server ID
|
||||
- **Server Info** - Button to load live data from VirtFusion API
|
||||
- **Server Object** - Full JSON response viewer
|
||||
- **Options** - Admin impersonation link
|
||||
|
||||
### Module Commands (Admin Buttons)
|
||||
- **Create** - Provision a new server
|
||||
- **Suspend** / **Unsuspend** - Manage server suspension
|
||||
- **Terminate** - Delete the server (with 5-minute grace period in VirtFusion)
|
||||
- **Change Package** - Update server to a different VirtFusion package
|
||||
- **Update Server Object** - Refresh cached data from VirtFusion
|
||||
- **Validate Server Config** - Dry run server creation to test configuration
|
||||
|
||||
## Theme Compatibility
|
||||
|
||||
This module is designed to work with **all WHMCS themes**:
|
||||
|
||||
| Theme | Status | Notes |
|
||||
|---|---|---|
|
||||
| Six (default) | Fully compatible | Bootstrap 3 |
|
||||
| Twenty-One | Fully compatible | Bootstrap 4 |
|
||||
| Lagom (ModulesGarden) | Fully compatible | Bootstrap 5 |
|
||||
| Custom themes | Compatible | Uses dual CSS classes |
|
||||
|
||||
### How Theme Compatibility Works
|
||||
|
||||
The module uses dual CSS class names that work across Bootstrap versions:
|
||||
- `panel card` - Works in BS3 (panel) and BS4/BS5 (card)
|
||||
- `panel-heading card-header` - Works in BS3 and BS4/BS5
|
||||
- `panel-body card-body` - Works in BS3 and BS4/BS5
|
||||
- `panel-title card-title` - Works in BS3 and BS4/BS5
|
||||
|
||||
The order form hooks use vanilla JavaScript (no jQuery dependency) for maximum compatibility.
|
||||
|
||||
### Theme Override
|
||||
|
||||
To customize templates for a specific theme:
|
||||
|
||||
```
|
||||
/templates/yourthemename/modules/servers/VirtFusionDirect/
|
||||
overview.tpl # Client area template
|
||||
error.tpl # Error template
|
||||
```
|
||||
|
||||
WHMCS automatically loads theme-specific templates when they exist. Copy the originals from `modules/servers/VirtFusionDirect/templates/` as a starting point.
|
||||
|
||||
## API Endpoints Used
|
||||
|
||||
### Core Provisioning
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `GET` | `/connect` | Connection testing |
|
||||
| `GET/POST` | `/users` | User lookup and creation |
|
||||
| `GET` | `/users/{id}/byExtRelation` | Find VirtFusion user by WHMCS ID |
|
||||
| `POST` | `/servers` | Server creation |
|
||||
| `POST` | `/servers?dryRun=true` | Dry run validation |
|
||||
| `POST` | `/servers/{id}/build` | OS installation / rebuild |
|
||||
| `GET` | `/servers/{id}` | Server details (also used by UsageUpdate) |
|
||||
| `DELETE` | `/servers/{id}` | Server termination |
|
||||
| `POST` | `/servers/{id}/suspend` | Server suspension |
|
||||
| `POST` | `/servers/{id}/unsuspend` | Server unsuspension |
|
||||
| `PUT` | `/servers/{id}/package/{pkgId}` | Package changes |
|
||||
|
||||
### Client Management
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `POST` | `/servers/{id}/power/{action}` | Power management |
|
||||
| `PATCH` | `/servers/{id}/name` | Server renaming |
|
||||
| `POST` | `/users/{id}/serverAuthenticationTokens/{serverId}` | SSO token |
|
||||
| `POST` | `/users/{id}/byExtRelation/resetPassword` | Password reset |
|
||||
| `GET` | `/media/templates/fromServerPackageSpec/{id}` | OS templates |
|
||||
| `GET` | `/ssh_keys/user/{id}` | SSH key listing |
|
||||
|
||||
### Network
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `DELETE` | `/servers/{id}/ipv4` | Remove IPv4 address |
|
||||
|
||||
### SSH Keys
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `POST` | `/ssh_keys` | Create SSH key for a user (checkout key paste) |
|
||||
|
||||
### Self-Service Billing
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `GET` | `/selfService/usage/byUserExtRelationId/{id}` | Usage data by WHMCS client ID |
|
||||
| `GET` | `/selfService/report/byUserExtRelationId/{id}` | Billing report by WHMCS client ID |
|
||||
| `POST` | `/selfService/credit/byUserExtRelationId/{id}` | Add credit by WHMCS client ID |
|
||||
| `GET` | `/selfService/currencies` | Available self-service currencies |
|
||||
|
||||
### Advanced
|
||||
|
||||
| Method | Endpoint | Purpose |
|
||||
|---|---|---|
|
||||
| `GET` | `/servers/{id}/vnc` | VNC console (v6.1.0+) |
|
||||
| `PUT` | `/servers/{id}/modify/memory` | Modify memory (v6.2.0+) |
|
||||
| `PUT` | `/servers/{id}/modify/cpuCores` | Modify CPU cores (v6.2.0+) |
|
||||
| `PUT` | `/servers/{id}/modify/traffic` | Modify traffic (v6.0.0+) |
|
||||
| `POST/DELETE` | `/servers/{id}/backup/plan` | Backup plan management (v4.3.0+) |
|
||||
|
||||
## Usage Update (Cron)
|
||||
|
||||
The module implements the `UsageUpdate` function that is called by the WHMCS daily cron. It automatically syncs:
|
||||
|
||||
- **Disk usage** (used and limit) from VirtFusion to WHMCS `tblhosting`
|
||||
- **Bandwidth usage** (used and limit) from VirtFusion to WHMCS `tblhosting`
|
||||
|
||||
This data appears in the WHMCS client area and admin product details.
|
||||
|
||||
**Requirements**: The WHMCS cron must be running (`php -q /path/to/whmcs/crons/cron.php`). No additional configuration is needed - the module registers itself automatically.
|
||||
|
||||
**How it works**:
|
||||
1. WHMCS calls `VirtFusionDirect_UsageUpdate()` once per configured server
|
||||
2. The module queries all Active services assigned to that server
|
||||
3. For each service, it fetches server data from VirtFusion API
|
||||
4. Disk and bandwidth usage/limits are written to `tblhosting`
|
||||
|
||||
**Data format conversion**:
|
||||
- VirtFusion traffic: bytes -> WHMCS expects: MB
|
||||
- VirtFusion storage: bytes -> WHMCS expects: MB
|
||||
- VirtFusion storage limit: GB -> WHMCS expects: MB
|
||||
- VirtFusion traffic limit: GB -> WHMCS expects: MB (0 = unlimited)
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### Connection Test Fails
|
||||
|
||||
| Symptom | Cause | Solution |
|
||||
|---|---|---|
|
||||
| "Authentication failed" | Invalid or expired API token | Generate a new token in VirtFusion |
|
||||
| "Connection failed" | Hostname unreachable or SSL issue | Verify hostname, check SSL cert validity |
|
||||
| "Unexpected response" | API version mismatch or server issue | Check VirtFusion is running, verify API version |
|
||||
|
||||
### Server Creation Fails
|
||||
|
||||
| Symptom | Cause | Solution |
|
||||
|---|---|---|
|
||||
| "Service already exists" | Duplicate provisioning attempt | Run termination first, then create |
|
||||
| "No Control server found" | No VirtFusion server in WHMCS | Add server in System Settings > Servers |
|
||||
| "Unable to create user" | API permission issue | Check token has user create permission |
|
||||
| "Server creation failed" | Invalid config options | Use "Validate Server Config" button to diagnose |
|
||||
| HTTP 423 response | Server is locked | Wait and retry, or check VirtFusion for lock reason |
|
||||
|
||||
### OS Templates Not Showing on Order Form
|
||||
|
||||
1. Verify the **Package ID** (Config Option 2) is correct
|
||||
2. Check that the package has OS templates assigned in VirtFusion
|
||||
3. Ensure the **"Initial Operating System"** custom field exists (exact name match required)
|
||||
4. Check that hooks are loading: re-save product settings to trigger hook detection
|
||||
5. Inspect browser console for JavaScript errors
|
||||
|
||||
### Client Area Shows Error Template
|
||||
|
||||
1. Ensure a VirtFusion server is configured and linked to the product
|
||||
2. Check the service status is Active or Suspended (not Pending/Terminated)
|
||||
3. Review **Utilities > Logs > Module Log** for API errors
|
||||
4. Verify the `mod_virtfusion_direct` table has an entry for the service
|
||||
|
||||
### SSO / Control Panel Login Fails
|
||||
|
||||
1. VirtFusion panel must be accessible from the client's browser
|
||||
2. Verify the VirtFusion user exists (check by external relation ID in VirtFusion admin)
|
||||
3. Ensure authentication token generation is enabled on the API token
|
||||
4. Check for popup blockers if the new window doesn't open
|
||||
|
||||
### VNC Console Not Working
|
||||
|
||||
1. Requires VirtFusion v6.1.0 or higher
|
||||
2. The server must be powered on and running
|
||||
3. Check that VNC is enabled for the hypervisor in VirtFusion
|
||||
4. Popup blockers may prevent the console window from opening
|
||||
|
||||
### UsageUpdate Not Syncing
|
||||
|
||||
1. Verify the WHMCS cron is running: `php -q /path/to/whmcs/crons/cron.php`
|
||||
2. Check **Utilities > Logs > Module Log** for UsageUpdate errors
|
||||
3. Ensure services are in "Active" status (other statuses are skipped)
|
||||
4. The cron runs daily; wait for the next cycle after initial setup
|
||||
|
||||
## Known Issues
|
||||
|
||||
1. **VNC Console** - Requires VirtFusion v6.1.0+. Earlier versions do not expose a VNC API endpoint. The module gracefully handles this by showing an error message.
|
||||
|
||||
2. **Resource Modification** - Memory and CPU modification requires VirtFusion v6.2.0+. Traffic modification requires v6.0.0+. Backup management requires v4.3.0+.
|
||||
|
||||
3. **IPv6 Display** - IPv6 subnet display depends on the VirtFusion installation having IPv6 pools configured. If no IPv6 is assigned, the network panel shows "No IPv6 subnets".
|
||||
|
||||
4. **Order Form Custom Fields** - The custom fields ("Initial Operating System" and "Initial SSH Key") must be named exactly as specified. The module matches by field name with spaces removed and converted to lowercase.
|
||||
|
||||
5. **Hooks File Detection** - WHMCS detects the `hooks.php` file when the module is first activated. If you add the module files to an already-active installation, you may need to deactivate and reactivate the module, or re-save the product settings.
|
||||
|
||||
6. **Bootstrap 3 Themes** - While the module supports BS3 themes, some visual differences may exist (e.g., `d-flex` not available in BS3). The module uses `display: flex` in CSS as a fallback.
|
||||
|
||||
7. **Concurrent API Calls** - The module makes individual API calls for each feature panel on the client area page. If the VirtFusion API is slow, the page may take longer to fully load. All panels load asynchronously to minimize perceived delay.
|
||||
|
||||
8. **Primary IPv4 Protection** - The first IPv4 address cannot be removed through the client area interface. This is by design to prevent users from accidentally removing their primary IP address.
|
||||
|
||||
9. **Self-Signed SSL Certificates** - SSL verification is enforced by default. VirtFusion panels using self-signed certificates will cause connection failures. Use a valid SSL certificate (e.g., Let's Encrypt) on your VirtFusion panel.
|
||||
|
||||
## Security
|
||||
|
||||
### Architecture
|
||||
- All client API endpoints validate service ownership before processing
|
||||
- Admin endpoints require WHMCS admin authentication
|
||||
- Input sanitization on all user-supplied parameters (type casting, regex filtering, `filter_var`)
|
||||
- Proper HTTP status codes (401, 403, 400, 500) for error responses
|
||||
- XSS prevention via `htmlspecialchars()`, `encodeURIComponent()`, and jQuery `.text()`
|
||||
|
||||
### Best Practices
|
||||
- **API Tokens**: Store only in the WHMCS server password field (encrypted at rest by WHMCS)
|
||||
- **SSL Verification**: Enabled by default. Never disable in production.
|
||||
- **File Access**: All PHP files include direct access prevention checks
|
||||
- **Module Updates**: Keep updated for security patches
|
||||
- **Permissions**: Use the minimum required API token permissions
|
||||
|
||||
### Reporting Vulnerabilities
|
||||
If you discover a security vulnerability, please report it responsibly by emailing the maintainers rather than opening a public issue. See [SECURITY.md](SECURITY.md) for details.
|
||||
|
||||
## File Structure
|
||||
|
||||
```
|
||||
modules/servers/VirtFusionDirect/
|
||||
VirtFusionDirect.php # WHMCS module entry point (MetaData, ConfigOptions, all module functions)
|
||||
client.php # Client-facing AJAX API (authenticated, ownership-validated)
|
||||
admin.php # Admin-facing AJAX API (admin authentication required)
|
||||
hooks.php # WHMCS hooks (order form OS/SSH dropdowns, checkout validation)
|
||||
modify.sql # SQL for creating custom fields
|
||||
lib/
|
||||
Module.php # Base class: API communication, power, network, VNC, rebuild
|
||||
ModuleFunctions.php # Provisioning: create, suspend, unsuspend, terminate, change package
|
||||
ConfigureService.php # Order configuration: OS templates, SSH keys, server build init
|
||||
Database.php # Database operations: custom table, WHMCS table queries
|
||||
Curl.php # HTTP client: GET, POST, PUT, PATCH, DELETE with SSL verification
|
||||
ServerResource.php # Data transformer: VirtFusion API response -> display format
|
||||
AdminHTML.php # Admin interface: HTML generation for admin services tab
|
||||
Log.php # Logging: WHMCS module log integration
|
||||
templates/
|
||||
overview.tpl # Client area Smarty template (all management panels)
|
||||
error.tpl # Error display template
|
||||
css/module.css # Module styles (responsive, BS3/4/5 compatible)
|
||||
js/module.js # Client JavaScript (all AJAX interactions)
|
||||
js/keygen.js # SSH Ed25519 key generator (Web Crypto API)
|
||||
config/
|
||||
ConfigOptionMapping-example.php # Example custom option name mapping
|
||||
```
|
||||
|
||||
## Contributing
|
||||
|
||||
Contributions are welcome! Please:
|
||||
|
||||
1. Fork the repository
|
||||
2. Create a feature branch (`git checkout -b feature/your-feature`)
|
||||
3. Commit your changes with clear messages
|
||||
4. Push to your fork and open a Pull Request
|
||||
|
||||
For bug reports, please include:
|
||||
- WHMCS version
|
||||
- VirtFusion version
|
||||
- PHP version
|
||||
- Steps to reproduce
|
||||
- Module Log output (Utilities > Logs > Module Log)
|
||||
|
||||
## License
|
||||
|
||||
This project is licensed under the GNU General Public License v3.0 - see the [LICENSE.md](LICENSE.md) file for details.
|
||||
|
||||
Copyright (c) EZSCALE
|
||||
|
||||
@@ -2,11 +2,11 @@
|
||||
|
||||
## Supported Versions
|
||||
|
||||
The support version of this module with VirtFusion
|
||||
Supported VirtFusion versions:
|
||||
|
||||
| Version | Supported |
|
||||
|---------|--------------------|
|
||||
| > 1.7.3 | :white_check_mark: |
|
||||
|----------|--------------------|
|
||||
| >= 1.7.3 | :white_check_mark: |
|
||||
| < 1.7.3 | :x: |
|
||||
|
||||
## Reporting a Vulnerability
|
||||
|
||||
49
modify.sql
Normal file
49
modify.sql
Normal file
@@ -0,0 +1,49 @@
|
||||
-- Insert records for Initial Operating System if they don't already exist
|
||||
INSERT INTO tblcustomfields
|
||||
(type, relid, fieldname, fieldtype, description, fieldoptions, regexpr, adminonly, required, showorder, showinvoice,
|
||||
sortorder, created_at, updated_at)
|
||||
SELECT 'product',
|
||||
id,
|
||||
'Initial Operating System',
|
||||
'text',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'on',
|
||||
'',
|
||||
0,
|
||||
UTC_TIMESTAMP(),
|
||||
UTC_TIMESTAMP()
|
||||
FROM tblproducts
|
||||
WHERE servertype = 'VirtFusionDirect'
|
||||
AND NOT EXISTS (SELECT 1
|
||||
FROM tblcustomfields
|
||||
WHERE fieldname = 'Initial Operating System'
|
||||
AND relid = tblproducts.id);
|
||||
|
||||
-- Insert records for Initial SSH Key if they don't already exist
|
||||
INSERT INTO tblcustomfields
|
||||
(type, relid, fieldname, fieldtype, description, fieldoptions, regexpr, adminonly, required, showorder, showinvoice,
|
||||
sortorder, created_at, updated_at)
|
||||
SELECT 'product',
|
||||
id,
|
||||
'Initial SSH Key',
|
||||
'text',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'on',
|
||||
'',
|
||||
0,
|
||||
UTC_TIMESTAMP(),
|
||||
UTC_TIMESTAMP()
|
||||
FROM tblproducts
|
||||
WHERE servertype = 'VirtFusionDirect'
|
||||
AND NOT EXISTS (SELECT 1
|
||||
FROM tblcustomfields
|
||||
WHERE fieldname = 'Initial SSH Key'
|
||||
AND relid = tblproducts.id);
|
||||
@@ -5,6 +5,8 @@ if (!defined("WHMCS")) {
|
||||
}
|
||||
|
||||
use WHMCS\Module\Server\VirtFusionDirect\ModuleFunctions;
|
||||
use WHMCS\Module\Server\VirtFusionDirect\Module;
|
||||
use WHMCS\Module\Server\VirtFusionDirect\Database;
|
||||
|
||||
function VirtFusionDirect_MetaData()
|
||||
{
|
||||
@@ -12,7 +14,7 @@ function VirtFusionDirect_MetaData()
|
||||
'DisplayName' => 'VirtFusion Direct Provisioning',
|
||||
'APIVersion' => '1.1',
|
||||
'RequiresServer' => true,
|
||||
'ServiceSingleSignOnLabel' => false,
|
||||
'ServiceSingleSignOnLabel' => 'Login to VirtFusion Panel',
|
||||
'AdminSingleSignOnLabel' => false,
|
||||
];
|
||||
}
|
||||
@@ -24,39 +26,109 @@ function VirtFusionDirect_ConfigOptions()
|
||||
"FriendlyName" => "Hypervisor Group ID",
|
||||
"Type" => "text",
|
||||
"Size" => "20",
|
||||
"Description" => "The default hypervisor group ID",
|
||||
"Description" => "The default hypervisor group ID for server placement.",
|
||||
"Default" => "1",
|
||||
],
|
||||
"packageID" => [
|
||||
"FriendlyName" => "Package ID",
|
||||
"Type" => "text",
|
||||
"Size" => "20",
|
||||
"Description" => "The package ID",
|
||||
"Description" => "The VirtFusion package ID that defines server resources.",
|
||||
"Default" => "1",
|
||||
],
|
||||
"defaultIPv4" => [
|
||||
"FriendlyName" => "Default IPv4",
|
||||
"Type" => "dropdown",
|
||||
"Options" => "0,1,2,3,4,5,6,7,8,9,10",
|
||||
"Description" => "The default amount of IPv4 addresses to assign to the server.",
|
||||
"Description" => "The default number of IPv4 addresses to assign to each server.",
|
||||
"Default" => "1",
|
||||
],
|
||||
"selfServiceMode" => [
|
||||
"FriendlyName" => "Self-Service Mode",
|
||||
"Type" => "dropdown",
|
||||
"Options" => "0|Disabled,1|Hourly,2|Resource Packs,3|Both",
|
||||
"Description" => "Enable VirtFusion self-service billing for users created by this product.",
|
||||
"Default" => "0",
|
||||
],
|
||||
"autoTopOffThreshold" => [
|
||||
"FriendlyName" => "Auto Top-Off Threshold",
|
||||
"Type" => "text",
|
||||
"Size" => "10",
|
||||
"Description" => "Credit balance below which auto top-off triggers during cron. 0 = disabled.",
|
||||
"Default" => "0",
|
||||
],
|
||||
"autoTopOffAmount" => [
|
||||
"FriendlyName" => "Auto Top-Off Amount",
|
||||
"Type" => "text",
|
||||
"Size" => "10",
|
||||
"Description" => "Credit amount to add when auto top-off triggers.",
|
||||
"Default" => "100",
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
function VirtFusionDirect_TestConnection(array $params)
|
||||
{
|
||||
try {
|
||||
$hostname = trim($params['serverhostname'] ?? '');
|
||||
$password = $params['serverpassword'] ?? '';
|
||||
|
||||
if (empty($hostname) || empty($password)) {
|
||||
return ['success' => false, 'error' => 'Server hostname and password are required. Please verify the server configuration.'];
|
||||
}
|
||||
|
||||
$url = 'https://' . $hostname . '/api/v1';
|
||||
$module = new Module();
|
||||
$request = $module->initCurl($password);
|
||||
$data = $request->get($url . '/connect');
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
|
||||
if ($httpCode == 200) {
|
||||
return ['success' => true, 'error' => ''];
|
||||
}
|
||||
|
||||
if ($httpCode == 401) {
|
||||
return ['success' => false, 'error' => 'Authentication failed. Please verify your API token is correct and has not expired.'];
|
||||
}
|
||||
|
||||
if ($httpCode == 0) {
|
||||
$curlError = $request->getRequestInfo('curl_error');
|
||||
return ['success' => false, 'error' => 'Connection failed: ' . ($curlError ?: 'Unable to reach the VirtFusion server. Verify the hostname and that SSL certificates are valid.')];
|
||||
}
|
||||
|
||||
return ['success' => false, 'error' => 'Unexpected response from VirtFusion API (HTTP ' . $httpCode . '). Please check the server configuration.'];
|
||||
} catch (\Throwable $e) {
|
||||
return ['success' => false, 'error' => 'Connection test failed: ' . $e->getMessage()];
|
||||
}
|
||||
}
|
||||
|
||||
function VirtFusionDirect_AdminCustomButtonArray()
|
||||
{
|
||||
$buttonarray = array(
|
||||
return [
|
||||
"Update Server Object" => "updateServerObject",
|
||||
);
|
||||
return $buttonarray;
|
||||
"Validate Server Config" => "validateServerConfig",
|
||||
];
|
||||
}
|
||||
|
||||
function VirtFusionDirect_ServiceSingleSignOn(array $params)
|
||||
{
|
||||
try {
|
||||
$module = new Module();
|
||||
$token = $module->fetchLoginTokens($params['serviceid']);
|
||||
|
||||
if ($token) {
|
||||
return ['success' => true, 'redirectTo' => $token];
|
||||
}
|
||||
|
||||
return ['success' => false, 'errorMsg' => 'Unable to generate a login token. The server may not be active or the VirtFusion API may be unreachable.'];
|
||||
} catch (\Exception $e) {
|
||||
return ['success' => false, 'errorMsg' => $e->getMessage()];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*
|
||||
* Service functions
|
||||
*
|
||||
*/
|
||||
function VirtFusionDirect_CreateAccount(array $params)
|
||||
{
|
||||
@@ -86,7 +158,6 @@ function VirtFusionDirect_updateServerObject(array $params)
|
||||
/**
|
||||
* Allows changing of the package of a server
|
||||
*
|
||||
* @author https://github.com/BlinkohHost/virtfusion-whmcs-module
|
||||
* @param array $params
|
||||
* @return string
|
||||
*/
|
||||
@@ -109,3 +180,122 @@ function VirtFusionDirect_ClientArea(array $params)
|
||||
{
|
||||
return (new ModuleFunctions())->clientArea($params);
|
||||
}
|
||||
|
||||
/**
|
||||
* Validates server configuration via dry run without creating the server.
|
||||
*
|
||||
* @param array $params
|
||||
* @return string 'success' or error message
|
||||
*/
|
||||
function VirtFusionDirect_validateServerConfig(array $params)
|
||||
{
|
||||
return (new ModuleFunctions())->validateServerConfig($params);
|
||||
}
|
||||
|
||||
/**
|
||||
* Usage Update - called by WHMCS daily cron to sync bandwidth and disk usage.
|
||||
*
|
||||
* Updates tblhosting with disk and bandwidth usage data from VirtFusion.
|
||||
* Fields updated: diskused, disklimit, bwused, bwlimit, lastupdate
|
||||
*
|
||||
* @param array $params Server access credentials
|
||||
* @return string 'success' or error message
|
||||
*/
|
||||
function VirtFusionDirect_UsageUpdate(array $params)
|
||||
{
|
||||
try {
|
||||
$module = new Module();
|
||||
$cp = $module->getCP($params['serverid']);
|
||||
|
||||
if (!$cp) {
|
||||
return 'No control server found for usage update.';
|
||||
}
|
||||
|
||||
$services = \WHMCS\Database\Capsule::table('tblhosting')
|
||||
->where('server', $params['serverid'])
|
||||
->where('domainstatus', 'Active')
|
||||
->get();
|
||||
|
||||
foreach ($services as $service) {
|
||||
try {
|
||||
$systemService = Database::getSystemService($service->id);
|
||||
if (!$systemService) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$request = $module->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/servers/' . (int) $systemService->server_id);
|
||||
|
||||
if ($request->getRequestInfo('http_code') != 200) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$serverData = json_decode($data, true);
|
||||
if (!isset($serverData['data'])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$server = $serverData['data'];
|
||||
$update = [];
|
||||
|
||||
// Disk usage (WHMCS expects MB)
|
||||
if (isset($server['usage']['storage']['used'])) {
|
||||
$update['diskused'] = round($server['usage']['storage']['used'] / 1048576);
|
||||
}
|
||||
if (isset($server['settings']['resources']['storage'])) {
|
||||
$update['disklimit'] = (int) $server['settings']['resources']['storage'] * 1024;
|
||||
}
|
||||
|
||||
// Bandwidth usage (WHMCS expects MB)
|
||||
if (isset($server['usage']['traffic']['used'])) {
|
||||
$update['bwused'] = round($server['usage']['traffic']['used'] / 1048576);
|
||||
}
|
||||
if (isset($server['settings']['resources']['traffic'])) {
|
||||
$trafficGB = (int) $server['settings']['resources']['traffic'];
|
||||
$update['bwlimit'] = $trafficGB > 0 ? $trafficGB * 1024 : 0;
|
||||
}
|
||||
|
||||
if (!empty($update)) {
|
||||
$update['lastupdate'] = date('Y-m-d H:i:s');
|
||||
\WHMCS\Database\Capsule::table('tblhosting')
|
||||
->where('id', $service->id)
|
||||
->update($update);
|
||||
}
|
||||
|
||||
// Self-service auto top-off
|
||||
$product = \WHMCS\Database\Capsule::table('tblproducts')
|
||||
->where('id', $service->packageid)
|
||||
->first();
|
||||
|
||||
if ($product) {
|
||||
$threshold = (float) ($product->configoption5 ?? 0);
|
||||
$topOffAmount = (float) ($product->configoption6 ?? 0);
|
||||
|
||||
if ($threshold > 0 && $topOffAmount > 0) {
|
||||
$usageData = $module->getSelfServiceUsage($service->id);
|
||||
if ($usageData) {
|
||||
$usageInner = $usageData['data'] ?? $usageData;
|
||||
$credit = $usageInner['credit'] ?? $usageInner['balance'] ?? null;
|
||||
if ($credit !== null && (float) $credit < $threshold) {
|
||||
$module->addSelfServiceCredit($service->id, $topOffAmount, 'Auto top-off');
|
||||
\WHMCS\Module\Server\VirtFusionDirect\Log::insert(
|
||||
'UsageUpdate:autoTopOff',
|
||||
['serviceId' => $service->id, 'credit' => $credit, 'threshold' => $threshold],
|
||||
['amount' => $topOffAmount]
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
// Log but continue processing other services
|
||||
\WHMCS\Module\Server\VirtFusionDirect\Log::insert('UsageUpdate:service:' . $service->id, [], $e->getMessage());
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
return 'success';
|
||||
} catch (\Exception $e) {
|
||||
return 'Usage update failed: ' . $e->getMessage();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -26,17 +26,17 @@ switch ($vf->validateAction(true)) {
|
||||
$whmcsService = Database::getWhmcsService((int)$_GET['serviceID']);
|
||||
|
||||
if (!$whmcsService) {
|
||||
$vf->output(['success' => false, 'errors' => 'Service not found.'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'Service not found.'], true, true, 404);
|
||||
}
|
||||
|
||||
if ($whmcsService->domainstatus == 'Pending' || $whmcsService->domainstatus == 'Terminated' || $whmcsService->domainstatus == 'Cancelled' || $whmcsService->domainstatus == 'Fraud') {
|
||||
$vf->output(['success' => false, 'errors' => 'Server is not Active, Suspended or Completed. Not fetching remote data.'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'Server is not Active, Suspended or Completed. Not fetching remote data.'], true, true, 400);
|
||||
}
|
||||
|
||||
$data = $vf->fetchServerData((int)$_GET['serviceID']);
|
||||
|
||||
if (!$data) {
|
||||
$vf->output(['success' => false, 'errors' => 'No data returned from VirtFusion.'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'No data returned from VirtFusion.'], true, true, 502);
|
||||
|
||||
}
|
||||
|
||||
@@ -58,12 +58,21 @@ switch ($vf->validateAction(true)) {
|
||||
$service = Database::getSystemService((int)$_GET['serviceID']);
|
||||
|
||||
if (!$service) {
|
||||
$vf->output(['success' => false, 'errors' => 'Service not found'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'Service not found'], true, true, 404);
|
||||
}
|
||||
|
||||
$whmcsService = Database::getWhmcsService((int)$_GET['serviceID']);
|
||||
|
||||
if (!$whmcsService) {
|
||||
$vf->output(['success' => false, 'errors' => 'WHMCS service not found'], true, true, 404);
|
||||
}
|
||||
|
||||
$cp = $vf->getCP($whmcsService->server);
|
||||
|
||||
if (!$cp) {
|
||||
$vf->output(['success' => false, 'errors' => 'Control server not found'], true, true, 500);
|
||||
}
|
||||
|
||||
$request = $vf->initCurl($cp['token']);
|
||||
|
||||
$data = $request->get($cp['url'] . '/users/' . $whmcsService->userid . '/byExtRelation');
|
||||
@@ -72,7 +81,7 @@ switch ($vf->validateAction(true)) {
|
||||
$vf->output(['success' => true, 'url' => $cp['base_url'], 'user' => json_decode($data, true)['data']], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Received HTTP code ' . $request->getRequestInfo('http_code')], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'Received HTTP code ' . $request->getRequestInfo('http_code')], true, true, 502);
|
||||
|
||||
}
|
||||
break;
|
||||
@@ -80,6 +89,5 @@ switch ($vf->validateAction(true)) {
|
||||
default:
|
||||
/** No valid action was specified **/
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'invalid action'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'invalid action'], true, true, 400);
|
||||
}
|
||||
|
||||
|
||||
@@ -9,103 +9,295 @@ $vf = new Module();
|
||||
|
||||
$vf->isAuthenticated();
|
||||
|
||||
switch ($vf->validateAction(true)) {
|
||||
$action = $vf->validateAction(true);
|
||||
|
||||
switch ($action) {
|
||||
|
||||
/**
|
||||
*
|
||||
* Reset Password.
|
||||
*
|
||||
*/
|
||||
case 'resetPassword':
|
||||
|
||||
if ($vf->validateServiceID(true)) {
|
||||
|
||||
$client = $vf->validateUserOwnsService((int)$_GET['serviceID']);
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
$client = $vf->validateUserOwnsService($serviceID);
|
||||
|
||||
if (!$client) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$data = $vf->resetUserPassword((int)$_GET['serviceID'], $client);
|
||||
$data = $vf->resetUserPassword($serviceID, $client);
|
||||
|
||||
if ($data) {
|
||||
$vf->output(['success' => true, 'data' => $data->data], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'error'], true, true, 200);
|
||||
|
||||
}
|
||||
$vf->output(['success' => false, 'errors' => 'Password reset failed'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
*
|
||||
* Get server information.
|
||||
*
|
||||
*/
|
||||
case 'serverData':
|
||||
|
||||
if ($vf->validateServiceID(true)) {
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService((int)$_GET['serviceID'])) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 200);
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$data = $vf->fetchServerData((int)$_GET['serviceID']);
|
||||
$data = $vf->fetchServerData($serviceID);
|
||||
|
||||
if ($data) {
|
||||
|
||||
(new Module())->updateWhmcsServiceParamsOnServerObject((int)$_GET['serviceID'], $data);
|
||||
|
||||
(new Module())->updateWhmcsServiceParamsOnServerObject($serviceID, $data);
|
||||
$vf->output(['success' => true, 'data' => (new ServerResource())->process($data)], true, true, 200);
|
||||
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'error'], true, true, 200);
|
||||
|
||||
}
|
||||
$vf->output(['success' => false, 'errors' => 'Unable to retrieve server data'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
*
|
||||
* Login as server owner.
|
||||
*
|
||||
*/
|
||||
case 'loginAsServerOwner':
|
||||
|
||||
if ($vf->validateServiceID(true)) {
|
||||
/**
|
||||
* A client can't log in as any user. Ownership should be validated.
|
||||
*/
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService((int)$_GET['serviceID'])) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 200);
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$token = $vf->fetchLoginTokens((int)$_GET['serviceID']);
|
||||
$token = $vf->fetchLoginTokens($serviceID);
|
||||
|
||||
if ($token) {
|
||||
|
||||
/**
|
||||
* A valid token/url was received.
|
||||
*/
|
||||
$vf->output(['success' => true, 'token_url' => $token], true, true, 200);
|
||||
}
|
||||
|
||||
/**
|
||||
* Failed to get the token from the control panel or the service ID doesn't exist.
|
||||
*/
|
||||
$vf->output(['success' => false, 'errors' => 'token request error'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'Unable to generate login token'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Power management actions: boot, shutdown, restart, poweroff
|
||||
*/
|
||||
case 'powerAction':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$powerAction = isset($_GET['powerAction']) ? preg_replace('/[^a-zA-Z]/', '', $_GET['powerAction']) : '';
|
||||
$allowedActions = ['boot', 'shutdown', 'restart', 'poweroff'];
|
||||
|
||||
if (!in_array($powerAction, $allowedActions, true)) {
|
||||
$vf->output(['success' => false, 'errors' => 'Invalid power action'], true, true, 400);
|
||||
}
|
||||
|
||||
$result = $vf->serverPowerAction($serviceID, $powerAction);
|
||||
|
||||
if ($result) {
|
||||
$vf->output(['success' => true, 'data' => ['action' => $powerAction, 'message' => 'Power action queued successfully']], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Power action failed. The server may be locked or unavailable.'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Rebuild/reinstall server with new OS.
|
||||
*/
|
||||
case 'rebuild':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$osId = isset($_GET['osId']) ? (int) $_GET['osId'] : 0;
|
||||
$hostname = isset($_GET['hostname']) ? preg_replace('/[^a-zA-Z0-9.\-]/', '', $_GET['hostname']) : null;
|
||||
|
||||
if ($osId <= 0) {
|
||||
$vf->output(['success' => false, 'errors' => 'Invalid operating system ID'], true, true, 400);
|
||||
}
|
||||
|
||||
$result = $vf->rebuildServer($serviceID, $osId, $hostname);
|
||||
|
||||
if ($result) {
|
||||
$vf->output(['success' => true, 'data' => ['message' => 'Server rebuild initiated successfully']], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Server rebuild failed. The server may be locked or unavailable.'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Rename server.
|
||||
*/
|
||||
case 'rename':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$newName = isset($_GET['name']) ? trim($_GET['name']) : '';
|
||||
$newName = htmlspecialchars($newName, ENT_QUOTES, 'UTF-8');
|
||||
|
||||
if (empty($newName) || strlen($newName) > 255) {
|
||||
$vf->output(['success' => false, 'errors' => 'Invalid server name'], true, true, 400);
|
||||
}
|
||||
|
||||
$result = $vf->renameServer($serviceID, $newName);
|
||||
|
||||
if ($result) {
|
||||
$vf->output(['success' => true, 'data' => ['message' => 'Server renamed successfully']], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Server rename failed'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Get available OS templates for rebuild.
|
||||
*/
|
||||
case 'osTemplates':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$templates = $vf->fetchOsTemplates($serviceID);
|
||||
|
||||
if ($templates !== false) {
|
||||
$vf->output(['success' => true, 'data' => $templates], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Unable to fetch OS templates'], true, true, 500);
|
||||
break;
|
||||
|
||||
// =================================================================
|
||||
// IP Address Management
|
||||
// =================================================================
|
||||
|
||||
/**
|
||||
* Remove an IPv4 address.
|
||||
*/
|
||||
case 'removeIPv4':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$ipAddress = isset($_GET['ip']) ? trim($_GET['ip']) : '';
|
||||
if (!filter_var($ipAddress, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)) {
|
||||
$vf->output(['success' => false, 'errors' => 'Invalid IPv4 address'], true, true, 400);
|
||||
}
|
||||
|
||||
$result = $vf->removeIPv4($serviceID, $ipAddress);
|
||||
|
||||
if ($result) {
|
||||
$vf->output(['success' => true, 'data' => ['message' => 'IPv4 address removed successfully']], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Failed to remove IPv4 address'], true, true, 500);
|
||||
break;
|
||||
|
||||
// =================================================================
|
||||
// VNC Console
|
||||
// =================================================================
|
||||
|
||||
/**
|
||||
* Get VNC console URL.
|
||||
*/
|
||||
case 'vnc':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$result = $vf->getVncConsole($serviceID);
|
||||
|
||||
if ($result !== false) {
|
||||
$vf->output(['success' => true, 'data' => $result], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'VNC console unavailable. The server may be powered off or VNC is not supported.'], true, true, 500);
|
||||
break;
|
||||
|
||||
// =================================================================
|
||||
// Self Service — Credit & Usage
|
||||
// =================================================================
|
||||
|
||||
/**
|
||||
* Get self-service usage data.
|
||||
*/
|
||||
case 'selfServiceUsage':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$result = $vf->getSelfServiceUsage($serviceID);
|
||||
|
||||
if ($result !== false) {
|
||||
$vf->output(['success' => true, 'data' => $result], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Unable to retrieve self-service usage data'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Get self-service billing report.
|
||||
*/
|
||||
case 'selfServiceReport':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$result = $vf->getSelfServiceReport($serviceID);
|
||||
|
||||
if ($result !== false) {
|
||||
$vf->output(['success' => true, 'data' => $result], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Unable to retrieve self-service report'], true, true, 500);
|
||||
break;
|
||||
|
||||
/**
|
||||
* Add self-service credit.
|
||||
*/
|
||||
case 'selfServiceAddCredit':
|
||||
|
||||
$serviceID = $vf->validateServiceID(true);
|
||||
|
||||
if (!$vf->validateUserOwnsService($serviceID)) {
|
||||
$vf->output(['success' => false, 'errors' => 'service <> owner mismatch'], true, true, 403);
|
||||
}
|
||||
|
||||
$tokens = isset($_GET['tokens']) ? (float) $_GET['tokens'] : 0;
|
||||
if ($tokens <= 0) {
|
||||
$vf->output(['success' => false, 'errors' => 'Invalid credit amount. Must be a positive number.'], true, true, 400);
|
||||
}
|
||||
|
||||
$result = $vf->addSelfServiceCredit($serviceID, $tokens);
|
||||
|
||||
if ($result !== false) {
|
||||
$vf->output(['success' => true, 'data' => $result], true, true, 200);
|
||||
}
|
||||
|
||||
$vf->output(['success' => false, 'errors' => 'Failed to add credit'], true, true, 500);
|
||||
break;
|
||||
|
||||
default:
|
||||
/**
|
||||
*
|
||||
* No valid action was specified.
|
||||
*
|
||||
*/
|
||||
$vf->output(['success' => false, 'errors' => 'invalid action'], true, true, 200);
|
||||
$vf->output(['success' => false, 'errors' => 'invalid action'], true, true, 400);
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -1,17 +1,81 @@
|
||||
<?php
|
||||
|
||||
use WHMCS\Module\Server\VirtFusionDirect\ConfigureService;
|
||||
use WHMCS\Module\Server\VirtFusionDirect\Database;
|
||||
use WHMCS\User\User;
|
||||
|
||||
if (!defined("WHMCS")) {
|
||||
die("This file cannot be accessed directly");
|
||||
}
|
||||
|
||||
/**
|
||||
* Shopping Cart Validation Hook
|
||||
*
|
||||
* Validates that an operating system has been selected before checkout
|
||||
* for all VirtFusion products in the cart.
|
||||
*/
|
||||
add_hook('ShoppingCartValidateCheckout', 1, function ($vars) {
|
||||
$errors = [];
|
||||
|
||||
if (!isset($_SESSION['cart']['products']) || !is_array($_SESSION['cart']['products'])) {
|
||||
return $errors;
|
||||
}
|
||||
|
||||
foreach ($_SESSION['cart']['products'] as $key => $product) {
|
||||
$pid = $product['pid'] ?? null;
|
||||
if (!$pid) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$dbProduct = \WHMCS\Database\Capsule::table('tblproducts')
|
||||
->where('id', $pid)
|
||||
->where('servertype', 'VirtFusionDirect')
|
||||
->first();
|
||||
|
||||
if (!$dbProduct) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check if Initial Operating System custom field has a value
|
||||
if (isset($product['customfields']) && is_array($product['customfields'])) {
|
||||
$osSelected = false;
|
||||
$customFields = \WHMCS\Database\Capsule::table('tblcustomfields')
|
||||
->where('relid', $pid)
|
||||
->where('type', 'product')
|
||||
->get();
|
||||
|
||||
foreach ($customFields as $field) {
|
||||
if (strtolower(str_replace(' ', '', $field->fieldname)) === 'initialoperatingsystem') {
|
||||
$fieldValue = $product['customfields'][$field->id] ?? '';
|
||||
if (!empty($fieldValue) && is_numeric($fieldValue)) {
|
||||
$osSelected = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!$osSelected) {
|
||||
$errors[] = 'Please select an Operating System for your VPS order.';
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return $errors;
|
||||
});
|
||||
|
||||
/**
|
||||
* Client Area Footer Output Hook
|
||||
*
|
||||
* Dynamically converts hidden text fields for OS templates and SSH keys
|
||||
* into dropdown selects populated from the VirtFusion API.
|
||||
* Works with all WHMCS themes by using vanilla JavaScript and standard form-control classes.
|
||||
*/
|
||||
add_hook('ClientAreaFooterOutput', 1, function ($vars) {
|
||||
if (!isset($vars['productinfo']['module']) || $vars['productinfo']['module'] !== 'VirtFusionDirect') {
|
||||
return null;
|
||||
}
|
||||
|
||||
try {
|
||||
$cs = new ConfigureService();
|
||||
|
||||
$templates_data = $cs->fetchTemplates(
|
||||
@@ -27,53 +91,75 @@ add_hook('ClientAreaFooterOutput', 1, function ($vars) {
|
||||
foreach ($templates_data['data'] as $osCategory) {
|
||||
foreach ($osCategory['templates'] as $template) {
|
||||
$optionValue = $template['id'];
|
||||
$optionLabel = $template['name'] . " " . $template['version'] . " " . $template['variant'];
|
||||
$optionLabel = htmlspecialchars($template['name'] . " " . $template['version'] . " " . $template['variant'], ENT_QUOTES, 'UTF-8');
|
||||
$dropdownOptions[] = ['id' => $optionValue, 'name' => $optionLabel];
|
||||
}
|
||||
}
|
||||
|
||||
// Sort dropdownOptions alphabetically by the 'name' key
|
||||
usort($dropdownOptions, function ($a, $b) {
|
||||
return strcmp($a['name'], $b['name']);
|
||||
});
|
||||
|
||||
$sshKeys = $cs->getUserSshKeys($vars['loggedinuser']);
|
||||
$sshKeysOptions = array_map(function ($sshKey) {
|
||||
$sshKeys = [];
|
||||
$sshKeysOptions = [];
|
||||
if (isset($vars['loggedinuser']) && $vars['loggedinuser']) {
|
||||
$sshKeysData = $cs->getUserSshKeys($vars['loggedinuser']);
|
||||
if ($sshKeysData && isset($sshKeysData['data'])) {
|
||||
$sshKeysOptions = array_values(array_filter(array_map(function ($sshKey) {
|
||||
if ($sshKey['enabled'] === false) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return [
|
||||
'id' => $sshKey['id'],
|
||||
'name' => $sshKey['name']
|
||||
'name' => htmlspecialchars($sshKey['name'], ENT_QUOTES, 'UTF-8')
|
||||
];
|
||||
}, $sshKeys['data'] ?? []);
|
||||
}, $sshKeysData['data'])));
|
||||
}
|
||||
}
|
||||
|
||||
$osID = array_values(array_filter(array_map(function ($option) {
|
||||
if ($option['textid'] === 'initialoperatingsystem') {
|
||||
return $option['id'];
|
||||
}
|
||||
}, $vars['customfields'])));
|
||||
}, $vars['customfields'] ?? [])));
|
||||
|
||||
$sshID = array_values(array_filter(array_map(function ($option) {
|
||||
if ($option['textid'] === 'initialsshkey') {
|
||||
return $option['id'];
|
||||
}
|
||||
}, $vars['customfields'])));
|
||||
}, $vars['customfields'] ?? [])));
|
||||
|
||||
$osFieldId = $osID[0] ?? null;
|
||||
$sshFieldId = $sshID[0] ?? null;
|
||||
|
||||
if ($osFieldId === null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$systemUrl = Database::getSystemUrl();
|
||||
|
||||
// Construct the JavaScript code
|
||||
return "
|
||||
<script src=\"" . htmlspecialchars($systemUrl, ENT_QUOTES, 'UTF-8') . "modules/servers/VirtFusionDirect/templates/js/keygen.js?v=20260207\"></script>
|
||||
<script>
|
||||
document.addEventListener('DOMContentLoaded', function() {
|
||||
var osTemplates = " . json_encode($dropdownOptions) . ";
|
||||
var sshKeys = " . json_encode($sshKeysOptions) . ";
|
||||
var osTemplates = " . json_encode($dropdownOptions, JSON_THROW_ON_ERROR | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT) . ";
|
||||
var sshKeys = " . json_encode($sshKeysOptions, JSON_THROW_ON_ERROR | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT) . ";
|
||||
|
||||
var osInputField = document.querySelector('[name=\"customfield[" . ($osID[0] ?? null) . "]\"]');
|
||||
var sshInputField = document.querySelector('[name=\"customfield[" . ($sshID[0] ?? null) . "]\"]');
|
||||
var osInputField = document.querySelector('[name=\"customfield[" . (int) $osFieldId . "]\"]');
|
||||
var sshInputField = " . ($sshFieldId !== null ? "document.querySelector('[name=\"customfield[" . (int) $sshFieldId . "]\"]')" : "null") . ";
|
||||
var sshInputLabel = " . ($sshFieldId !== null ? "document.querySelector('[for=\"customfield" . (int) $sshFieldId . "\"]')" : "null") . ";
|
||||
|
||||
// Create dropdown options menu, then add it to the DOM then on change, update the regular input.
|
||||
if (!osInputField) return;
|
||||
|
||||
// Create OS dropdown
|
||||
var osSelect = document.createElement('select');
|
||||
osSelect.className = 'form-control';
|
||||
osSelect.setAttribute('id', 'vf-os-select');
|
||||
|
||||
var defaultOption = document.createElement('option');
|
||||
defaultOption.value = '';
|
||||
defaultOption.text = '-- Select Operating System --';
|
||||
osSelect.appendChild(defaultOption);
|
||||
|
||||
osTemplates.forEach(function(template) {
|
||||
var option = document.createElement('option');
|
||||
@@ -82,21 +168,148 @@ add_hook('ClientAreaFooterOutput', 1, function ($vars) {
|
||||
osSelect.appendChild(option);
|
||||
});
|
||||
|
||||
// Set the default value of the input field to the first option in the dropdown.
|
||||
osInputField.value = osSelect.options[0].value;
|
||||
|
||||
osSelect.addEventListener('change', function() {
|
||||
osInputField.value = this.value;
|
||||
console.log(this.value);
|
||||
});
|
||||
|
||||
osInputField.parentNode.insertBefore(osSelect, osInputField.nextSibling);
|
||||
osInputField.style.display = 'none';
|
||||
|
||||
// Handle SSH keys
|
||||
if (sshInputField) {
|
||||
// Create the paste-key textarea (hidden initially if keys exist)
|
||||
var sshPasteContainer = document.createElement('div');
|
||||
sshPasteContainer.setAttribute('id', 'vf-ssh-paste-container');
|
||||
sshPasteContainer.style.display = 'none';
|
||||
sshPasteContainer.style.marginTop = '8px';
|
||||
|
||||
var pasteLabel = document.createElement('label');
|
||||
pasteLabel.textContent = 'Paste your SSH public key:';
|
||||
pasteLabel.style.display = 'block';
|
||||
pasteLabel.style.marginBottom = '4px';
|
||||
|
||||
var pasteArea = document.createElement('textarea');
|
||||
pasteArea.className = 'form-control';
|
||||
pasteArea.setAttribute('id', 'vf-ssh-paste');
|
||||
pasteArea.setAttribute('rows', '3');
|
||||
pasteArea.setAttribute('placeholder', 'ssh-rsa AAAA... or ssh-ed25519 AAAA...');
|
||||
|
||||
pasteArea.addEventListener('input', function() {
|
||||
sshInputField.value = this.value.trim();
|
||||
});
|
||||
|
||||
sshPasteContainer.appendChild(pasteLabel);
|
||||
sshPasteContainer.appendChild(pasteArea);
|
||||
|
||||
// Generate key button
|
||||
var generateBtn = document.createElement('button');
|
||||
generateBtn.type = 'button';
|
||||
generateBtn.className = 'btn btn-outline-secondary btn-sm';
|
||||
generateBtn.textContent = 'Generate a new key';
|
||||
generateBtn.style.marginTop = '8px';
|
||||
|
||||
// Private key panel (hidden initially)
|
||||
var privKeyPanel = document.createElement('div');
|
||||
privKeyPanel.setAttribute('id', 'vf-privkey-panel');
|
||||
privKeyPanel.style.display = 'none';
|
||||
privKeyPanel.style.marginTop = '12px';
|
||||
privKeyPanel.style.border = '2px solid #dc3545';
|
||||
privKeyPanel.style.borderRadius = '6px';
|
||||
privKeyPanel.style.padding = '12px';
|
||||
|
||||
var privKeyWarning = document.createElement('div');
|
||||
privKeyWarning.style.color = '#dc3545';
|
||||
privKeyWarning.style.fontWeight = 'bold';
|
||||
privKeyWarning.style.marginBottom = '8px';
|
||||
privKeyWarning.textContent = 'Private Key — Save This Now! It will not be shown again.';
|
||||
|
||||
var privKeyArea = document.createElement('textarea');
|
||||
privKeyArea.className = 'form-control';
|
||||
privKeyArea.setAttribute('rows', '6');
|
||||
privKeyArea.setAttribute('readonly', 'readonly');
|
||||
privKeyArea.style.fontFamily = 'monospace';
|
||||
privKeyArea.style.fontSize = '12px';
|
||||
privKeyArea.style.marginBottom = '8px';
|
||||
|
||||
var privKeyBtnRow = document.createElement('div');
|
||||
privKeyBtnRow.style.display = 'flex';
|
||||
privKeyBtnRow.style.gap = '8px';
|
||||
privKeyBtnRow.style.alignItems = 'center';
|
||||
privKeyBtnRow.style.flexWrap = 'wrap';
|
||||
|
||||
var downloadBtn = document.createElement('button');
|
||||
downloadBtn.type = 'button';
|
||||
downloadBtn.className = 'btn btn-primary btn-sm';
|
||||
downloadBtn.textContent = 'Download';
|
||||
|
||||
var copyBtn = document.createElement('button');
|
||||
copyBtn.type = 'button';
|
||||
copyBtn.className = 'btn btn-default btn-secondary btn-sm';
|
||||
copyBtn.textContent = 'Copy to Clipboard';
|
||||
|
||||
var pubKeyConfirm = document.createElement('span');
|
||||
pubKeyConfirm.style.color = '#28a745';
|
||||
pubKeyConfirm.style.fontWeight = 'bold';
|
||||
pubKeyConfirm.textContent = 'Public key set automatically.';
|
||||
|
||||
privKeyBtnRow.appendChild(downloadBtn);
|
||||
privKeyBtnRow.appendChild(copyBtn);
|
||||
privKeyBtnRow.appendChild(pubKeyConfirm);
|
||||
privKeyPanel.appendChild(privKeyWarning);
|
||||
privKeyPanel.appendChild(privKeyArea);
|
||||
privKeyPanel.appendChild(privKeyBtnRow);
|
||||
|
||||
downloadBtn.addEventListener('click', function() {
|
||||
vfDownloadFile('id_ed25519', privKeyArea.value);
|
||||
});
|
||||
|
||||
copyBtn.addEventListener('click', function() {
|
||||
navigator.clipboard.writeText(privKeyArea.value).then(function() {
|
||||
copyBtn.textContent = 'Copied!';
|
||||
setTimeout(function() { copyBtn.textContent = 'Copy to Clipboard'; }, 2000);
|
||||
});
|
||||
});
|
||||
|
||||
// Error message for unsupported browsers
|
||||
var genErrorMsg = document.createElement('div');
|
||||
genErrorMsg.style.display = 'none';
|
||||
genErrorMsg.style.marginTop = '8px';
|
||||
genErrorMsg.style.color = '#dc3545';
|
||||
genErrorMsg.textContent = 'Your browser does not support Ed25519 key generation. Please paste your public key manually.';
|
||||
|
||||
generateBtn.addEventListener('click', async function() {
|
||||
generateBtn.disabled = true;
|
||||
generateBtn.textContent = 'Generating...';
|
||||
try {
|
||||
var keys = await vfGenerateSSHKey();
|
||||
var sshSelect = document.getElementById('vf-ssh-select');
|
||||
if (sshSelect) {
|
||||
sshSelect.value = '__new__';
|
||||
sshPasteContainer.style.display = 'block';
|
||||
}
|
||||
pasteArea.value = keys.publicKey;
|
||||
sshInputField.value = keys.publicKey;
|
||||
privKeyArea.value = keys.privateKey;
|
||||
privKeyPanel.style.display = 'block';
|
||||
genErrorMsg.style.display = 'none';
|
||||
} catch (e) {
|
||||
genErrorMsg.style.display = 'block';
|
||||
privKeyPanel.style.display = 'none';
|
||||
} finally {
|
||||
generateBtn.disabled = false;
|
||||
generateBtn.textContent = 'Generate a new key';
|
||||
}
|
||||
});
|
||||
|
||||
if (sshKeys.length > 0) {
|
||||
// Create dropdown options menu, then add it to the DOM then on change, update the regular input.
|
||||
var sshSelect = document.createElement('select');
|
||||
sshSelect.className = 'form-control';
|
||||
sshSelect.setAttribute('id', 'vf-ssh-select');
|
||||
|
||||
var sshDefaultOption = document.createElement('option');
|
||||
sshDefaultOption.value = '';
|
||||
sshDefaultOption.text = '-- No SSH Key (Optional) --';
|
||||
sshSelect.appendChild(sshDefaultOption);
|
||||
|
||||
sshKeys.forEach(function(sshkey) {
|
||||
var option = document.createElement('option');
|
||||
@@ -105,19 +318,113 @@ add_hook('ClientAreaFooterOutput', 1, function ($vars) {
|
||||
sshSelect.appendChild(option);
|
||||
});
|
||||
|
||||
// Set the default value of the input field to the first option in the dropdown.
|
||||
sshInputField.value = sshSelect.options[0].value;
|
||||
// Add new key option
|
||||
var addNewOption = document.createElement('option');
|
||||
addNewOption.value = '__new__';
|
||||
addNewOption.text = 'Add new key...';
|
||||
sshSelect.appendChild(addNewOption);
|
||||
|
||||
sshSelect.addEventListener('change', function() {
|
||||
if (this.value === '__new__') {
|
||||
sshPasteContainer.style.display = 'block';
|
||||
sshInputField.value = '';
|
||||
} else {
|
||||
sshPasteContainer.style.display = 'none';
|
||||
document.getElementById('vf-ssh-paste').value = '';
|
||||
sshInputField.value = this.value;
|
||||
}
|
||||
});
|
||||
|
||||
sshInputField.parentNode.insertBefore(sshSelect, sshInputField.nextSibling);
|
||||
sshSelect.parentNode.insertBefore(sshPasteContainer, sshSelect.nextSibling);
|
||||
sshPasteContainer.parentNode.insertBefore(generateBtn, sshPasteContainer.nextSibling);
|
||||
generateBtn.parentNode.insertBefore(genErrorMsg, generateBtn.nextSibling);
|
||||
genErrorMsg.parentNode.insertBefore(privKeyPanel, genErrorMsg.nextSibling);
|
||||
sshInputField.style.display = 'none';
|
||||
} else {
|
||||
// No existing keys — show the paste textarea directly
|
||||
sshPasteContainer.style.display = 'block';
|
||||
sshInputField.parentNode.insertBefore(sshPasteContainer, sshInputField.nextSibling);
|
||||
sshPasteContainer.parentNode.insertBefore(generateBtn, sshPasteContainer.nextSibling);
|
||||
generateBtn.parentNode.insertBefore(genErrorMsg, generateBtn.nextSibling);
|
||||
genErrorMsg.parentNode.insertBefore(privKeyPanel, genErrorMsg.nextSibling);
|
||||
sshInputField.style.display = 'none';
|
||||
}
|
||||
}
|
||||
|
||||
// Slider UI: enhance known configurable option selects with range sliders
|
||||
var sliderResourceNames = ['Memory', 'CPU Cores', 'Storage', 'Bandwidth', 'Inbound Network Speed', 'Outbound Network Speed'];
|
||||
var sliderUnits = {
|
||||
'Memory': 'MB', 'CPU Cores': 'Core(s)', 'Storage': 'GB',
|
||||
'Bandwidth': 'GB', 'Inbound Network Speed': 'Mbps', 'Outbound Network Speed': 'Mbps'
|
||||
};
|
||||
|
||||
var configSelects = document.querySelectorAll('select[name^=\"configoption[\"]');
|
||||
configSelects.forEach(function(sel) {
|
||||
// Find the label for this select
|
||||
var label = null;
|
||||
var labelEl = sel.closest('.form-group, .row');
|
||||
if (labelEl) {
|
||||
label = labelEl.querySelector('label');
|
||||
}
|
||||
if (!label) return;
|
||||
|
||||
var labelText = label.textContent.trim();
|
||||
var matchedResource = null;
|
||||
sliderResourceNames.forEach(function(name) {
|
||||
if (labelText.indexOf(name) !== -1) {
|
||||
matchedResource = name;
|
||||
}
|
||||
});
|
||||
if (!matchedResource) return;
|
||||
|
||||
var options = [];
|
||||
for (var i = 0; i < sel.options.length; i++) {
|
||||
options.push({
|
||||
value: sel.options[i].value,
|
||||
label: sel.options[i].text
|
||||
});
|
||||
}
|
||||
if (options.length < 2) return;
|
||||
|
||||
var unit = sliderUnits[matchedResource] || '';
|
||||
|
||||
// Create slider container
|
||||
var container = document.createElement('div');
|
||||
container.className = 'vf-slider-container';
|
||||
|
||||
var valueDisplay = document.createElement('div');
|
||||
valueDisplay.className = 'vf-slider-value';
|
||||
valueDisplay.textContent = options[sel.selectedIndex || 0].label + (unit ? ' ' + unit : '');
|
||||
|
||||
var slider = document.createElement('input');
|
||||
slider.type = 'range';
|
||||
slider.className = 'vf-slider form-range';
|
||||
slider.min = '0';
|
||||
slider.max = String(options.length - 1);
|
||||
slider.step = '1';
|
||||
slider.value = String(sel.selectedIndex || 0);
|
||||
|
||||
slider.addEventListener('input', function() {
|
||||
var idx = parseInt(this.value);
|
||||
sel.selectedIndex = idx;
|
||||
valueDisplay.textContent = options[idx].label + (unit ? ' ' + unit : '');
|
||||
// Trigger change event on hidden select for WHMCS pricing
|
||||
var evt = new Event('change', { bubbles: true });
|
||||
sel.dispatchEvent(evt);
|
||||
});
|
||||
|
||||
container.appendChild(valueDisplay);
|
||||
container.appendChild(slider);
|
||||
|
||||
sel.parentNode.insertBefore(container, sel.nextSibling);
|
||||
sel.style.display = 'none';
|
||||
});
|
||||
});
|
||||
</script>
|
||||
";
|
||||
} catch (\Throwable $e) {
|
||||
// Silently fail - don't break the checkout page
|
||||
return null;
|
||||
}
|
||||
});
|
||||
@@ -7,6 +7,7 @@ class AdminHTML
|
||||
|
||||
public static function options($systemUrl, $serviceId)
|
||||
{
|
||||
$systemUrl = htmlspecialchars($systemUrl, ENT_QUOTES, 'UTF-8');
|
||||
return <<<EOT
|
||||
<button onclick="impersonateServerOwner('${serviceId}', '${systemUrl}')" type="button" class="btn btn-primary">Impersonate Server Owner</button>
|
||||
<span class="text-info"> A valid VirtFusion admin session in the same browser is required for this functionality to work.</span>
|
||||
@@ -15,6 +16,7 @@ EOT;
|
||||
|
||||
public static function serverObject($serverObject)
|
||||
{
|
||||
$serverObject = htmlspecialchars($serverObject, ENT_QUOTES, 'UTF-8');
|
||||
return <<<EOT
|
||||
<textarea class="form-control" name="modulefields[1]" rows="10" style="width: 100%" disabled>${serverObject}</textarea>
|
||||
EOT;
|
||||
@@ -31,9 +33,10 @@ EOT;
|
||||
|
||||
public static function serverInfo($systemUrl, $serviceId)
|
||||
{
|
||||
$systemUrl = htmlspecialchars($systemUrl, ENT_QUOTES, 'UTF-8');
|
||||
return <<<EOT
|
||||
<link href="${systemUrl}modules/servers/VirtFusionDirect/templates/css/module.css" rel="stylesheet">
|
||||
<script src="${systemUrl}modules/servers/VirtFusionDirect/templates/js/module.js"></script>
|
||||
<link href="${systemUrl}modules/servers/VirtFusionDirect/templates/css/module.css?v=20260207" rel="stylesheet">
|
||||
<script src="${systemUrl}modules/servers/VirtFusionDirect/templates/js/module.js?v=20260207"></script>
|
||||
<div id="vf-loader" class="vf-loader">
|
||||
<div id="vf-loading"></div>
|
||||
</div>
|
||||
|
||||
@@ -26,6 +26,8 @@ class ConfigureService extends Module
|
||||
*/
|
||||
public function fetchPackageId(string $packageName): ?int
|
||||
{
|
||||
if (!$this->cp) return null;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
$response = $request->get(
|
||||
@@ -70,6 +72,8 @@ class ConfigureService extends Module
|
||||
return null;
|
||||
}
|
||||
|
||||
if (!$this->cp) return null;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
$response = $request->get(
|
||||
@@ -90,10 +94,14 @@ class ConfigureService extends Module
|
||||
return null;
|
||||
}
|
||||
|
||||
if (!$this->cp) return null;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
$vfUser = $this->getVFUserDetails($user['id']);
|
||||
|
||||
if (!$vfUser) return null;
|
||||
|
||||
$response = $request->get(
|
||||
sprintf("%s/ssh_keys/user/%d", $this->cp['url'], $vfUser['id'])
|
||||
);
|
||||
@@ -108,6 +116,8 @@ class ConfigureService extends Module
|
||||
*/
|
||||
public function getVFUserDetails(int $id): ?array
|
||||
{
|
||||
if (!$this->cp) return null;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
$response = $this->decodeResponseFromJson($request->get(
|
||||
@@ -120,34 +130,83 @@ class ConfigureService extends Module
|
||||
/**
|
||||
* @param int $id
|
||||
* @param array $vars
|
||||
* @param int|null $vfUserId VirtFusion user ID (for creating SSH keys from raw public key)
|
||||
* @return bool
|
||||
*/
|
||||
public function initServerBuild(int $id, array $vars): bool
|
||||
public function initServerBuild(int $id, array $vars, ?int $vfUserId = null): bool
|
||||
{
|
||||
if (!$this->cp) return false;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
// Generate a random 8 character hostname
|
||||
$hostname = substr(str_shuffle('abcdefghijklmnopqrstuvwxyz'), 0, 8);
|
||||
|
||||
$sshKeyValue = $vars['customfields']['Initial SSH Key'] ?? null;
|
||||
$sshKeyId = null;
|
||||
|
||||
if (!empty($sshKeyValue)) {
|
||||
if (is_numeric($sshKeyValue)) {
|
||||
// Existing SSH key ID
|
||||
$sshKeyId = (int) $sshKeyValue;
|
||||
} elseif (preg_match('/^ssh-/', $sshKeyValue) && $vfUserId) {
|
||||
// Raw public key — create it via API
|
||||
$sshKeyId = $this->createUserSshKey($vfUserId, $sshKeyValue);
|
||||
}
|
||||
}
|
||||
|
||||
$inputData = [
|
||||
"operatingSystemId" => $vars['customfields']['Initial Operating System'],
|
||||
"operatingSystemId" => $vars['customfields']['Initial Operating System'] ?? null,
|
||||
"name" => $hostname,
|
||||
"sshKeys" => [
|
||||
$vars['customfields']['Initial SSH Key']
|
||||
],
|
||||
'email' => true
|
||||
];
|
||||
|
||||
if (empty($vars['customfields']['Initial SSH Key'])) {
|
||||
unset($inputData['sshKeys']);
|
||||
if ($sshKeyId) {
|
||||
$inputData['sshKeys'] = [$sshKeyId];
|
||||
}
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode($inputData));
|
||||
|
||||
$request->post(
|
||||
$response = $request->post(
|
||||
sprintf("%s/servers/%d/build", $this->cp['url'], $id)
|
||||
);
|
||||
|
||||
return true;
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $response);
|
||||
|
||||
return ($httpCode == 200 || $httpCode == 201);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an SSH key for a VirtFusion user from a raw public key string.
|
||||
*
|
||||
* @param int $userId VirtFusion user ID
|
||||
* @param string $publicKey Raw SSH public key (ssh-rsa ..., ssh-ed25519 ..., etc.)
|
||||
* @return int|null Created key ID or null on failure
|
||||
*/
|
||||
public function createUserSshKey(int $userId, string $publicKey): ?int
|
||||
{
|
||||
if (!$this->cp) return null;
|
||||
|
||||
$request = $this->initCurl($this->cp['token']);
|
||||
|
||||
$keyData = [
|
||||
'userId' => $userId,
|
||||
'name' => 'WHMCS-' . date('Y-m-d'),
|
||||
'publicKey' => trim($publicKey),
|
||||
];
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode($keyData));
|
||||
$response = $request->post($this->cp['url'] . '/ssh_keys');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $response);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
$data = json_decode($response, true);
|
||||
return $data['data']['id'] ?? null;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
@@ -8,12 +8,14 @@ class Curl
|
||||
private $data;
|
||||
private $customOptions = [];
|
||||
private $defaultOptions = [
|
||||
CURLOPT_SSL_VERIFYPEER => false,
|
||||
CURLOPT_SSL_VERIFYHOST => false,
|
||||
CURLOPT_SSL_VERIFYPEER => true,
|
||||
CURLOPT_SSL_VERIFYHOST => 2,
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_USERAGENT => 'CURL',
|
||||
CURLOPT_USERAGENT => 'VirtFusion-WHMCS/2.0',
|
||||
CURLOPT_HEADER => false,
|
||||
CURLOPT_NOBODY => false,
|
||||
CURLOPT_TIMEOUT => 30,
|
||||
CURLOPT_CONNECTTIMEOUT => 10,
|
||||
];
|
||||
|
||||
|
||||
@@ -24,7 +26,7 @@ class Curl
|
||||
|
||||
public function useCookies()
|
||||
{
|
||||
$cookiesFile = tempnam('/tmp', 'virtfusion_cookies');
|
||||
$cookiesFile = tempnam(sys_get_temp_dir(), 'virtfusion_cookies');
|
||||
$this->defaultOptions[CURLOPT_COOKIEFILE] = $cookiesFile;
|
||||
$this->defaultOptions[CURLOPT_COOKIEJAR] = $cookiesFile;
|
||||
}
|
||||
@@ -57,6 +59,15 @@ class Curl
|
||||
return $this->send('PUT', $url);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param null $url
|
||||
* @return bool|string|void
|
||||
*/
|
||||
public function patch($url = null)
|
||||
{
|
||||
return $this->send('PATCH', $url);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $method
|
||||
* @param $url
|
||||
@@ -66,7 +77,7 @@ class Curl
|
||||
{
|
||||
if ($url === null) {
|
||||
if (!isset($this->customOptions[CURLOPT_URL]) || empty($this->customOptions[CURLOPT_URL])) {
|
||||
exit('empty url');
|
||||
throw new \RuntimeException('Curl: empty URL provided');
|
||||
}
|
||||
}
|
||||
$this->addOption(CURLOPT_CUSTOMREQUEST, $method);
|
||||
@@ -84,6 +95,12 @@ class Curl
|
||||
$response = curl_exec($this->ch);
|
||||
|
||||
$this->data['info'] = curl_getinfo($this->ch);
|
||||
|
||||
if ($response === false) {
|
||||
$this->data['info']['curl_error'] = curl_error($this->ch);
|
||||
$this->data['info']['curl_errno'] = curl_errno($this->ch);
|
||||
}
|
||||
|
||||
if (isset($this->customOptions[CURLOPT_HEADER]) && $this->customOptions[CURLOPT_HEADER]) {
|
||||
$this->data['info']['request_header'] = trim($this->data['info']['request_header']);
|
||||
$this->processHeaders($response);
|
||||
|
||||
@@ -54,6 +54,7 @@ class Database
|
||||
public static function getSystemUrl()
|
||||
{
|
||||
$url = DB::table('tblconfiguration')->where('setting', '=', 'SystemURL')->first();
|
||||
if (!$url) return '';
|
||||
return $url->value;
|
||||
}
|
||||
|
||||
|
||||
@@ -2,46 +2,45 @@
|
||||
|
||||
namespace WHMCS\Module\Server\VirtFusionDirect;
|
||||
|
||||
|
||||
class Module
|
||||
{
|
||||
public function __construct()
|
||||
{
|
||||
error_reporting(0);
|
||||
Database::schema();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param bool $exitOnError
|
||||
* @return mixed
|
||||
* @return string
|
||||
*/
|
||||
public function validateAction($exitOnError = true)
|
||||
{
|
||||
if (!isset($_GET['action'])) {
|
||||
$this->output(['errors' => 'no action specified'], true, $exitOnError, 200);
|
||||
$this->output(['success' => false, 'errors' => 'no action specified'], true, $exitOnError, 400);
|
||||
}
|
||||
return $_GET['action'];
|
||||
return preg_replace('/[^a-zA-Z0-9_]/', '', $_GET['action']);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param bool $exitOnError
|
||||
* @return mixed
|
||||
* @return int
|
||||
*/
|
||||
public function validateServiceID($exitOnError = true)
|
||||
{
|
||||
if (!isset($_GET['serviceID'])) {
|
||||
$this->output(['errors' => 'no serviceID specified'], true, $exitOnError, 200);
|
||||
if (!isset($_GET['serviceID']) || !is_numeric($_GET['serviceID'])) {
|
||||
$this->output(['success' => false, 'errors' => 'no valid serviceID specified'], true, $exitOnError, 400);
|
||||
}
|
||||
return $_GET['serviceID'];
|
||||
return (int) $_GET['serviceID'];
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $serviceID
|
||||
* @param int $serviceID
|
||||
* @param bool $exitOnError
|
||||
* @return bool
|
||||
* @return int|false
|
||||
*/
|
||||
public function validateUserOwnsService($serviceID, $exitOnError = true)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$currentUser = new \WHMCS\Authentication\CurrentUser;
|
||||
$client = $currentUser->client();
|
||||
|
||||
@@ -57,27 +56,33 @@ class Module
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $serviceID
|
||||
* @param int $serviceID
|
||||
* @return false|string
|
||||
*/
|
||||
public function fetchLoginTokens($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/users/' . $whmcsService->userid . '/serverAuthenticationTokens/' . $service->server_id);
|
||||
$data = $request->post($cp['url'] . '/users/' . (int) $whmcsService->userid . '/serverAuthenticationTokens/' . (int) $service->server_id);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == '200') {
|
||||
$data = json_decode($data);
|
||||
if (isset($data->data->authentication->endpoint_complete)) {
|
||||
return $cp['base_url'] . $data->data->authentication->endpoint_complete;
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -117,13 +122,18 @@ class Module
|
||||
|
||||
public function fetchServerData($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/servers/' . $service->server_id);
|
||||
$data = $request->get($cp['url'] . '/servers/' . (int) $service->server_id);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
@@ -134,13 +144,505 @@ class Module
|
||||
return false;
|
||||
}
|
||||
|
||||
public function resetUserPassword($serviceID, $clientID)
|
||||
/**
|
||||
* Execute a power action on a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param string $action One of: boot, shutdown, restart, poweroff
|
||||
* @return object|false
|
||||
*/
|
||||
public function serverPowerAction($serviceID, $action)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$allowedActions = ['boot', 'shutdown', 'restart', 'poweroff'];
|
||||
if (!in_array($action, $allowedActions, true)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/power/' . $action);
|
||||
|
||||
Log::insert(__FUNCTION__ . ':' . $action, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 204) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Rebuild/reinstall a server with a new OS.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param int $osId Operating system template ID
|
||||
* @param string|null $hostname Optional new hostname
|
||||
* @return object|false
|
||||
*/
|
||||
public function rebuildServer($serviceID, $osId, $hostname = null)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$osId = (int) $osId;
|
||||
|
||||
if ($osId <= 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
|
||||
$buildData = [
|
||||
'operatingSystemId' => $osId,
|
||||
'email' => true,
|
||||
];
|
||||
|
||||
if ($hostname !== null && $hostname !== '') {
|
||||
$buildData['hostname'] = $hostname;
|
||||
}
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode($buildData));
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/build');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Rename a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param string $newName
|
||||
* @return bool
|
||||
*/
|
||||
public function renameServer($serviceID, $newName)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$newName = trim($newName);
|
||||
|
||||
if (empty($newName) || strlen($newName) > 255) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode(['name' => $newName]));
|
||||
$data = $request->patch($cp['url'] . '/servers/' . (int) $service->server_id . '/name');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
return ($httpCode == 200 || $httpCode == 204);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch available OS templates for a server's package.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return array|false
|
||||
*/
|
||||
public function fetchOsTemplates($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$product = \WHMCS\Database\Capsule::table('tblproducts')->where('id', $whmcsService->packageid)->first();
|
||||
if (!$product || !$product->configoption2) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/media/templates/fromServerPackageSpec/' . (int) $product->configoption2);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == '200') {
|
||||
$templates = json_decode($data, true);
|
||||
$result = [];
|
||||
if (isset($templates['data'])) {
|
||||
foreach ($templates['data'] as $osCategory) {
|
||||
foreach ($osCategory['templates'] as $template) {
|
||||
$result[] = [
|
||||
'id' => $template['id'],
|
||||
'name' => $template['name'] . ' ' . $template['version'] . ' ' . $template['variant'],
|
||||
];
|
||||
}
|
||||
}
|
||||
usort($result, function ($a, $b) {
|
||||
return strcmp($a['name'], $b['name']);
|
||||
});
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// IP Address Management
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Add an IPv4 address to a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return object|false
|
||||
*/
|
||||
public function addIPv4($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/ipv4');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove an IPv4 address from a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param string $ipAddress The IPv4 address to remove
|
||||
* @return object|false
|
||||
*/
|
||||
public function removeIPv4($serviceID, $ipAddress)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$ipAddress = filter_var($ipAddress, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4);
|
||||
if (!$ipAddress) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode(['address' => $ipAddress]));
|
||||
$data = $request->delete($cp['url'] . '/servers/' . (int) $service->server_id . '/ipv4');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 204) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add an IPv6 subnet to a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return object|false
|
||||
*/
|
||||
public function addIPv6($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/ipv6');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove an IPv6 subnet from a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param string $subnet The IPv6 subnet to remove
|
||||
* @return object|false
|
||||
*/
|
||||
public function removeIPv6($serviceID, $subnet)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$subnet = trim($subnet);
|
||||
if (empty($subnet)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode(['subnet' => $subnet]));
|
||||
$data = $request->delete($cp['url'] . '/servers/' . (int) $service->server_id . '/ipv6');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 204) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Backup Management
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Assign a backup plan to a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param int $planId Backup plan ID (0 to remove)
|
||||
* @return object|false
|
||||
*/
|
||||
public function assignBackupPlan($serviceID, $planId)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$planId = (int) $planId;
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode(['planId' => $planId]));
|
||||
|
||||
if ($planId > 0) {
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/backup/plan');
|
||||
} else {
|
||||
$data = $request->delete($cp['url'] . '/servers/' . (int) $service->server_id . '/backup/plan');
|
||||
}
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 204) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// VNC Console
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Get VNC console connection details for a server.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return array|false
|
||||
*/
|
||||
public function getVncConsole($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/servers/' . (int) $service->server_id . '/vnc');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == 200) {
|
||||
return json_decode($data, true);
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Resource Modification
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Modify a server resource (memory, cpuCores, or traffic).
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param string $resource One of: memory, cpuCores, traffic
|
||||
* @param int $value New value for the resource
|
||||
* @return object|false
|
||||
*/
|
||||
public function modifyResource($serviceID, $resource, $value)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$allowedResources = ['memory', 'cpuCores', 'traffic'];
|
||||
if (!in_array($resource, $allowedResources, true)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$value = (int) $value;
|
||||
if ($value < 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode([$resource => $value]));
|
||||
$data = $request->put($cp['url'] . '/servers/' . (int) $service->server_id . '/modify/' . $resource);
|
||||
|
||||
Log::insert(__FUNCTION__ . ':' . $resource, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 204) {
|
||||
return json_decode($data) ?: (object) ['success' => true];
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Dry Run Validation
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Validate server creation parameters without actually creating a server.
|
||||
*
|
||||
* @param array $options Server creation options
|
||||
* @param int $serverId WHMCS server ID for API credentials
|
||||
* @return array ['valid' => bool, 'errors' => array]
|
||||
*/
|
||||
public function validateServerCreation($options, $serverId)
|
||||
{
|
||||
$cp = $this->getCP($serverId, !$serverId);
|
||||
if (!$cp) {
|
||||
return ['valid' => false, 'errors' => ['No control server found']];
|
||||
}
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode($options));
|
||||
$data = $request->post($cp['url'] . '/servers?dryRun=true');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
$response = json_decode($data, true);
|
||||
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
return ['valid' => true, 'errors' => []];
|
||||
}
|
||||
|
||||
$errors = [];
|
||||
if (isset($response['errors']) && is_array($response['errors'])) {
|
||||
$errors = $response['errors'];
|
||||
} elseif (isset($response['msg'])) {
|
||||
$errors = [$response['msg']];
|
||||
} else {
|
||||
$errors = ['Validation failed with HTTP ' . $httpCode];
|
||||
}
|
||||
|
||||
return ['valid' => false, 'errors' => $errors];
|
||||
}
|
||||
|
||||
public function resetUserPassword($serviceID, $clientID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$clientID = (int) $clientID;
|
||||
$service = Database::getSystemService($serviceID);
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/users/' . $clientID . '/byExtRelation/resetPassword');
|
||||
|
||||
@@ -201,7 +703,7 @@ class Module
|
||||
return true;
|
||||
}
|
||||
|
||||
$this->output(['errors' => 'unauthenticated'], true, true, 200);
|
||||
$this->output(['success' => false, 'errors' => 'unauthenticated'], true, true, 401);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -213,7 +715,7 @@ class Module
|
||||
return true;
|
||||
}
|
||||
|
||||
$this->output(['errors' => 'unauthenticated'], true, true, 200);
|
||||
$this->output(['success' => false, 'errors' => 'unauthenticated'], true, true, 401);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -233,6 +735,128 @@ class Module
|
||||
return $curl;
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Self Service — Credit & Usage
|
||||
// =========================================================================
|
||||
|
||||
/**
|
||||
* Get self-service usage data for a WHMCS client.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return array|false
|
||||
*/
|
||||
public function getSelfServiceUsage($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/selfService/usage/byUserExtRelationId/' . (int) $whmcsService->userid);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == 200) {
|
||||
return json_decode($data, true);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get self-service billing report for a WHMCS client.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return array|false
|
||||
*/
|
||||
public function getSelfServiceReport($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/selfService/report/byUserExtRelationId/' . (int) $whmcsService->userid);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == 200) {
|
||||
return json_decode($data, true);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add self-service credit for a WHMCS client.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @param float $tokens Amount of credit tokens to add
|
||||
* @param string $reference Reference text for the transaction
|
||||
* @return array|false
|
||||
*/
|
||||
public function addSelfServiceCredit($serviceID, $tokens, $reference = '')
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$tokens = (float) $tokens;
|
||||
|
||||
if ($tokens <= 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode([
|
||||
'tokens' => $tokens,
|
||||
'reference_1' => $reference ?: 'WHMCS Top-up',
|
||||
'reference_2' => 'Service #' . $serviceID,
|
||||
]));
|
||||
$data = $request->post($cp['url'] . '/selfService/credit/byUserExtRelationId/' . (int) $whmcsService->userid);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
$httpCode = $request->getRequestInfo('http_code');
|
||||
if ($httpCode == 200 || $httpCode == 201) {
|
||||
return json_decode($data, true);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get available self-service currencies.
|
||||
*
|
||||
* @param int $serviceID
|
||||
* @return array|false
|
||||
*/
|
||||
public function getSelfServiceCurrencies($serviceID)
|
||||
{
|
||||
$serviceID = (int) $serviceID;
|
||||
$whmcsService = Database::getWhmcsService($serviceID);
|
||||
if (!$whmcsService) return false;
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return false;
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/selfService/currencies');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') == 200) {
|
||||
return json_decode($data, true);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Decodes a response from JSON into an associative array.
|
||||
*
|
||||
|
||||
@@ -23,88 +23,82 @@ class ModuleFunctions extends Module
|
||||
try {
|
||||
|
||||
/**
|
||||
*
|
||||
* If the service exists in the custom table, Cancel the create account action.
|
||||
*
|
||||
* If the service exists in the custom table, cancel the create account action.
|
||||
*/
|
||||
if (Database::checkSystemService($params['serviceid'])) {
|
||||
return 'Service already exists. You must run a termination first.';
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* If no VirtFusionDirect control server exists, cancel the create account action.
|
||||
*
|
||||
*/
|
||||
|
||||
$server = $params['serverid'] ?: false;
|
||||
$cp = $this->getCP($server, $server ? false : true);
|
||||
$cp = $this->getCP($server, !$server);
|
||||
|
||||
if (!$cp) {
|
||||
return 'No Control server found.';
|
||||
return 'No Control server found. Please ensure a VirtFusion server is configured in WHMCS.';
|
||||
}
|
||||
|
||||
Log::insert(__FUNCTION__, $params, []);
|
||||
|
||||
/**
|
||||
*
|
||||
* Does a user account in VirtFusion match this account (byExtRelationId) in WHMCS.
|
||||
*
|
||||
*/
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/users/' . $params['userid'] . '/byExtRelation');
|
||||
$data = $request->get($cp['url'] . '/users/' . (int) $params['userid'] . '/byExtRelation');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
switch ($request->getRequestInfo('http_code')) {
|
||||
case 200:
|
||||
|
||||
/**
|
||||
*
|
||||
* A user with relation ID exists in VirtFusion. We can provision under that account.
|
||||
*
|
||||
*/
|
||||
break;
|
||||
|
||||
case 404:
|
||||
|
||||
/**
|
||||
*
|
||||
* A user doesn't exist in VirtFusion. We should attempt to create one.
|
||||
*
|
||||
*/
|
||||
$user = Database::getUser($params['userid']);
|
||||
|
||||
if (!$user) {
|
||||
return 'WHMCS user not found for ID ' . (int) $params['userid'];
|
||||
}
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode(
|
||||
[
|
||||
$userData = [
|
||||
"name" => $user->firstname . ' ' . $user->lastname,
|
||||
"email" => $user->email,
|
||||
"extRelationId" => $user->id
|
||||
]
|
||||
));
|
||||
"extRelationId" => $user->id,
|
||||
];
|
||||
|
||||
// Enable self-service billing if configured
|
||||
$selfServiceMode = (int) ($params['configoption4'] ?? 0);
|
||||
if ($selfServiceMode > 0) {
|
||||
$userData['selfService'] = $selfServiceMode;
|
||||
$userData['selfServiceHourlyCredit'] = in_array($selfServiceMode, [1, 3]);
|
||||
}
|
||||
|
||||
$request->addOption(CURLOPT_POSTFIELDS, json_encode($userData));
|
||||
|
||||
$data = $request->post($cp['url'] . '/users');
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
|
||||
if ($request->getRequestInfo('http_code') !== 201) {
|
||||
return 'Unable to create user.';
|
||||
return 'Unable to create user in VirtFusion. API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
break;
|
||||
default:
|
||||
return 'Error processing user account.';
|
||||
break;
|
||||
return 'Error processing user account. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
|
||||
$data = json_decode($data);
|
||||
|
||||
/**
|
||||
*
|
||||
* A user is available. We can now attempt to create a server.
|
||||
*
|
||||
*/
|
||||
|
||||
$configOptionDefaultNaming = [
|
||||
@@ -124,26 +118,27 @@ class ModuleFunctions extends Module
|
||||
$configOptionCustomNaming = [];
|
||||
|
||||
if (file_exists(ROOTDIR . '/modules/servers/VirtFusionDirect/config/ConfigOptionMapping.php')) {
|
||||
$configOptionCustomNaming = require_once ROOTDIR . '/modules/servers/VirtFusionDirect/config/ConfigOptionMapping.php';
|
||||
$configOptionCustomNaming = require ROOTDIR . '/modules/servers/VirtFusionDirect/config/ConfigOptionMapping.php';
|
||||
}
|
||||
|
||||
$options = [
|
||||
"packageId" => $params['configoption2'],
|
||||
"packageId" => (int) $params['configoption2'],
|
||||
"userId" => $data->data->id,
|
||||
"hypervisorId" => $params['configoption1'],
|
||||
"ipv4" => $params['configoption3'],
|
||||
"hypervisorId" => (int) $params['configoption1'],
|
||||
"ipv4" => (int) $params['configoption3'],
|
||||
];
|
||||
|
||||
if (array_key_exists('configoptions', $params)) {
|
||||
foreach ($configOptionDefaultNaming as $key => $option) {
|
||||
$currentOption = array_key_exists($key, $configOptionCustomNaming) ? $configOptionCustomNaming[$key] : $option;
|
||||
if (array_key_exists($currentOption, $params['configoptions'])) {
|
||||
// If the option key is "Memory" and the value is less than 1024, we need to convert it to MB
|
||||
$value = $params['configoptions'][$currentOption];
|
||||
// If the option key is "Memory" and the value is less than 1024, convert to MB
|
||||
// VirtFusion expects memory in MB.
|
||||
if ($currentOption === 'Memory' && $params['configoptions'][$currentOption] < 1024) {
|
||||
$options[$key] = $params['configoptions'][$currentOption] * 1024;
|
||||
if ($key === 'memory' && is_numeric($value) && $value < 1024) {
|
||||
$options[$key] = (int) ($value * 1024);
|
||||
} else {
|
||||
$options[$key] = $params['configoptions'][$currentOption];
|
||||
$options[$key] = is_numeric($value) ? (int) $value : $value;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -165,19 +160,18 @@ class ModuleFunctions extends Module
|
||||
|
||||
// If the server is created successfully, we can initialize the server build.
|
||||
$cs = new ConfigureService();
|
||||
$cs->initServerBuild($data->data->id, $params);
|
||||
$vfUserId = isset($data->data->owner->id) ? (int) $data->data->owner->id : null;
|
||||
$cs->initServerBuild($data->data->id, $params, $vfUserId);
|
||||
|
||||
/**
|
||||
*
|
||||
* Server was created successfully.
|
||||
*
|
||||
*/
|
||||
return 'success';
|
||||
} else {
|
||||
if ($data->errors[0]) {
|
||||
if (isset($data->errors) && is_array($data->errors) && isset($data->errors[0])) {
|
||||
return $data->errors[0];
|
||||
}
|
||||
return 'Unknown error.';
|
||||
if (isset($data->msg)) {
|
||||
return $data->msg;
|
||||
}
|
||||
return 'Server creation failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
Log::insert(__FUNCTION__, $params, $e->getMessage());
|
||||
@@ -185,13 +179,9 @@ class ModuleFunctions extends Module
|
||||
}
|
||||
}
|
||||
|
||||
// This function was implemented by Zander Scott / awildboop of Blinkoh, LLC
|
||||
// Please read this function thoroughly before use to ensure security & integrity
|
||||
|
||||
/**
|
||||
* Allows changing of the package of a server
|
||||
*
|
||||
* @author https://github.com/BlinkohHost/virtfusion-whmcs-module
|
||||
* @param $params
|
||||
* @return string
|
||||
*/
|
||||
@@ -201,9 +191,13 @@ class ModuleFunctions extends Module
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($params['serviceid']);
|
||||
if (!$whmcsService) return 'WHMCS service record not found.';
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return 'No control server found.';
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->put($cp['url'] . '/servers/' . $service->server_id . '/package/' . $params['configoption2']);
|
||||
$data = $request->put($cp['url'] . '/servers/' . (int) $service->server_id . '/package/' . (int) $params['configoption2']);
|
||||
$data = json_decode($data);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
@@ -211,18 +205,46 @@ class ModuleFunctions extends Module
|
||||
switch ($request->getRequestInfo('http_code')) {
|
||||
|
||||
case 204:
|
||||
return 'success';
|
||||
break;
|
||||
case 404:
|
||||
return '404 was returned from the web service without the msg property. The service may be currently unavailable.';
|
||||
return 'The server or package was not found in VirtFusion (HTTP 404).';
|
||||
case 423:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
return $data->msg;
|
||||
}
|
||||
return 'The server is currently locked. Please try again later.';
|
||||
default:
|
||||
return 'Update package request failed. The web service reported HTTP code ' . $request->getRequestInfo('http_code');
|
||||
return 'Update package request failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
|
||||
// Apply individual resource modifications from configurable options
|
||||
if (isset($params['configoptions']) && is_array($params['configoptions'])) {
|
||||
$configOptionDefaultNaming = [
|
||||
'memory' => 'Memory',
|
||||
'cpuCores' => 'CPU Cores',
|
||||
'traffic' => 'Bandwidth',
|
||||
];
|
||||
|
||||
$configOptionCustomNaming = [];
|
||||
if (file_exists(ROOTDIR . '/modules/servers/VirtFusionDirect/config/ConfigOptionMapping.php')) {
|
||||
$configOptionCustomNaming = require ROOTDIR . '/modules/servers/VirtFusionDirect/config/ConfigOptionMapping.php';
|
||||
}
|
||||
|
||||
foreach ($configOptionDefaultNaming as $resource => $optionName) {
|
||||
$currentOption = array_key_exists($resource, $configOptionCustomNaming) ? $configOptionCustomNaming[$resource] : $optionName;
|
||||
if (isset($params['configoptions'][$currentOption]) && is_numeric($params['configoptions'][$currentOption])) {
|
||||
$value = (int) $params['configoptions'][$currentOption];
|
||||
if ($resource === 'memory' && $value < 1024) {
|
||||
$value = $value * 1024;
|
||||
}
|
||||
$this->modifyResource($params['serviceid'], $resource, $value);
|
||||
}
|
||||
}
|
||||
return 'Service not found.';
|
||||
}
|
||||
|
||||
return 'success';
|
||||
}
|
||||
return 'Service not found in module database.';
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -241,11 +263,13 @@ class ModuleFunctions extends Module
|
||||
if ($service) {
|
||||
|
||||
$whmcsService = Database::getWhmcsService($params['serviceid']);
|
||||
if (!$whmcsService) return 'WHMCS service record not found.';
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return 'No control server found.';
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->delete($cp['url'] . '/servers/' . $service->server_id);
|
||||
$data = $request->delete($cp['url'] . '/servers/' . (int) $service->server_id);
|
||||
$data = json_decode($data);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
@@ -256,27 +280,24 @@ class ModuleFunctions extends Module
|
||||
Database::deleteSystemService($params['serviceid']);
|
||||
$this->updateWhmcsServiceParamsOnDestroy($params['serviceid']);
|
||||
return 'success';
|
||||
break;
|
||||
|
||||
case 404:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
if ($data->msg == 'server not found') {
|
||||
Database::deleteSystemService($params['serviceid']);
|
||||
return 'success';
|
||||
} else {
|
||||
return '404 was returned from the web service with the msg property but doesn\'t contain appropriate data to process a termination.';
|
||||
return 'VirtFusion returned 404: ' . $data->msg;
|
||||
}
|
||||
} else {
|
||||
return '404 was returned from the web service without the msg property. The service may be currently unavailable.';
|
||||
return 'VirtFusion returned 404 without details. The API may be unavailable.';
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
return 'Termination request failed. The web service reported HTTP code ' . $request->getRequestInfo('http_code');
|
||||
break;
|
||||
return 'Termination request failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
}
|
||||
return 'Service not found. Termination routine has already been run?';
|
||||
return 'Service not found in module database. Has termination already been run?';
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -295,10 +316,13 @@ class ModuleFunctions extends Module
|
||||
if ($service) {
|
||||
|
||||
$whmcsService = Database::getWhmcsService($params['serviceid']);
|
||||
if (!$whmcsService) return 'WHMCS service record not found.';
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return 'No control server found.';
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/servers/' . $service->server_id . '/suspend');
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/suspend');
|
||||
$data = json_decode($data);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
@@ -307,32 +331,29 @@ class ModuleFunctions extends Module
|
||||
|
||||
case 204:
|
||||
return 'success';
|
||||
break;
|
||||
|
||||
case 404:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
if ($data->msg == 'server not found') {
|
||||
Database::deleteSystemService($params['serviceid']);
|
||||
return 'success';
|
||||
} else {
|
||||
|
||||
return '404 was returned from the web service with the msg property but doesn\'t contain appropriate data to process a suspension.';
|
||||
return 'VirtFusion returned 404: ' . $data->msg;
|
||||
}
|
||||
} else {
|
||||
return '404 was returned from the web service without the msg property. The service may be currently unavailable.';
|
||||
return 'VirtFusion returned 404 without details. The API may be unavailable.';
|
||||
}
|
||||
break;
|
||||
case 423:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
return $data->msg;
|
||||
}
|
||||
return 'The server is currently locked. Please try again later.';
|
||||
|
||||
default:
|
||||
return 'Suspend request failed. The web service reported HTTP code ' . $request->getRequestInfo('http_code');
|
||||
break;
|
||||
return 'Suspend request failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
}
|
||||
return 'Service not found.';
|
||||
return 'Service not found in module database.';
|
||||
}
|
||||
|
||||
function updateServerObject($params)
|
||||
@@ -342,10 +363,13 @@ class ModuleFunctions extends Module
|
||||
if ($service) {
|
||||
|
||||
$whmcsService = Database::getWhmcsService($params['serviceid']);
|
||||
if (!$whmcsService) return 'WHMCS service record not found.';
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return 'No control server found.';
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/servers/' . $service->server_id);
|
||||
$data = $request->get($cp['url'] . '/servers/' . (int) $service->server_id);
|
||||
$data = json_decode($data);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
@@ -358,13 +382,11 @@ class ModuleFunctions extends Module
|
||||
$this->updateWhmcsServiceParamsOnServerObject($params['serviceid'], $data);
|
||||
|
||||
return 'success';
|
||||
break;
|
||||
default:
|
||||
return 'Request failed. The web service reported HTTP code ' . $request->getRequestInfo('http_code');
|
||||
break;
|
||||
return 'Request failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
}
|
||||
return 'Service not found.';
|
||||
return 'Service not found in module database.';
|
||||
}
|
||||
|
||||
|
||||
@@ -374,10 +396,13 @@ class ModuleFunctions extends Module
|
||||
|
||||
if ($service) {
|
||||
$whmcsService = Database::getWhmcsService($params['serviceid']);
|
||||
if (!$whmcsService) return 'WHMCS service record not found.';
|
||||
|
||||
$cp = $this->getCP($whmcsService->server);
|
||||
if (!$cp) return 'No control server found.';
|
||||
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->post($cp['url'] . '/servers/' . $service->server_id . '/unsuspend');
|
||||
$data = $request->post($cp['url'] . '/servers/' . (int) $service->server_id . '/unsuspend');
|
||||
$data = json_decode($data);
|
||||
|
||||
Log::insert(__FUNCTION__, $request->getRequestInfo(), $data);
|
||||
@@ -386,31 +411,29 @@ class ModuleFunctions extends Module
|
||||
|
||||
case 204:
|
||||
return 'success';
|
||||
break;
|
||||
|
||||
case 404:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
if ($data->msg == 'server not found') {
|
||||
Database::deleteSystemService($params['serviceid']);
|
||||
return 'success';
|
||||
} else {
|
||||
return '404 was returned from the web service with the msg property but doesn\'t contain appropriate data to process an unsuspension.';
|
||||
return 'VirtFusion returned 404: ' . $data->msg;
|
||||
}
|
||||
} else {
|
||||
return '404 was returned from the web service without the msg property. The service may be currently unavailable.';
|
||||
return 'VirtFusion returned 404 without details. The API may be unavailable.';
|
||||
}
|
||||
break;
|
||||
case 423:
|
||||
if (property_exists($data, 'msg')) {
|
||||
if (isset($data->msg)) {
|
||||
return $data->msg;
|
||||
}
|
||||
return 'The server is currently locked. Please try again later.';
|
||||
|
||||
default:
|
||||
return 'Unsuspend request failed. The web service reported HTTP code ' . $request->getRequestInfo('http_code');
|
||||
break;
|
||||
return 'Unsuspend request failed. VirtFusion API returned HTTP ' . $request->getRequestInfo('http_code');
|
||||
}
|
||||
}
|
||||
return 'Service not found';
|
||||
return 'Service not found in module database.';
|
||||
}
|
||||
|
||||
public function adminServicesTabFields($params)
|
||||
@@ -440,12 +463,57 @@ class ModuleFunctions extends Module
|
||||
|
||||
public function adminServicesTabFieldsSave($params)
|
||||
{
|
||||
|
||||
if ($_POST['modulefields'][0] == '') {
|
||||
if (!isset($_POST['modulefields'][0]) || $_POST['modulefields'][0] === '') {
|
||||
Database::deleteSystemService($params['serviceid']);
|
||||
} else {
|
||||
$serverId = (int) $_POST['modulefields'][0];
|
||||
if ($serverId > 0) {
|
||||
Database::updateSystemServiceServerId($params['serviceid'], $serverId);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Database::updateSystemServiceServerId($params['serviceid'], $_POST['modulefields'][0]);
|
||||
/**
|
||||
* Validate server creation parameters via dry run.
|
||||
*
|
||||
* @param array $params WHMCS service params
|
||||
* @return string 'success' or error message
|
||||
*/
|
||||
public function validateServerConfig($params)
|
||||
{
|
||||
try {
|
||||
$server = $params['serverid'] ?: false;
|
||||
$cp = $this->getCP($server, !$server);
|
||||
|
||||
if (!$cp) {
|
||||
return 'No Control server found.';
|
||||
}
|
||||
|
||||
$options = [
|
||||
"packageId" => (int) $params['configoption2'],
|
||||
"hypervisorId" => (int) $params['configoption1'],
|
||||
"ipv4" => (int) $params['configoption3'],
|
||||
];
|
||||
|
||||
// We need a userId for dry run - use the service owner
|
||||
if (isset($params['userid'])) {
|
||||
$request = $this->initCurl($cp['token']);
|
||||
$data = $request->get($cp['url'] . '/users/' . (int) $params['userid'] . '/byExtRelation');
|
||||
if ($request->getRequestInfo('http_code') == 200) {
|
||||
$userData = json_decode($data);
|
||||
$options['userId'] = $userData->data->id;
|
||||
}
|
||||
}
|
||||
|
||||
$result = $this->validateServerCreation($options, $params['serverid']);
|
||||
|
||||
if ($result['valid']) {
|
||||
return 'success';
|
||||
}
|
||||
|
||||
return 'Validation failed: ' . implode(', ', $result['errors']);
|
||||
} catch (\Exception $e) {
|
||||
return 'Validation error: ' . $e->getMessage();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -463,6 +531,7 @@ class ModuleFunctions extends Module
|
||||
'systemURL' => Database::getSystemUrl(),
|
||||
'serviceStatus' => $params['status'],
|
||||
'serverHostname' => $serverHostname,
|
||||
'selfServiceMode' => (int) ($params['configoption4'] ?? 0),
|
||||
],
|
||||
];
|
||||
} catch (\Throwable $e) {
|
||||
|
||||
@@ -8,41 +8,72 @@ class ServerResource
|
||||
{
|
||||
$server = json_decode(json_encode($data->data), true);
|
||||
|
||||
$traffic = '∞';
|
||||
$traffic = '-';
|
||||
|
||||
if ($server['settings']['resources']['traffic']) {
|
||||
if (isset($server['settings']['resources']['traffic'])) {
|
||||
if ($server['settings']['resources']['traffic'] > 0) {
|
||||
$traffic = $server['settings']['resources']['traffic'] . ' GB';
|
||||
} else {
|
||||
$traffic = 'Unlimited';
|
||||
}
|
||||
}
|
||||
|
||||
$trafficUsed = '-';
|
||||
if (isset($server['usage']['traffic']['used'])) {
|
||||
$trafficUsed = round($server['usage']['traffic']['used'] / 1073741824, 2) . ' GB';
|
||||
} elseif (isset($server['settings']['resources']['traffic']) && $server['settings']['resources']['traffic'] > 0) {
|
||||
$trafficUsed = '0 GB';
|
||||
}
|
||||
|
||||
$data = [
|
||||
'name' => $server['name'] ?: '-',
|
||||
'hostname' => $server['hostname'] ?: '-',
|
||||
'memory' => $server['settings']['resources']['memory'] . ' MB',
|
||||
'memory' => isset($server['settings']['resources']['memory']) ? $server['settings']['resources']['memory'] . ' MB' : '-',
|
||||
'traffic' => $traffic,
|
||||
'storage' => $server['settings']['resources']['storage'] . ' GB',
|
||||
'cpu' => $server['settings']['resources']['cpuCores'] . ' Core(s)',
|
||||
'trafficUsed' => $trafficUsed,
|
||||
'storage' => isset($server['settings']['resources']['storage']) ? $server['settings']['resources']['storage'] . ' GB' : '-',
|
||||
'cpu' => isset($server['settings']['resources']['cpuCores']) ? $server['settings']['resources']['cpuCores'] . ' Core(s)' : '-',
|
||||
'status' => isset($server['state']) ? $server['state'] : 'unknown',
|
||||
'powerStatus' => isset($server['hypervisor']['settings']['state']) ? $server['hypervisor']['settings']['state'] : 'unknown',
|
||||
'username' => isset($server['owner']['email']) ? $server['owner']['email'] : '',
|
||||
'password' => '',
|
||||
'primaryNetwork' => [
|
||||
'ipv4' => ['-'],
|
||||
'ipv4Unformatted' => [],
|
||||
'ipv6' => ['-'],
|
||||
'ipv6Unformatted' => [],
|
||||
]
|
||||
'mac' => '-',
|
||||
],
|
||||
'networkSpeed' => [
|
||||
'inbound' => isset($server['settings']['resources']['networkSpeedInbound']) ? $server['settings']['resources']['networkSpeedInbound'] . ' Mbps' : '-',
|
||||
'outbound' => isset($server['settings']['resources']['networkSpeedOutbound']) ? $server['settings']['resources']['networkSpeedOutbound'] . ' Mbps' : '-',
|
||||
],
|
||||
'vncEnabled' => isset($server['vnc']['enabled']) ? (bool) $server['vnc']['enabled'] : false,
|
||||
'memoryRaw' => isset($server['settings']['resources']['memory']) ? (int) $server['settings']['resources']['memory'] : 0,
|
||||
'cpuRaw' => isset($server['settings']['resources']['cpuCores']) ? (int) $server['settings']['resources']['cpuCores'] : 0,
|
||||
'storageRaw' => isset($server['settings']['resources']['storage']) ? (int) $server['settings']['resources']['storage'] : 0,
|
||||
'trafficRaw' => isset($server['settings']['resources']['traffic']) ? (int) $server['settings']['resources']['traffic'] : 0,
|
||||
'trafficUsedRaw' => isset($server['usage']['traffic']['used']) ? round($server['usage']['traffic']['used'] / 1073741824, 2) : 0,
|
||||
'networkSpeedInboundRaw' => isset($server['settings']['resources']['networkSpeedInbound']) ? (int) $server['settings']['resources']['networkSpeedInbound'] : 0,
|
||||
'networkSpeedOutboundRaw' => isset($server['settings']['resources']['networkSpeedOutbound']) ? (int) $server['settings']['resources']['networkSpeedOutbound'] : 0,
|
||||
];
|
||||
|
||||
if (array_key_exists('network', $server)) {
|
||||
if (array_key_exists('interfaces', $server['network'])) {
|
||||
if (count($server['network']['interfaces'])) {
|
||||
|
||||
if (count($server['network']['interfaces'][0]['ipv4'])) {
|
||||
if (isset($server['network']['interfaces'][0]['mac'])) {
|
||||
$data['primaryNetwork']['mac'] = $server['network']['interfaces'][0]['mac'];
|
||||
}
|
||||
|
||||
if (isset($server['network']['interfaces'][0]['ipv4']) && count($server['network']['interfaces'][0]['ipv4'])) {
|
||||
$data['primaryNetwork']['ipv4'] = [];
|
||||
foreach ($server['network']['interfaces'][0]['ipv4'] as $ip) {
|
||||
$data['primaryNetwork']['ipv4'][] = $ip['address'];
|
||||
}
|
||||
}
|
||||
|
||||
if (count($server['network']['interfaces'][0]['ipv6'])) {
|
||||
if (isset($server['network']['interfaces'][0]['ipv6']) && count($server['network']['interfaces'][0]['ipv6'])) {
|
||||
$data['primaryNetwork']['ipv6'] = [];
|
||||
foreach ($server['network']['interfaces'][0]['ipv6'] as $ip) {
|
||||
$data['primaryNetwork']['ipv6'][] = $ip['subnet'] . '/' . $ip['cidr'];
|
||||
|
||||
@@ -1 +1,200 @@
|
||||
.vf-bold{font-weight:800}.vf-small{font-size:.9rem}.vf-button{font-size:.8rem;padding:.95rem 1.5rem;font-weight:600}.vf-button-small{font-size:.8rem;padding:.75rem 1.3rem;font-weight:500}.vf-spinner-margin{margin-right:7px}.vf-badge{font-size:.8rem;padding:.5rem .9rem;text-transform:uppercase;font-weight:800}.vf-badge-active{background-color:rgba(32,177,0,.12);color:#276900;border-radius:6px}.vf-badge-awaiting{background-color:rgba(177,89,0,.12);color:#692000;border-radius:6px}#vf-login-button-spinner{display:none}#vf-password-reset-button-spinner{display:none}#vf-password-reset-error{display:none}#vf-password-reset-success{display:none}#vf-login-error{display:none}#vf-server-info{display:none}#vf-server-info-error{display:none}#vf-server-info-loader{min-height:136px}#vf-loading{display:inline-block;width:30px;height:30px;border:3px solid rgba(225,224,224,.3);border-radius:50%;border-top-color:#0e151a;animation:vf-spin 1s ease-in-out infinite;-webkit-animation:vf-spin 1s ease-in-out infinite}.vf-loader{margin:30px}@keyframes vf-spin{to{transform:rotate(360deg)}}@-webkit-keyframes vf-spin{to{transform:rotate(360deg)}}#vf-server-info-error{margin:10px}
|
||||
/* VirtFusion Direct Provisioning Module Styles */
|
||||
|
||||
/* Typography */
|
||||
.vf-bold {
|
||||
font-weight: 800;
|
||||
}
|
||||
.vf-small {
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
/* Buttons */
|
||||
.vf-button {
|
||||
font-size: 0.8rem;
|
||||
padding: 0.95rem 1.5rem;
|
||||
font-weight: 600;
|
||||
}
|
||||
.vf-button-small {
|
||||
font-size: 0.8rem;
|
||||
padding: 0.75rem 1.3rem;
|
||||
font-weight: 500;
|
||||
}
|
||||
.vf-spinner-margin {
|
||||
margin-right: 7px;
|
||||
}
|
||||
|
||||
/* Status Badges */
|
||||
.vf-badge {
|
||||
font-size: 0.75rem;
|
||||
padding: 0.35rem 0.75rem;
|
||||
text-transform: uppercase;
|
||||
font-weight: 700;
|
||||
border-radius: 6px;
|
||||
display: inline-block;
|
||||
}
|
||||
.vf-badge-active {
|
||||
background-color: rgba(32, 177, 0, 0.12);
|
||||
color: #276900;
|
||||
}
|
||||
.vf-badge-awaiting {
|
||||
background-color: rgba(177, 89, 0, 0.12);
|
||||
color: #692000;
|
||||
}
|
||||
.vf-badge-suspended {
|
||||
background-color: rgba(220, 53, 69, 0.12);
|
||||
color: #721c24;
|
||||
}
|
||||
|
||||
/* Power Management */
|
||||
.vf-power-buttons {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 8px;
|
||||
}
|
||||
.vf-btn-power {
|
||||
min-width: 100px;
|
||||
font-weight: 600;
|
||||
text-transform: uppercase;
|
||||
font-size: 0.8rem;
|
||||
padding: 0.5rem 1rem;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
/* Hidden elements (initial state) */
|
||||
#vf-login-button-spinner {
|
||||
display: none;
|
||||
}
|
||||
#vf-password-reset-button-spinner {
|
||||
display: none;
|
||||
}
|
||||
#vf-password-reset-error {
|
||||
display: none;
|
||||
}
|
||||
#vf-password-reset-success {
|
||||
display: none;
|
||||
}
|
||||
#vf-login-error {
|
||||
display: none;
|
||||
}
|
||||
#vf-server-info {
|
||||
display: none;
|
||||
}
|
||||
#vf-server-info-error {
|
||||
display: none;
|
||||
}
|
||||
#vf-data-server-traffic-sep {
|
||||
display: inline;
|
||||
}
|
||||
|
||||
/* Loader */
|
||||
#vf-server-info-loader {
|
||||
min-height: 136px;
|
||||
}
|
||||
#vf-loading {
|
||||
display: inline-block;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
border: 3px solid rgba(225, 224, 224, 0.3);
|
||||
border-radius: 50%;
|
||||
border-top-color: #0e151a;
|
||||
animation: vf-spin 1s ease-in-out infinite;
|
||||
-webkit-animation: vf-spin 1s ease-in-out infinite;
|
||||
}
|
||||
.vf-loader {
|
||||
margin: 30px;
|
||||
}
|
||||
|
||||
@keyframes vf-spin {
|
||||
to {
|
||||
transform: rotate(360deg);
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes vf-spin {
|
||||
to {
|
||||
transform: rotate(360deg);
|
||||
}
|
||||
}
|
||||
|
||||
/* Error message spacing */
|
||||
#vf-server-info-error {
|
||||
margin: 10px;
|
||||
}
|
||||
|
||||
/* Network / IP Management */
|
||||
.vf-ip-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 4px 0;
|
||||
}
|
||||
.vf-ip-address {
|
||||
font-family: monospace;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
.vf-ip-remove {
|
||||
font-size: 0.7rem;
|
||||
padding: 0.15rem 0.4rem;
|
||||
}
|
||||
|
||||
/* Resource panel */
|
||||
.vf-resource-item .progress {
|
||||
background-color: rgba(0,0,0,0.08);
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
/* Order form slider UI */
|
||||
.vf-slider-container {
|
||||
padding: 8px 0;
|
||||
}
|
||||
.vf-slider-value {
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
margin-bottom: 4px;
|
||||
text-align: center;
|
||||
}
|
||||
.vf-slider {
|
||||
width: 100%;
|
||||
height: 6px;
|
||||
-webkit-appearance: none;
|
||||
appearance: none;
|
||||
background: #ddd;
|
||||
border-radius: 3px;
|
||||
outline: none;
|
||||
cursor: pointer;
|
||||
}
|
||||
.vf-slider::-webkit-slider-thumb {
|
||||
-webkit-appearance: none;
|
||||
appearance: none;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
border-radius: 50%;
|
||||
background: #337ab7;
|
||||
cursor: pointer;
|
||||
border: 2px solid #fff;
|
||||
box-shadow: 0 1px 3px rgba(0,0,0,0.2);
|
||||
}
|
||||
.vf-slider::-moz-range-thumb {
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
border-radius: 50%;
|
||||
background: #337ab7;
|
||||
cursor: pointer;
|
||||
border: 2px solid #fff;
|
||||
box-shadow: 0 1px 3px rgba(0,0,0,0.2);
|
||||
}
|
||||
|
||||
/* Responsive adjustments */
|
||||
@media (max-width: 768px) {
|
||||
.vf-power-buttons {
|
||||
flex-direction: column;
|
||||
}
|
||||
.vf-btn-power {
|
||||
width: 100%;
|
||||
}
|
||||
.vf-ip-row {
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1 +1,11 @@
|
||||
<div class="alert alert-danger"><p>Oops! Something went wrong.</p></div><p>Please go back and try again.</p><p>If the problem persists, please contact support.</p>
|
||||
<div class="panel card panel-default mb-3">
|
||||
<div class="panel-heading card-header">
|
||||
<h3 class="panel-title card-title m-0">Error</h3>
|
||||
</div>
|
||||
<div class="panel-body card-body p-4">
|
||||
<div class="alert alert-danger mb-0">
|
||||
<p><strong>Something went wrong.</strong></p>
|
||||
<p class="mb-0">Please go back and try again. If the problem persists, please contact support.</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
124
modules/servers/VirtFusionDirect/templates/js/keygen.js
Normal file
124
modules/servers/VirtFusionDirect/templates/js/keygen.js
Normal file
@@ -0,0 +1,124 @@
|
||||
/**
|
||||
* VirtFusion SSH Ed25519 Key Generator
|
||||
*
|
||||
* Client-side Ed25519 keypair generation using Web Crypto API.
|
||||
* Produces OpenSSH-format public and private keys.
|
||||
*/
|
||||
|
||||
function vfConcatArrays() {
|
||||
var total = 0;
|
||||
for (var i = 0; i < arguments.length; i++) total += arguments[i].length;
|
||||
var result = new Uint8Array(total);
|
||||
var offset = 0;
|
||||
for (var i = 0; i < arguments.length; i++) {
|
||||
result.set(arguments[i], offset);
|
||||
offset += arguments[i].length;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
function vfSshEncodeUint32(value) {
|
||||
return new Uint8Array([
|
||||
(value >>> 24) & 0xff,
|
||||
(value >>> 16) & 0xff,
|
||||
(value >>> 8) & 0xff,
|
||||
value & 0xff
|
||||
]);
|
||||
}
|
||||
|
||||
function vfSshEncodeString(data) {
|
||||
if (typeof data === 'string') {
|
||||
data = new TextEncoder().encode(data);
|
||||
}
|
||||
return vfConcatArrays(vfSshEncodeUint32(data.length), data);
|
||||
}
|
||||
|
||||
function vfArrayToBase64(uint8Array) {
|
||||
var binary = '';
|
||||
for (var i = 0; i < uint8Array.length; i++) {
|
||||
binary += String.fromCharCode(uint8Array[i]);
|
||||
}
|
||||
return btoa(binary);
|
||||
}
|
||||
|
||||
function vfEncodeSSHPublicKey(pubKeyBytes) {
|
||||
var blob = vfConcatArrays(
|
||||
vfSshEncodeString('ssh-ed25519'),
|
||||
vfSshEncodeString(pubKeyBytes)
|
||||
);
|
||||
return 'ssh-ed25519 ' + vfArrayToBase64(blob);
|
||||
}
|
||||
|
||||
function vfEncodeSSHPrivateKey(seed, pubKeyBytes) {
|
||||
var keyType = vfSshEncodeString('ssh-ed25519');
|
||||
var pubBlob = vfConcatArrays(keyType, vfSshEncodeString(pubKeyBytes));
|
||||
|
||||
var checkInt = crypto.getRandomValues(new Uint8Array(4));
|
||||
var privKey = vfConcatArrays(seed, pubKeyBytes); // 64 bytes: seed || pubkey
|
||||
|
||||
var privateSection = vfConcatArrays(
|
||||
checkInt,
|
||||
checkInt,
|
||||
vfSshEncodeString('ssh-ed25519'),
|
||||
vfSshEncodeString(pubKeyBytes),
|
||||
vfSshEncodeString(privKey),
|
||||
vfSshEncodeString('') // empty comment
|
||||
);
|
||||
|
||||
// Pad to 8-byte boundary with 1,2,3,4,5...
|
||||
var padLen = 8 - (privateSection.length % 8);
|
||||
if (padLen < 8) {
|
||||
var padding = new Uint8Array(padLen);
|
||||
for (var i = 0; i < padLen; i++) padding[i] = i + 1;
|
||||
privateSection = vfConcatArrays(privateSection, padding);
|
||||
}
|
||||
|
||||
var authMagic = new TextEncoder().encode('openssh-key-v1\0');
|
||||
var body = vfConcatArrays(
|
||||
authMagic,
|
||||
vfSshEncodeString('none'), // cipher
|
||||
vfSshEncodeString('none'), // kdf
|
||||
vfSshEncodeString(''), // kdf options
|
||||
vfSshEncodeUint32(1), // number of keys
|
||||
vfSshEncodeString(pubBlob),
|
||||
vfSshEncodeString(privateSection)
|
||||
);
|
||||
|
||||
var b64 = vfArrayToBase64(body);
|
||||
var lines = ['-----BEGIN OPENSSH PRIVATE KEY-----'];
|
||||
for (var i = 0; i < b64.length; i += 70) {
|
||||
lines.push(b64.substring(i, i + 70));
|
||||
}
|
||||
lines.push('-----END OPENSSH PRIVATE KEY-----');
|
||||
lines.push('');
|
||||
return lines.join('\n');
|
||||
}
|
||||
|
||||
async function vfGenerateSSHKey() {
|
||||
var keyPair = await crypto.subtle.generateKey('Ed25519', true, ['sign', 'verify']);
|
||||
|
||||
var pubRaw = await crypto.subtle.exportKey('raw', keyPair.publicKey);
|
||||
var pubKeyBytes = new Uint8Array(pubRaw);
|
||||
|
||||
// PKCS#8 for Ed25519 is exactly 48 bytes; bytes 16-47 are the 32-byte seed
|
||||
var privPkcs8 = await crypto.subtle.exportKey('pkcs8', keyPair.privateKey);
|
||||
var privBytes = new Uint8Array(privPkcs8);
|
||||
var seed = privBytes.slice(16, 48);
|
||||
|
||||
return {
|
||||
publicKey: vfEncodeSSHPublicKey(pubKeyBytes),
|
||||
privateKey: vfEncodeSSHPrivateKey(seed, pubKeyBytes)
|
||||
};
|
||||
}
|
||||
|
||||
function vfDownloadFile(filename, content) {
|
||||
var blob = new Blob([content], { type: 'application/octet-stream' });
|
||||
var url = URL.createObjectURL(blob);
|
||||
var a = document.createElement('a');
|
||||
a.href = url;
|
||||
a.download = filename;
|
||||
document.body.appendChild(a);
|
||||
a.click();
|
||||
document.body.removeChild(a);
|
||||
URL.revokeObjectURL(url);
|
||||
}
|
||||
@@ -1 +1,551 @@
|
||||
function vfServerData(e,r){$("#vf-server-info-error").hide(),$.ajax({type:"GET",dataType:"json",url:r+"modules/servers/VirtFusionDirect/client.php?serviceID="+e+"&action=serverData"}).done(function(e){e.success?($("#vf-data-server-name").text(e.data.name),$("#vf-data-server-hostname").text(e.data.hostname),$("#vf-data-server-memory").text(e.data.memory),$("#vf-data-server-traffic").text(e.data.traffic),$("#vf-data-server-storage").text(e.data.storage),$("#vf-data-server-cpu").text(e.data.cpu),$("#vf-data-server-ipv4").text(e.data.primaryNetwork.ipv4),$("#vf-data-server-ipv6").text(e.data.primaryNetwork.ipv6),$("#vf-server-info").show()):($("#vf-server-info-error").show(),$("#vf-server-info").hide())}).fail(function(e){}).always(function(e){$("#vf-server-info-loader-container").hide()})}function vfServerDataAdmin(e,r){$("#vf-loader").show(),$("#vf-server-info").hide(),$("#vf-server-info-error").hide(),$.ajax({type:"GET",dataType:"json",url:r+"modules/servers/VirtFusionDirect/admin.php?serviceID="+e+"&action=serverData"}).done(function(e){e.success?($("#vf-data-server-name").text(e.data.name),$("#vf-data-server-hostname").text(e.data.hostname),$("#vf-data-server-memory").text(e.data.memory),$("#vf-data-server-traffic").text(e.data.traffic),$("#vf-data-server-storage").text(e.data.storage),$("#vf-data-server-cpu").text(e.data.cpu),$("#vf-data-server-ipv4").text(e.data.primaryNetwork.ipv4),$("#vf-data-server-ipv6").text(e.data.primaryNetwork.ipv6),$("#vf-server-info").show()):($("#vf-server-info-error").show(),$("#vf-server-info-error-message").text(e.errors),$("#vf-server-info").hide())}).fail(function(e){}).always(function(e){$("#vf-loader").hide()})}function vfUserPasswordReset(e,r){$("#vf-password-reset-button-spinner").show(),$("#vf-password-reset-error").hide(),$("#vf-password-reset-success").hide(),$.ajax({type:"GET",dataType:"json",url:r+"modules/servers/VirtFusionDirect/client.php?serviceID="+e+"&action=resetPassword"}).done(function(e){e.success?($("#vf-password-reset-success").show(),$("#vf-data-user-email").text(e.data.email),$("#vf-data-user-password").text(e.data.password),console.log(e.data.email)):$("#vf-password-reset-error").show()}).fail(function(e){}).always(function(e){$("#vf-password-reset-button-spinner").hide()})}function vfLoginAsServerOwner(e,r,t=!0){vfLoginError(!1),$("#vf-login-button").prop("disabled",!0),$("#vf-login-button-spinner").show(),$.ajax({type:"GET",dataType:"json",url:r+"modules/servers/VirtFusionDirect/client.php?serviceID="+e+"&action=loginAsServerOwner"}).done(function(e){e.success?e.token_url&&(t?window.open(e.token_url):window.location.href=e.token_url):vfLoginError(!0)}).fail(function(e){vfLoginError(!0)}).always(function(e){$("#vf-login-button-spinner").hide(),$("#vf-login-button").prop("disabled",!1)})}function vfLoginError(e,r="Oops! Something went wrong. Try again later."){e?($("#vf-login-error").text(r),$("#vf-login-error").show()):$("#vf-login-error").hide()}function impersonateServerOwner(e,r){$.ajax({type:"GET",dataType:"json",url:r+"modules/servers/VirtFusionDirect/admin.php?serviceID="+e+"&action=impersonateServerOwner"}).done(function(e){e.success&&e.user&&window.open(e.url+"/_imp/in/"+e.user.id+"/-")}).fail(function(e){}).always(function(e){})}
|
||||
/**
|
||||
* VirtFusion Direct Provisioning Module - Client JavaScript
|
||||
*
|
||||
* Handles client-side interactions for server management including:
|
||||
* - Server data display
|
||||
* - Power management (boot, shutdown, restart, power off)
|
||||
* - Control panel login (SSO)
|
||||
* - Password reset
|
||||
* - Server rebuild
|
||||
* - OS template loading
|
||||
*/
|
||||
|
||||
function vfServerData(serviceId, systemUrl) {
|
||||
$("#vf-server-info-error").hide();
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=serverData"
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
$("#vf-data-server-name").text(response.data.name);
|
||||
$("#vf-data-server-hostname").text(response.data.hostname);
|
||||
$("#vf-data-server-memory").text(response.data.memory);
|
||||
$("#vf-data-server-traffic").text(response.data.traffic);
|
||||
$("#vf-data-server-traffic-used").text(response.data.trafficUsed || "-");
|
||||
$("#vf-data-server-storage").text(response.data.storage);
|
||||
$("#vf-data-server-cpu").text(response.data.cpu);
|
||||
var pn = response.data.primaryNetwork || {};
|
||||
$("#vf-data-server-ipv4").text(pn.ipv4 || "-");
|
||||
$("#vf-data-server-ipv6").text(pn.ipv6 || "-");
|
||||
|
||||
// Update status badge
|
||||
var statusBadge = $("#vf-status-badge");
|
||||
var status = (response.data.status || "unknown").toLowerCase();
|
||||
statusBadge.text(status.charAt(0).toUpperCase() + status.slice(1));
|
||||
statusBadge.removeClass("vf-badge-active vf-badge-suspended vf-badge-awaiting");
|
||||
if (status === "active" || status === "running") {
|
||||
statusBadge.addClass("vf-badge-active");
|
||||
} else if (status === "suspended") {
|
||||
statusBadge.addClass("vf-badge-suspended");
|
||||
} else {
|
||||
statusBadge.addClass("vf-badge-awaiting");
|
||||
}
|
||||
|
||||
// Show/hide VNC panel based on API response
|
||||
if (response.data.vncEnabled) {
|
||||
$("#vf-vnc-panel").show();
|
||||
}
|
||||
|
||||
// Populate resources panel
|
||||
var d = response.data;
|
||||
$("#vf-res-memory").text(d.memory || "-");
|
||||
$("#vf-res-cpu").text(d.cpu || "-");
|
||||
$("#vf-res-storage").text(d.storage || "-");
|
||||
|
||||
var trafficUsed = d.trafficUsedRaw || 0;
|
||||
var trafficTotal = d.trafficRaw || 0;
|
||||
if (trafficTotal > 0) {
|
||||
$("#vf-res-traffic").text(trafficUsed + " / " + trafficTotal + " GB");
|
||||
var pct = Math.min(100, Math.round((trafficUsed / trafficTotal) * 100));
|
||||
$("#vf-res-traffic-bar").css("width", pct + "%").removeClass("bg-danger bg-warning");
|
||||
if (pct > 90) {
|
||||
$("#vf-res-traffic-bar").addClass("bg-danger");
|
||||
} else if (pct > 70) {
|
||||
$("#vf-res-traffic-bar").addClass("bg-warning");
|
||||
}
|
||||
} else {
|
||||
$("#vf-res-traffic").text(d.traffic || "Unlimited");
|
||||
$("#vf-res-traffic-bar").css("width", "0%");
|
||||
}
|
||||
|
||||
var speedIn = d.networkSpeedInboundRaw || 0;
|
||||
var speedOut = d.networkSpeedOutboundRaw || 0;
|
||||
if (speedIn > 0 || speedOut > 0) {
|
||||
$("#vf-res-network-speed").text(speedIn + " / " + speedOut + " Mbps");
|
||||
} else {
|
||||
$("#vf-res-network-speed").text("-");
|
||||
}
|
||||
|
||||
$("#vf-resources-panel").show();
|
||||
|
||||
// Populate network panel from server data
|
||||
var ipv4List = $("#vf-ipv4-list");
|
||||
var ipv6List = $("#vf-ipv6-list");
|
||||
ipv4List.empty();
|
||||
ipv6List.empty();
|
||||
|
||||
var net = response.data.primaryNetwork || {};
|
||||
var ipv4Arr = net.ipv4Unformatted || [];
|
||||
var ipv6Arr = net.ipv6Unformatted || [];
|
||||
|
||||
if (ipv4Arr.length > 0) {
|
||||
$.each(ipv4Arr, function (i, ip) {
|
||||
var row = $('<div class="vf-ip-row"></div>');
|
||||
row.append('<span class="vf-ip-address">' + $('<span>').text(ip).html() + '</span>');
|
||||
if (i > 0) {
|
||||
row.append(' <button class="btn btn-sm btn-outline-danger vf-ip-remove" onclick="vfRemoveIP(\'' + serviceId + '\',\'' + systemUrl + '\',\'removeIPv4\',\'' + encodeURIComponent(ip) + '\')">Remove</button>');
|
||||
}
|
||||
ipv4List.append(row);
|
||||
});
|
||||
} else {
|
||||
ipv4List.append('<span class="text-muted">No IPv4 addresses</span>');
|
||||
}
|
||||
|
||||
if (ipv6Arr.length > 0) {
|
||||
$.each(ipv6Arr, function (i, subnet) {
|
||||
var row = $('<div class="vf-ip-row"></div>');
|
||||
row.append('<span class="vf-ip-address">' + $('<span>').text(subnet).html() + '</span>');
|
||||
ipv6List.append(row);
|
||||
});
|
||||
} else {
|
||||
ipv6List.append('<span class="text-muted">No IPv6 subnets</span>');
|
||||
}
|
||||
|
||||
$("#vf-network-content").show();
|
||||
|
||||
$("#vf-server-info").show();
|
||||
} else {
|
||||
$("#vf-server-info-error").show();
|
||||
$("#vf-server-info").hide();
|
||||
}
|
||||
}).fail(function () {
|
||||
$("#vf-server-info-error").show();
|
||||
}).always(function () {
|
||||
$("#vf-server-info-loader-container").hide();
|
||||
});
|
||||
}
|
||||
|
||||
function vfServerDataAdmin(serviceId, systemUrl) {
|
||||
$("#vf-loader").show();
|
||||
$("#vf-server-info").hide();
|
||||
$("#vf-server-info-error").hide();
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/admin.php?serviceID=" + encodeURIComponent(serviceId) + "&action=serverData"
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
$("#vf-data-server-name").text(response.data.name);
|
||||
$("#vf-data-server-hostname").text(response.data.hostname);
|
||||
$("#vf-data-server-memory").text(response.data.memory);
|
||||
$("#vf-data-server-traffic").text(response.data.traffic);
|
||||
$("#vf-data-server-storage").text(response.data.storage);
|
||||
$("#vf-data-server-cpu").text(response.data.cpu);
|
||||
var pnAdmin = response.data.primaryNetwork || {};
|
||||
$("#vf-data-server-ipv4").text(pnAdmin.ipv4 || "-");
|
||||
$("#vf-data-server-ipv6").text(pnAdmin.ipv6 || "-");
|
||||
$("#vf-server-info").show();
|
||||
} else {
|
||||
$("#vf-server-info-error").show();
|
||||
$("#vf-server-info-error-message").text(response.errors);
|
||||
$("#vf-server-info").hide();
|
||||
}
|
||||
}).fail(function () {
|
||||
$("#vf-server-info-error").show();
|
||||
}).always(function () {
|
||||
$("#vf-loader").hide();
|
||||
});
|
||||
}
|
||||
|
||||
function vfUserPasswordReset(serviceId, systemUrl) {
|
||||
$("#vf-password-reset-button-spinner").show();
|
||||
$("#vf-password-reset-error").hide();
|
||||
$("#vf-password-reset-success").hide();
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=resetPassword"
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
$("#vf-password-reset-success").show();
|
||||
$("#vf-data-user-email").text(response.data.email);
|
||||
$("#vf-data-user-password").text(response.data.password);
|
||||
} else {
|
||||
$("#vf-password-reset-error").show();
|
||||
}
|
||||
}).fail(function () {
|
||||
$("#vf-password-reset-error").show();
|
||||
}).always(function () {
|
||||
$("#vf-password-reset-button-spinner").hide();
|
||||
});
|
||||
}
|
||||
|
||||
function vfLoginAsServerOwner(serviceId, systemUrl, newWindow) {
|
||||
newWindow = newWindow !== false;
|
||||
vfLoginError(false);
|
||||
$("#vf-login-button").prop("disabled", true);
|
||||
$("#vf-login-button-spinner").show();
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=loginAsServerOwner"
|
||||
}).done(function (response) {
|
||||
if (response.success && response.token_url) {
|
||||
if (newWindow) {
|
||||
window.open(response.token_url);
|
||||
} else {
|
||||
window.location.href = response.token_url;
|
||||
}
|
||||
} else {
|
||||
vfLoginError(true);
|
||||
}
|
||||
}).fail(function () {
|
||||
vfLoginError(true);
|
||||
}).always(function () {
|
||||
$("#vf-login-button-spinner").hide();
|
||||
$("#vf-login-button").prop("disabled", false);
|
||||
});
|
||||
}
|
||||
|
||||
function vfLoginError(show, message) {
|
||||
message = message || "Unable to open the control panel. Please try again later.";
|
||||
if (show) {
|
||||
$("#vf-login-error").text(message);
|
||||
$("#vf-login-error").show();
|
||||
} else {
|
||||
$("#vf-login-error").hide();
|
||||
}
|
||||
}
|
||||
|
||||
function vfPowerAction(serviceId, systemUrl, action) {
|
||||
var btn = $("#vf-power-" + action);
|
||||
var spinner = btn.find(".vf-btn-spinner");
|
||||
var alertDiv = $("#vf-power-alert");
|
||||
|
||||
// Disable all power buttons during action
|
||||
$(".vf-btn-power").prop("disabled", true);
|
||||
spinner.show();
|
||||
alertDiv.hide();
|
||||
|
||||
var actionLabels = {
|
||||
boot: "Starting",
|
||||
shutdown: "Shutting down",
|
||||
restart: "Restarting",
|
||||
poweroff: "Forcing off"
|
||||
};
|
||||
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=powerAction&powerAction=" + encodeURIComponent(action)
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
alertDiv.removeClass("alert-danger").addClass("alert-success");
|
||||
alertDiv.text(response.data.message || (actionLabels[action] + " server..."));
|
||||
} else {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text(response.errors || "Power action failed.");
|
||||
}
|
||||
alertDiv.show();
|
||||
}).fail(function () {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("An error occurred. Please try again.");
|
||||
alertDiv.show();
|
||||
}).always(function () {
|
||||
spinner.hide();
|
||||
$(".vf-btn-power").prop("disabled", false);
|
||||
});
|
||||
}
|
||||
|
||||
function vfLoadOsTemplates(serviceId, systemUrl) {
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=osTemplates"
|
||||
}).done(function (response) {
|
||||
var select = $("#vf-rebuild-os");
|
||||
select.empty();
|
||||
if (response.success && response.data && response.data.length > 0) {
|
||||
select.append('<option value="">-- Select Operating System --</option>');
|
||||
$.each(response.data, function (i, template) {
|
||||
select.append('<option value="' + template.id + '">' + $('<span>').text(template.name).html() + '</option>');
|
||||
});
|
||||
} else {
|
||||
select.append('<option value="">No templates available</option>');
|
||||
}
|
||||
}).fail(function () {
|
||||
var select = $("#vf-rebuild-os");
|
||||
select.empty();
|
||||
select.append('<option value="">Error loading templates</option>');
|
||||
});
|
||||
}
|
||||
|
||||
function vfRebuildServer(serviceId, systemUrl) {
|
||||
var osId = $("#vf-rebuild-os").val();
|
||||
var alertDiv = $("#vf-rebuild-alert");
|
||||
|
||||
if (!osId) {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("Please select an operating system.");
|
||||
alertDiv.show();
|
||||
return;
|
||||
}
|
||||
|
||||
if (!confirm("Are you sure you want to rebuild this server? ALL DATA WILL BE ERASED. This action cannot be undone.")) {
|
||||
return;
|
||||
}
|
||||
|
||||
$("#vf-rebuild-button").prop("disabled", true);
|
||||
$("#vf-rebuild-spinner").show();
|
||||
alertDiv.hide();
|
||||
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=rebuild&osId=" + encodeURIComponent(osId)
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
alertDiv.removeClass("alert-danger").addClass("alert-success");
|
||||
alertDiv.text(response.data.message || "Server rebuild initiated. You will receive an email when the process is complete.");
|
||||
} else {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text(response.errors || "Rebuild failed.");
|
||||
}
|
||||
alertDiv.show();
|
||||
}).fail(function () {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("An error occurred. Please try again.");
|
||||
alertDiv.show();
|
||||
}).always(function () {
|
||||
$("#vf-rebuild-spinner").hide();
|
||||
$("#vf-rebuild-button").prop("disabled", false);
|
||||
});
|
||||
}
|
||||
|
||||
function impersonateServerOwner(serviceId, systemUrl) {
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/admin.php?serviceID=" + encodeURIComponent(serviceId) + "&action=impersonateServerOwner"
|
||||
}).done(function (response) {
|
||||
if (response.success && response.user) {
|
||||
window.open(response.url + "/_imp/in/" + response.user.id + "/-");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Network / IP Management
|
||||
// =========================================================================
|
||||
|
||||
function vfRemoveIP(serviceId, systemUrl, action, identifier) {
|
||||
if (!confirm("Are you sure you want to remove this IP address?")) {
|
||||
return;
|
||||
}
|
||||
|
||||
var alertDiv = $("#vf-network-alert");
|
||||
alertDiv.hide();
|
||||
|
||||
var paramName = action === "removeIPv4" ? "ip" : "subnet";
|
||||
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=" + encodeURIComponent(action) + "&" + paramName + "=" + identifier
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
alertDiv.removeClass("alert-danger").addClass("alert-success");
|
||||
alertDiv.text(response.data.message || "IP address removed successfully.");
|
||||
alertDiv.show();
|
||||
vfServerData(serviceId, systemUrl);
|
||||
} else {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text(response.errors || "Failed to remove IP address.");
|
||||
alertDiv.show();
|
||||
}
|
||||
}).fail(function () {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("An error occurred. Please try again.");
|
||||
alertDiv.show();
|
||||
});
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// VNC Console
|
||||
// =========================================================================
|
||||
|
||||
function vfOpenVnc(serviceId, systemUrl) {
|
||||
var btn = $("#vf-vnc-button");
|
||||
var spinner = $("#vf-vnc-spinner");
|
||||
var alertDiv = $("#vf-vnc-alert");
|
||||
|
||||
btn.prop("disabled", true);
|
||||
spinner.show();
|
||||
alertDiv.hide();
|
||||
|
||||
// Open window immediately in click context to avoid popup blockers
|
||||
var vncWindow = window.open("", "_blank");
|
||||
if (!vncWindow) {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("Popup blocked. Please allow popups for this site and try again.");
|
||||
alertDiv.show();
|
||||
spinner.hide();
|
||||
btn.prop("disabled", false);
|
||||
return;
|
||||
}
|
||||
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=vnc"
|
||||
}).done(function (response) {
|
||||
if (response.success && response.data) {
|
||||
var data = response.data.data || response.data;
|
||||
if (data.url) {
|
||||
vncWindow.location.href = data.url;
|
||||
} else if (data.host && data.port) {
|
||||
// Build noVNC URL if available
|
||||
var vncUrl = "https://" + data.host + ":" + data.port;
|
||||
if (data.token) {
|
||||
vncUrl += "?token=" + encodeURIComponent(data.token);
|
||||
}
|
||||
vncWindow.location.href = vncUrl;
|
||||
} else {
|
||||
vncWindow.close();
|
||||
alertDiv.removeClass("alert-danger").addClass("alert-success");
|
||||
alertDiv.text("VNC session is ready. Check your VirtFusion control panel for access.");
|
||||
alertDiv.show();
|
||||
}
|
||||
} else {
|
||||
vncWindow.close();
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text(response.errors || "VNC console is not available.");
|
||||
alertDiv.show();
|
||||
}
|
||||
}).fail(function () {
|
||||
vncWindow.close();
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("An error occurred. The server may be powered off.");
|
||||
alertDiv.show();
|
||||
}).always(function () {
|
||||
spinner.hide();
|
||||
btn.prop("disabled", false);
|
||||
});
|
||||
}
|
||||
|
||||
// =========================================================================
|
||||
// Self Service — Credit & Usage
|
||||
// =========================================================================
|
||||
|
||||
function vfLoadSelfServiceUsage(serviceId, systemUrl) {
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=selfServiceUsage"
|
||||
}).done(function (response) {
|
||||
if (response.success && response.data) {
|
||||
var data = response.data.data || response.data;
|
||||
|
||||
// Credit balance
|
||||
var balance = "-";
|
||||
if (data.credit !== undefined) {
|
||||
balance = parseFloat(data.credit).toFixed(2);
|
||||
} else if (data.balance !== undefined) {
|
||||
balance = parseFloat(data.balance).toFixed(2);
|
||||
}
|
||||
$("#vf-ss-credit-balance").text(balance);
|
||||
|
||||
// Usage breakdown
|
||||
var tbody = $("#vf-ss-usage-table");
|
||||
tbody.empty();
|
||||
|
||||
var items = data.usage || data.items || [];
|
||||
if (Array.isArray(items) && items.length > 0) {
|
||||
$.each(items, function (i, item) {
|
||||
var desc = item.description || item.name || item.server || "Item";
|
||||
var cost = item.cost !== undefined ? parseFloat(item.cost).toFixed(2) : "-";
|
||||
tbody.append('<tr><td>' + $('<span>').text(desc).html() + '</td><td class="text-right">' + $('<span>').text(cost).html() + '</td></tr>');
|
||||
});
|
||||
} else {
|
||||
tbody.append('<tr><td colspan="2" class="text-muted">No usage data available</td></tr>');
|
||||
}
|
||||
|
||||
$("#vf-selfservice-content").show();
|
||||
$("#vf-selfservice-panel").show();
|
||||
}
|
||||
}).fail(function () {
|
||||
// Self-service not available — keep panel hidden
|
||||
}).always(function () {
|
||||
$("#vf-selfservice-loader").hide();
|
||||
});
|
||||
}
|
||||
|
||||
function vfLoadSelfServiceReport(serviceId, systemUrl) {
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=selfServiceReport"
|
||||
}).done(function (response) {
|
||||
if (response.success && response.data) {
|
||||
var data = response.data.data || response.data;
|
||||
var tbody = $("#vf-ss-usage-table");
|
||||
tbody.empty();
|
||||
|
||||
var items = data.items || data.report || [];
|
||||
if (Array.isArray(items) && items.length > 0) {
|
||||
$.each(items, function (i, item) {
|
||||
var desc = item.description || item.name || "Item";
|
||||
var cost = item.cost !== undefined ? parseFloat(item.cost).toFixed(2) : "-";
|
||||
tbody.append('<tr><td>' + $('<span>').text(desc).html() + '</td><td class="text-right">' + $('<span>').text(cost).html() + '</td></tr>');
|
||||
});
|
||||
} else {
|
||||
tbody.append('<tr><td colspan="2" class="text-muted">No report data available</td></tr>');
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function vfAddCredit(serviceId, systemUrl) {
|
||||
var amount = $("#vf-ss-credit-amount").val();
|
||||
var alertDiv = $("#vf-selfservice-alert");
|
||||
var btn = $("#vf-ss-add-credit-btn");
|
||||
var spinner = $("#vf-ss-add-credit-spinner");
|
||||
|
||||
if (!amount || parseFloat(amount) <= 0) {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("Please enter a valid positive amount.");
|
||||
alertDiv.show();
|
||||
return;
|
||||
}
|
||||
|
||||
btn.prop("disabled", true);
|
||||
spinner.show();
|
||||
alertDiv.hide();
|
||||
|
||||
$.ajax({
|
||||
type: "GET",
|
||||
dataType: "json",
|
||||
url: systemUrl + "modules/servers/VirtFusionDirect/client.php?serviceID=" + encodeURIComponent(serviceId) + "&action=selfServiceAddCredit&tokens=" + encodeURIComponent(amount)
|
||||
}).done(function (response) {
|
||||
if (response.success) {
|
||||
alertDiv.removeClass("alert-danger").addClass("alert-success");
|
||||
alertDiv.text("Credit added successfully.");
|
||||
alertDiv.show();
|
||||
$("#vf-ss-credit-amount").val("");
|
||||
// Refresh usage data
|
||||
vfLoadSelfServiceUsage(serviceId, systemUrl);
|
||||
} else {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text(response.errors || "Failed to add credit.");
|
||||
alertDiv.show();
|
||||
}
|
||||
}).fail(function () {
|
||||
alertDiv.removeClass("alert-success").addClass("alert-danger");
|
||||
alertDiv.text("An error occurred. Please try again.");
|
||||
alertDiv.show();
|
||||
}).always(function () {
|
||||
spinner.hide();
|
||||
btn.prop("disabled", false);
|
||||
});
|
||||
}
|
||||
|
||||
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user