Compare commits
	
		
			40 Commits
		
	
	
		
			07106b9aed
			...
			develop
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1145b900e4 | |||
| 436cd86d8c | |||
| 540444dfbf | |||
| 2d7c51ed15 | |||
| c842d9774f | |||
| 5963504f6b | |||
| e64a59714a | |||
| dd5d908e43 | |||
| 28846af43c | |||
| b1e3953960 | |||
| 7e00e23801 | |||
| bd12bbe862 | |||
| 48c203f028 | |||
| d859e40e4a | |||
| 8c3649f691 | |||
| b47194ed99 | |||
| 6bca42f2a5 | |||
| f1211f3cca | |||
| 0913daebc6 | |||
| d1a4b4d42f | |||
| 2906b8724b | |||
| 5e6e6c394a | |||
| 315050f095 | |||
| f795ba24f8 | |||
| 738307d070 | |||
| 04cff2e2c3 | |||
| 0b5981242b | |||
| f0dc598f44 | |||
| d4236f455e | |||
| 96fcf19c14 | |||
| 8c51bd8690 | |||
| 43464bb550 | |||
| 6a1b87e250 | |||
| 9473e26b50 | |||
| f19444201d | |||
| f907b7e072 | |||
| 80d0183391 | |||
| 5f3de290f3 | |||
| dbc5105b9b | |||
| a61289a318 | 
							
								
								
									
										65
									
								
								.drone.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								.drone.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: build | ||||
|  | ||||
| steps: | ||||
|   - name: build-image | ||||
|     image: plugins/docker | ||||
|     settings: | ||||
|       repo: t3hn3rd/asuro-build | ||||
|       tags: latest | ||||
|       dockerfile: Dockerfile | ||||
|       registry: docker.io | ||||
|       username: | ||||
|         from_secret: docker_username | ||||
|       password: | ||||
|         from_secret: docker_password | ||||
|  | ||||
|   - name: compile | ||||
|     image: t3hn3rd/asuro-build:latest | ||||
|     depends_on: | ||||
|       - build-image | ||||
|     commands: | ||||
|       - git fetch --tags | ||||
|       - find . -type f -print0 | xargs -0 dos2unix | ||||
|       - chmod +x /drone/src/*.sh | ||||
|       - /drone/src/compile.sh | ||||
|        | ||||
|   - name: upload-iso-artifact | ||||
|     image: alpine/git | ||||
|     depends_on: | ||||
|       - compile | ||||
|     when: | ||||
|       branch: | ||||
|         - master | ||||
|     environment: | ||||
|       GITEA_TOKEN: | ||||
|         from_secret: gitea_token | ||||
|     commands: | ||||
|       - apk add --no-cache curl | ||||
|       - chmod +w Asuro.iso | ||||
|       - cp Asuro.iso "Asuro-$(git rev-parse --short HEAD).iso" | ||||
|       - echo "Uploading Asuro-$(git rev-parse --short HEAD).iso to Gitea Packages..." | ||||
|       - | | ||||
|          curl -X PUT "https://gitea.spexeah.com/api/packages/Spexeah/generic/asuro-iso/$(git rev-parse --short HEAD)/Asuro.iso" \ | ||||
|          -H "Authorization: token $GITEA_TOKEN" --upload-file "Asuro-$(git rev-parse --short HEAD).iso" | ||||
|       - echo "Updating latest ISO reference..." | ||||
|       - | | ||||
|         curl -X DELETE https://gitea.spexeah.com/api/packages/Spexeah/generic/asuro-iso/latest/Asuro.iso \ | ||||
|         -H "Authorization: token $GITEA_TOKEN" || echo "No previous latest version found." | ||||
|       - | | ||||
|         curl -X PUT https://gitea.spexeah.com/api/packages/Spexeah/generic/asuro-iso/latest/Asuro.iso \ | ||||
|         -H "Authorization: token $GITEA_TOKEN" --upload-file "Asuro-$(git rev-parse --short HEAD).iso" | ||||
|  | ||||
|   - name: msg status | ||||
|     image: appleboy/drone-discord | ||||
|     depends_on: | ||||
|       - compile | ||||
|     when: | ||||
|       status: [success, failure, changed] | ||||
|     settings: | ||||
|       webhook_id: | ||||
|         from_secret: discord_webhook_id | ||||
|       webhook_token: | ||||
|         from_secret: discord_webhook_secret | ||||
|       message: "**Asuro Build**\n\n{{#success build.status}}✅ Build successful!\n\n{{else}}❌ Build failed!\n\n{{/success}}Repository: `{{repo.namespace}}/{{repo.name}}`\nBranch: `{{commit.branch}}`\nCommit: `{{commit.sha}}`\nAuthor: `{{commit.author}} <{{commit.email}}>`\n\nGitea Diff: [Link](<{{commit.link}}>)\nDrone Build: [Link](<{{build.link}}>)\n\nMessage: {{commit.message}}" | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -10,5 +10,4 @@ | ||||
| /*.sh~ | ||||
| /*.img | ||||
| src/include/asuro.pas | ||||
| .vscode/launch.json | ||||
| .vscode | ||||
| localenv.json | ||||
|   | ||||
| @@ -1,73 +0,0 @@ | ||||
| stages: | ||||
|   - Compile Versions | ||||
|   - Compile Sources | ||||
|   - Link | ||||
|   - Generate Documentation | ||||
|   - Deploy | ||||
|  | ||||
| cache: | ||||
|   - key: ${CI_COMMIT_REF_SLUG} | ||||
|     paths: | ||||
|       - lib/*.o | ||||
|       - bin/kernel.bin | ||||
|       - doc | ||||
|  | ||||
| before_script: | ||||
|   - chmod +x *.sh | ||||
|  | ||||
| versions: | ||||
|   stage: Compile Versions | ||||
|   script: | ||||
|     - ./compile_vergen.sh | ||||
|   artifacts: | ||||
|     paths: | ||||
|       - release/*.svg | ||||
|       - src/include/asuro.pas | ||||
|  | ||||
| compile_sources: | ||||
|   stage: Compile Sources | ||||
|   script: | ||||
|     - rm -f lib/*.so | ||||
|     - ./compile_sources.sh | ||||
|   needs: | ||||
|     - versions | ||||
|  | ||||
| link: | ||||
|   stage: Link | ||||
|   script: | ||||
|     - ./compile_stub.sh | ||||
|     - ./compile_link.sh | ||||
|   needs: | ||||
|     - versions | ||||
|     - compile_sources | ||||
|  | ||||
| isogen: | ||||
|   stage: Deploy | ||||
|   script: | ||||
|     - ./compile_isogen.sh | ||||
|     - ./compile_sumgen.sh | ||||
|   artifacts: | ||||
|     paths: | ||||
|       - ./Asuro.iso | ||||
|       - ./release/checksum.svg | ||||
|   needs: | ||||
|     - link | ||||
|  | ||||
| docgen: | ||||
|   stage: Generate Documentation | ||||
|   only: | ||||
|     - master | ||||
|     - develop | ||||
|   script: | ||||
|     - ./compile_sourcelist.sh | ||||
|     - ./compile_docs.sh | ||||
|     # Remove comments when we want to use gitlab pages. | ||||
|     #- cp doc public | ||||
|   allow_failure: true | ||||
|   artifacts: | ||||
|     paths: | ||||
|       - doc | ||||
|       #- public/* | ||||
|       #- ./sources.list | ||||
|   needs: | ||||
|     - versions | ||||
							
								
								
									
										13
									
								
								.mailmap
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.mailmap
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| # Map all Aaron identities to a single preferred identity | ||||
| Aaron Hance <ah@aaronhance.me> aaron <aaron@6dbc8c32-bb84-406f-8558-d1cf31a0ab0c> | ||||
| Aaron Hance <ah@aaronhance.me> aaronhance <ah@aaronhance.me> | ||||
| Aaron Hance <ah@aaronhance.me> aaron hance <ah@aaronhance.me> | ||||
| Aaron Hance <ah@aaronhance.me> Aaron <ah@aaronhance.me> | ||||
|  | ||||
| # Map all Kieron identities to a single preferred identity | ||||
| Kieron Morris <kjm@kieronmorris.me> kieron <kieron@6dbc8c32-bb84-406f-8558-d1cf31a0ab0c> | ||||
| Kieron Morris <kjm@kieronmorris.me> Kieron <kjm@kieronmorris.me> | ||||
| Kieron Morris <kjm@kieronmorris.me> t3hn3rd <kjm@kieronmorris.me> | ||||
|  | ||||
| # Map goose's multiple identities | ||||
| goose <goose@6dbc8c32-bb84-406f-8558-d1cf31a0ab0c> goose <angus@actm.uk> | ||||
							
								
								
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ | ||||
|             "type": "PowerShell", | ||||
|             "preLaunchTask": "Build", | ||||
|             "script": "${workspaceFolder}/virtualbox-wrapper.ps1", | ||||
|             "args": ["-MachineName", "7d395c96-891c-4139-b77d-9b6b144b0b93"], | ||||
|             "args": ["-Command", "up"], | ||||
|             "cwd": "${workspaceFolder}", | ||||
|         } | ||||
|     ] | ||||
|   | ||||
							
								
								
									
										26
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -9,14 +9,18 @@ | ||||
|             "command": "docker-compose", | ||||
|             "args": [ | ||||
|                 "run", | ||||
|                 "builder" | ||||
|                 "builder", | ||||
|             ], | ||||
|             "type": "shell", | ||||
|             "problemMatcher": [], | ||||
|             "group": { | ||||
|                 "kind": "build", | ||||
|                 "isDefault": true | ||||
|             } | ||||
|             }, | ||||
|             "dependsOn": [ | ||||
|                 "Close VirtualBox", | ||||
|                 "Clean" | ||||
|             ] | ||||
|         }, | ||||
|         { | ||||
|             "label": "Build (Builder)", | ||||
| @@ -26,6 +30,24 @@ | ||||
|                 "builder" | ||||
|             ], | ||||
|             "type": "shell" | ||||
|         }, | ||||
|         { | ||||
|             "label": "Clean", | ||||
|             "command": "docker-compose", | ||||
|             "args": [ | ||||
|                 "down", | ||||
|                 "--remove-orphans" | ||||
|             ], | ||||
|             "type": "shell" | ||||
|         }, | ||||
|         { | ||||
|             "label": "Close VirtualBox", | ||||
|             "command": "./virtualbox-wrapper.ps1", | ||||
|             "args": [ | ||||
|                 "-Command",  | ||||
|                 "down" | ||||
|             ], | ||||
|             "type": "shell" | ||||
|         } | ||||
|     ] | ||||
| } | ||||
| @@ -16,7 +16,7 @@ RUN curl -sL https://sourceforge.net/projects/freepascal/files/Linux/$FPC_VERSIO | ||||
|  | ||||
| COPY compile.sh /compile.sh | ||||
| ADD https://raw.githubusercontent.com/fsaintjacques/semver-tool/master/src/semver /usr/bin/semver | ||||
| RUN mkdir /code && chmod +x /usr/bin/semver | ||||
| RUN chmod +x /usr/bin/semver | ||||
| WORKDIR /code | ||||
| RUN find . -type f -print0 | xargs -0 dos2unix | ||||
| ENTRYPOINT ["/bin/bash", "-c"] | ||||
|   | ||||
							
								
								
									
										32
									
								
								compile.sh
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								compile.sh
									
									
									
									
									
								
							| @@ -20,26 +20,30 @@ runOrFail() { | ||||
| 	fi | ||||
| } | ||||
|  | ||||
| runOrFail $(pwd)/compile_stub.sh "Failed to compile stub!" | ||||
| declare -a run_steps=( | ||||
| 	"compile_stub.sh" "Failed to compile stub!" | ||||
| 	"compile_vergen.sh" "Versions failed to compile" | ||||
| 	"compile_sources.sh" "Failed to compile FPC Sources!" | ||||
| 	"compile_link.sh" "Failed linking!" | ||||
| 	"compile_isogen.sh" "Failed to create ISO!" | ||||
| ) | ||||
|  | ||||
| #Generate .pas with versioning headers. | ||||
| runOrFail $(pwd)/compile_vergen.sh "Versions failed to compile" | ||||
|  | ||||
| #Compile all .pas sources | ||||
| runOrFail $(pwd)/compile_sources.sh "Failed to compile FPC Sources!" | ||||
|  | ||||
| #Link into a binary. | ||||
| runOrFail $(pwd)/compile_link.sh "Failed linking!" | ||||
|  | ||||
| #Generate an ISO with GRUB as the Bootloader. | ||||
| runOrFail ./compile_isogen.sh "Failed to create ISO!" | ||||
| for ((i=0; i<${#run_steps[@]}; i+=2)) | ||||
| do | ||||
| 	if [ "$ERRCOUNT" -eq "0" ] | ||||
| 	then | ||||
| 			script=$(pwd)/"${run_steps[$i]}" | ||||
| 			message="${run_steps[$i+1]}" | ||||
| 			runOrFail "$script" "$message" | ||||
| 	fi | ||||
| done | ||||
|  | ||||
| #Call generate final artifacts based on failure or success of the above. | ||||
| if [ "$ERRCOUNT" -ne "0" ] | ||||
| then | ||||
| 	./compile_finish.sh "failed" | ||||
| 	. ./compile_finish.sh "failed" | ||||
| else | ||||
| 	./compile_finish.sh "success" | ||||
| 	. ./compile_finish.sh "success" | ||||
| fi | ||||
|  | ||||
| cd .. | ||||
| @@ -1,4 +1,3 @@ | ||||
| version: "3.9" | ||||
| services: | ||||
|   builder: | ||||
|     build: . | ||||
|   | ||||
							
								
								
									
										27
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								readme.md
									
									
									
									
									
								
							| @@ -50,23 +50,26 @@ We welcome everyone to give building/breaking/fixing/shooting Asuro a go, feel f | ||||
|     ```xml | ||||
|     <Machine uuid="{7d395c96-891c-4139-b77d-9b6b144b0b93}" name="Asuro" OSType="Linux" snapshotFolder="Snapshots" lastStateChange="2021-06-20T20:33:07Z"> | ||||
|     ``` | ||||
|     Copy the uuid, in our case `7d395c96-891c-4139-b77d-9b6b144b0b93` and replace the uuid found in `.vscode\launch.json` under `args`, so that it looks something like this: | ||||
|     Copy the uuid, in our case `7d395c96-891c-4139-b77d-9b6b144b0b93` & create a `localenv.json` file in the project root with the following content: | ||||
|     ```json | ||||
|     { | ||||
|         "configurations": [ | ||||
|             { | ||||
|                 "name":"Run", | ||||
|                 "request": "launch", | ||||
|                 "type": "PowerShell", | ||||
|                 "preLaunchTask": "Build", | ||||
|                 "script": "${workspaceFolder}/virtualbox-wrapper.ps1", | ||||
|                 "args": ["-MachineName", "7d395c96-891c-4139-b77d-9b6b144b0b93"], | ||||
|                 "cwd": "${workspaceFolder}", | ||||
|             } | ||||
|         ] | ||||
|         "VirtualBox":{ | ||||
|             "MachineName":"<YOUR_UUID_OR_MACHINE_NAME>" | ||||
|         } | ||||
|     } | ||||
|     ``` | ||||
|     This will allow VSCode to automatically launch VirtualBox once Asuro has been compiled. | ||||
|      | ||||
|     You can also enable the serial adapter "COM1" in mode "Raw File", give it a path, and provide this path in the `localenv.json` as follows: | ||||
|     ```json | ||||
|     { | ||||
|         "VirtualBox" : { | ||||
|             "MachineName": "<YOUR_UUID_OR_MACHINE_NAME>", | ||||
|             "LogLocation": "Fully\\Qualified\\Path\\To\\Your\\Log\\File" | ||||
|         } | ||||
|     } | ||||
|     ``` | ||||
|     This will allow you to see the console output from Asuro in your host terminal. | ||||
| 13. Open your project folder in VSCode, use CTRL+SHIFT+B to build & F5 to build + run in VBox. | ||||
| 14. Congratulations! You can now play with Asuro! | ||||
|  | ||||
|   | ||||
| @@ -491,22 +491,26 @@ begin | ||||
|          | ||||
|         { Create a new header to the use in our DHCP REQUEST packet } | ||||
|         SendHeader:= createHeader(); | ||||
|         CopyIPv4(puint8(@Header^.Your_IP[0]), puint8(@SendHeader^.Client_IP[0])); | ||||
|         CopyIPv4(@getIPv4Config^.Address[0], puint8(@SendHeader^.Client_IP[0])); | ||||
|         CopyIPv4(puint8(@Header^.Server_IP[0]), puint8(@SendHeader^.Server_IP[0])); | ||||
|         processHeader(SendHeader); | ||||
|  | ||||
|         { Setup Options } | ||||
|         SendOptions:= newOptions(); | ||||
|  | ||||
|         { Create a message type option and assign it the value REQUEST } | ||||
|         SendMsgType:= ord(TDHCPMessageType.REQUEST); | ||||
|         NewOption(SendOptions, TDHCPOpCode.DHCP_MESSAGE_TYPE, void(@SendMsgType), 1, false); | ||||
|  | ||||
|         { Create a Requested IP option and assign it the value from the OFFER packet header } | ||||
|         NewOption(SendOptions, TDHCPOpCode.REQUESTED_IP_ADDRESS, void(@SendHeader^.Client_IP[0]), 4, false); | ||||
|         NewOption(SendOptions, TDHCPOpCode.REQUESTED_IP_ADDRESS, void(@Header^.Your_IP[0]), 4, false); | ||||
|  | ||||
|         { Create a Server Identifier Option and assign it the value from the OFFER packet options } | ||||
|         Option:= getOptionByOpcode(Options, TDHCPOpCode.SERVER_IDENTIFIER); | ||||
|         if Option <> nil then begin | ||||
|             NewOption(SendOptions, TDHCPOpCode.SERVER_IDENTIFIER, void(@Option^.Value[0]), 4, false); | ||||
|         end; | ||||
|  | ||||
|         { Create a Parameter Request List, Request the following: Netmask, Gateway, DNS Name & DNS Server } | ||||
|         RequestParams[0]:= Ord(TDHCPOpCode.SUBNET_MASK); | ||||
|         RequestParams[1]:= Ord(TDHCPOpCode.ROUTER); | ||||
| @@ -522,9 +526,13 @@ begin | ||||
|         MAC:= getMAC(); | ||||
|         packetCtx:= PPacketContext(Kalloc(sizeof(TPacketContext))); | ||||
|         packetCtx^.TTL:= 128; | ||||
|         copyMAC(@BROADCAST_MAC[0], @packetCtx^.MAC.Destination[0]); | ||||
|  | ||||
|         { Copy over MAC - src: broadcast - dst: DHCP Server } | ||||
|         copyMAC(MAC, @packetCtx^.MAC.Source[0]); | ||||
|         copyIPv4(@NULL_IP[0], @packetCtx^.IP.Source[0]); | ||||
|         copyMAC(@BROADCAST_MAC[0], @packetCtx^.MAC.Destination[0]); | ||||
|          | ||||
|         { Copy over IP - src: NULL - dst: Broadcast } | ||||
|         CopyIPv4(@getIPv4Config^.Address[0], @packetCtx^.IP.Source[0]); | ||||
|         copyIPv4(@BROADCAST_IP[0], @packetCtx^.IP.Destination[0]); | ||||
|  | ||||
|         { Setup UDPContext (UDP) & copy in the correct details } | ||||
| @@ -570,7 +578,7 @@ begin | ||||
|  | ||||
|     { Check the frame is for us and then process } | ||||
|     MAC:= getMAC; | ||||
|     if MACEqual(@context^.PacketContext^.MAC.Destination[0], MAC) then begin | ||||
|     if MACEqual(@context^.PacketContext^.MAC.Destination[0], MAC) or MACEqual(@context^.PacketContext^.MAC.Destination[0], BROADCAST_MAC) then begin | ||||
|         Outputln('DHCP','Frame is addressed to us.'); | ||||
|         { Check the message type is client specific } | ||||
|         If Header^.Message_Type = $02 then begin | ||||
| @@ -655,7 +663,7 @@ begin | ||||
|         copyMAC(@BROADCAST_MAC[0], @packetCtx^.MAC.Destination[0]); | ||||
|         MAC:= getMAC; | ||||
|         copyMAC(MAC, @packetCtx^.MAC.Source[0]); | ||||
|         copyIPv4(@NULL_IP[0], @packetCtx^.IP.Source[0]); | ||||
|         CopyIPv4(@getIPv4Config^.Address[0], @packetCtx^.IP.Source[0]); | ||||
|         copyIPv4(@BROADCAST_IP[0], @packetCtx^.IP.Destination[0]); | ||||
|  | ||||
|         { Setup UDPContext (UDP) } | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,16 +1,59 @@ | ||||
| <#  | ||||
| You need a local git-ignored localenv.json file with the following content: | ||||
| { | ||||
|   "virtualbox": { | ||||
|     "MachineName": "your-machine-name or guid" | ||||
|   } | ||||
| } | ||||
| #> | ||||
|  | ||||
| param ( | ||||
|   $MachineName | ||||
|   [Parameter(Mandatory=$true)] | ||||
|   [ValidateSet('up', 'down')] | ||||
|   [String]$Command | ||||
| ) | ||||
|  | ||||
| VBoxManage.exe startvm $MachineName | ||||
| $Config = Get-Content .\localenv.json | ConvertFrom-Json | ||||
| $MachineName = $Config.virtualbox.MachineName | ||||
| $LogLocation = $Config.virtualbox.LogLocation | ||||
| $LogOutputEnabled = $LogLocation -ne $null | ||||
|  | ||||
| $running=$true | ||||
| while($running) { | ||||
|   Start-Sleep -Seconds 1 | ||||
|   $status=(VBoxManage.exe list runningvms) | ||||
|   if($status) { | ||||
|     $running=$status.contains($MachineName) | ||||
|   } else { | ||||
|     $running=$false | ||||
| if ($Command -eq 'up') { | ||||
|  | ||||
|   if($LogOutputEnabled) { | ||||
|     Clear-Content $LogLocation | ||||
|   } | ||||
|  | ||||
|   $MonitorJob = Start-Job -ArgumentList $MachineName -ScriptBlock { | ||||
|     param($MachineName) | ||||
|     Write-Output "Starting $MachineName" | ||||
|     VBoxManage.exe startvm $MachineName | ||||
|     $running=$true | ||||
|     while($running) { | ||||
|       $status=(VBoxManage.exe list runningvms) | ||||
|       if($status) { | ||||
|         $running=$status.contains($MachineName) | ||||
|       } else { | ||||
|         $running=$false | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   if($LogOutputEnabled) { | ||||
|     $LogJob = Start-Job -ArgumentList $LogLocation -ScriptBlock { | ||||
|       param($LogLocation) | ||||
|       Get-Content -Path $LogLocation -Wait | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   while($MonitorJob.State -eq 'Running') { | ||||
|     if($LogOutputEnabled) { | ||||
|       Receive-Job $LogJob | ||||
|     } | ||||
|     Receive-Job $MonitorJob | ||||
|   } | ||||
|   Get-Job | Stop-Job | Remove-Job -Force | ||||
| } elseif ($Command -eq 'down') { | ||||
|   Write-Output "Stopping $MachineName" | ||||
|   VBoxManage.exe controlvm $MachineName poweroff | ||||
| } | ||||
		Reference in New Issue
	
	Block a user