12 Commits

Author SHA1 Message Date
eb4fae52ab improve zipping 2025-08-05 20:29:50 +02:00
Adrian Zürcher
e188813adf fix static bug and load main page
All checks were successful
Build Process Supervisor / build (amd64, .exe, windows) (push) Successful in 2m32s
Build Process Supervisor / build (amd64, , linux) (push) Successful in 2m41s
Build Process Supervisor / build (arm, 6, , linux) (push) Successful in 2m35s
Build Process Supervisor / build (arm64, , linux) (push) Successful in 2m31s
2025-08-05 20:09:28 +02:00
Adrian Zürcher
e1ba6e49a8 add and modify new index page 2025-08-05 18:45:19 +02:00
Adrian Zürcher
20912ba17c add first supervisor process 2025-08-05 17:17:55 +02:00
Adrian Zürcher
dd5ebb60fd move template in bin 2025-08-05 17:17:02 +02:00
Adrian Zürcher
0e8864d4da remove sudo
All checks were successful
Build Process Supervisor / build (amd64, .exe, windows) (push) Successful in 2m32s
Build Process Supervisor / build (amd64, , linux) (push) Successful in 2m45s
Build Process Supervisor / build (arm, 6, , linux) (push) Successful in 2m32s
Build Process Supervisor / build (arm64, , linux) (push) Successful in 2m34s
2025-08-05 15:50:52 +02:00
Adrian Zürcher
490fae7d5b fix install zip
Some checks failed
Build Process Supervisor / build (amd64, , linux) (push) Failing after 16s
Build Process Supervisor / build (amd64, .exe, windows) (push) Failing after 16s
Build Process Supervisor / build (arm64, , linux) (push) Has been cancelled
Build Process Supervisor / build (arm, 6, , linux) (push) Has been cancelled
2025-08-05 15:48:01 +02:00
Adrian Zürcher
ca9d8af13f add install zip
Some checks failed
Build Process Supervisor / build (amd64, , linux) (push) Failing after 1m19s
Build Process Supervisor / build (amd64, .exe, windows) (push) Failing after 1m16s
Build Process Supervisor / build (arm, 6, , linux) (push) Failing after 1m17s
Build Process Supervisor / build (arm64, , linux) (push) Has been cancelled
2025-08-05 15:44:09 +02:00
Adrian Zürcher
2537d5f1c8 second fix wrong go version
Some checks failed
Build Process Supervisor / build (amd64, .exe, windows) (push) Failing after 6m39s
Build Process Supervisor / build (amd64, , linux) (push) Failing after 6m52s
Build Process Supervisor / build (arm, 6, , linux) (push) Failing after 6m42s
Build Process Supervisor / build (arm64, , linux) (push) Failing after 6m33s
2025-08-05 15:29:28 +02:00
Adrian Zürcher
79ad631494 fix wrong go version
Some checks failed
Build Process Supervisor / build (amd64, , linux) (push) Failing after 22s
Build Process Supervisor / build (amd64, .exe, windows) (push) Failing after 23s
Build Process Supervisor / build (arm, 6, , linux) (push) Failing after 22s
Build Process Supervisor / build (arm64, , linux) (push) Failing after 22s
2025-08-05 15:23:16 +02:00
Adrian Zürcher
a0046e1848 try to cache go
Some checks failed
Build Process Supervisor / build (amd64, , linux) (push) Failing after 2m52s
Build Process Supervisor / build (amd64, .exe, windows) (push) Failing after 2m53s
Build Process Supervisor / build (arm, 6, , linux) (push) Failing after 26s
Build Process Supervisor / build (arm64, , linux) (push) Failing after 26s
2025-08-05 15:17:58 +02:00
Adrian Zürcher
d5f314dad2 fix yaml
Some checks failed
Build Process Supervisor / build (amd64, , linux) (push) Successful in 20m56s
Build Process Supervisor / build (arm, 6, , linux) (push) Has been cancelled
Build Process Supervisor / build (arm64, , linux) (push) Has been cancelled
Build Process Supervisor / build (amd64, .exe, windows) (push) Has been cancelled
2025-08-05 14:16:20 +02:00
10 changed files with 647 additions and 77 deletions

View File

@@ -30,29 +30,53 @@ jobs:
ext: "" ext: ""
steps: steps:
- name: Install zip
run: |
apt-get update && apt-get install -y zip
- name: Checkout repo - name: Checkout repo
uses: actions/checkout@v4 uses: actions/checkout@v4
- name: Set up Go - name: Ensure latest Go is installed in /data/go
uses: actions/setup-go@v4 run: |
with: export GOROOT=/data/go/go
go-version: '1.24.5' export PATH=$GOROOT/bin:$PATH
export GOCACHE=/data/gocache
export GOMODCACHE=/data/gomodcache
mkdir -p $GOCACHE $GOMODCACHE
- name: Cache Go modules if [ ! -x "$GOROOT/bin/go" ]; then
uses: actions/cache@v4 echo "Go not found in $GOROOT, downloading latest stable..."
with:
path: | GO_VERSION=$(curl -s https://go.dev/VERSION?m=text | head -n1)
~/.cache/go-build echo "Latest version is $GO_VERSION"
~/go/pkg/mod
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }} mkdir -p /data/go
restore-keys: | curl -sSL "https://go.dev/dl/${GO_VERSION}.linux-amd64.tar.gz" -o /tmp/go.tar.gz
${{ runner.os }}-go- tar -C /data/go -xzf /tmp/go.tar.gz
else
echo "Using cached Go from $GOROOT"
fi
go version
- name: Download Go dependencies - name: Download Go dependencies
run: go mod download
- name: Build binaries
run: | run: |
export GOROOT=/data/go/go
export PATH=$GOROOT/bin:$PATH
export GOCACHE=/data/gocache
export GOMODCACHE=/data/gomodcache
mkdir -p $GOCACHE $GOMODCACHE
go mod download
- name: Build binary
run: |
export GOROOT=/data/go/go
export PATH=$GOROOT/bin:$PATH
export GOCACHE=/data/gocache
export GOMODCACHE=/data/gomodcache
mkdir -p $GOCACHE $GOMODCACHE
OUTPUT="bin/${APP_NAME}-${{ matrix.os }}-${{ matrix.arch }}" OUTPUT="bin/${APP_NAME}-${{ matrix.os }}-${{ matrix.arch }}"
if [ -n "${{ matrix.arm_version }}" ]; then if [ -n "${{ matrix.arm_version }}" ]; then
OUTPUT="${OUTPUT}v${{ matrix.arm_version }}" OUTPUT="${OUTPUT}v${{ matrix.arm_version }}"
@@ -63,8 +87,13 @@ jobs:
GOOS=${{ matrix.os }} GOARCH=${{ matrix.arch }} go build -ldflags="-s -w" -trimpath -o "$OUTPUT" GOOS=${{ matrix.os }} GOARCH=${{ matrix.arch }} go build -ldflags="-s -w" -trimpath -o "$OUTPUT"
shell: bash shell: bash
- name: Upload artifacts - name: Zip artifact
run: |
cd bin
zip "${{ env.APP_NAME }}-${{ matrix.os }}-${{ matrix.arch }}.zip" "${{ env.APP_NAME }}-"* -x "*.zip"
- name: Upload zipped artifact
uses: actions/upload-artifact@v3 uses: actions/upload-artifact@v3
with: with:
name: ${{ env.APP_NAME }}-${{ matrix.os }}-${{ matrix.arch }}${{ matrix.arm_version && format('v{0}', matrix.arm_version) || '' }}${{ matrix.ext }} name: ${{ env.APP_NAME }}-${{ matrix.os }}-${{ matrix.arch }}
path: bin/${{ env.APP_NAME }}-${{ matrix.os }}-${{ matrix.arch }}${{ matrix.arm_version && format('v{0}', matrix.arm_version) || '' }}${{ matrix.ext }} path: bin/${{ env.APP_NAME }}-${{ matrix.os }}-${{ matrix.arch }}.zip

11
dist/supervisorTemplate.json vendored Normal file
View File

@@ -0,0 +1,11 @@
{
"processes":[{
"name":"Database DBM",
"executePath":"dist/test-windows-amd64.exe",
"workingDirectory":".",
"startDelay":1000,
"priority":0,
"arguments":["-sleep 5"]
}
]
}

BIN
dist/test-windows-amd64.exe vendored Normal file

Binary file not shown.

View File

@@ -1,6 +1,7 @@
package handlers package handlers
import ( import (
"embed"
"encoding/json" "encoding/json"
"html/template" "html/template"
"net/http" "net/http"
@@ -9,15 +10,17 @@ import (
type HTopHandler struct { type HTopHandler struct {
Table *models.HtopTable Table *models.HtopTable
templates *embed.FS
} }
func NewHTopHandler() (*HTopHandler, error) { func NewHTopHandler(templates *embed.FS) (*HTopHandler, error) {
table, err := models.NewTable() table, err := models.NewTable()
if err != nil { if err != nil {
return nil, err return nil, err
} }
return &HTopHandler{ return &HTopHandler{
Table: table, Table: table,
templates: templates,
}, nil }, nil
} }
@@ -38,14 +41,18 @@ func (h *HTopHandler) UpdateHTop(w http.ResponseWriter, r *http.Request) {
// Detect HTMX request via the HX-Request header // Detect HTMX request via the HX-Request header
if r.Header.Get("HX-Request") == "true" { if r.Header.Get("HX-Request") == "true" {
tmpl := template.Must( tmpl := template.Must(
template.New("table.html").Funcs(funcMap).ParseFiles("templates/htop/table.html"), template.New("table.html").Funcs(funcMap).ParseFS(
h.templates,
"templates/htop/table.html",
),
) )
tmpl.Execute(w, h.Table) tmpl.Execute(w, h.Table)
return return
} }
tmpl := template.Must( tmpl := template.Must(
template.New("htop.html").Funcs(funcMap).ParseFiles( template.New("htop.html").Funcs(funcMap).ParseFS(
h.templates,
"templates/htop/htop.html", "templates/htop/htop.html",
"templates/htop/table.html", "templates/htop/table.html",
), ),

View File

@@ -1,17 +1,100 @@
package handlers package handlers
import ( import (
"embed"
"html/template" "html/template"
"log"
"net/http" "net/http"
"os"
"processSupervisor/models"
) )
type MainPage struct { type MainPage struct {
templates *embed.FS
Supervisor *models.Supervisor
} }
func UpdateMainPage(w http.ResponseWriter, r *http.Request) { func NewMainPage(templates *embed.FS) (*MainPage, error) {
w.Header().Set("Content-Type", "text/html; charset=utf-8") var supervisor models.Supervisor
tmpl := template.Must( if _, err := os.Stat("cfg/"); err != nil {
template.New("index.html").ParseFiles("templates/index.html"), s, err := models.ReadTemplate("dist/supervisorTemplate.json")
) if err != nil {
tmpl.Execute(w, nil) return nil, err
}
supervisor = s
}
if err := supervisor.StartProcesses(); err != nil {
return nil, err
}
return &MainPage{
templates: templates,
Supervisor: &supervisor,
}, nil
}
func (m *MainPage) UpdateMainPage(w http.ResponseWriter, r *http.Request) {
tmpl := template.Must(
template.New("index.html").ParseFS(
m.templates,
"templates/index.html",
),
)
// tmpl := template.Must(
// template.New("index.html").ParseFiles(
// "templates/index.html",
// ),
// )
tmpl.Execute(w, m)
}
func (m *MainPage) StartProcess(w http.ResponseWriter, r *http.Request) {
name := r.PostFormValue("name")
// This is where you trigger the .Start method
err := m.Supervisor.StartProcessByName(name)
if err != nil {
log.Println("Failed to start process", err)
//http.Error(w, "Failed to start process", http.StatusInternalServerError)
//return
}
m.UpdateMainPage(w, r)
//w.WriteHeader(http.StatusOK)
}
func (m *MainPage) StopProcess(w http.ResponseWriter, r *http.Request) {
name := r.PostFormValue("name")
err := m.Supervisor.StopProcessByName(name)
if err != nil {
log.Println("Failed to stop process", err)
//http.Error(w, "Failed to stop process", http.StatusInternalServerError)
//return
}
m.UpdateMainPage(w, r)
//w.WriteHeader(http.StatusOK)
}
func (m *MainPage) RestartProcess(w http.ResponseWriter, r *http.Request) {
name := r.PostFormValue("name")
err := m.Supervisor.StopProcessByName(name)
if err != nil {
log.Println("Failed to start process", err)
//http.Error(w, "Failed to start process", http.StatusInternalServerError)
//return
}
err = m.Supervisor.StartProcessByName(name)
if err != nil {
log.Println("Failed to stop process", err)
// http.Error(w, "Failed to stop process", http.StatusInternalServerError)
// return
}
m.UpdateMainPage(w, r)
//w.WriteHeader(http.StatusOK)
} }

34
main.go
View File

@@ -1,31 +1,49 @@
package main package main
import ( import (
"embed"
"flag" "flag"
"fmt" "fmt"
"io/fs"
"log" "log"
"net/http" "net/http"
"processSupervisor/handlers" "processSupervisor/handlers"
) )
//go:embed templates/*.html templates/*/*.html
var templatesFS embed.FS
//go:embed static
var staticFS embed.FS
func main() { func main() {
port := flag.Uint("port", 9400, "listenig port") port := flag.Uint("port", 9400, "listening port")
flag.Parse() htop, err := handlers.NewHTopHandler(&templatesFS)
htop, err := handlers.NewHTopHandler()
if err != nil { if err != nil {
panic(err) panic(err)
} }
fs := http.FileServer(http.Dir("static")) mainPage, err := handlers.NewMainPage(&templatesFS)
http.Handle("/static/", http.StripPrefix("/static/", fs)) if err != nil {
panic(err)
}
staticSub, err := fs.Sub(staticFS, "static")
if err != nil {
log.Fatal(err)
}
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.FS(staticSub))))
http.HandleFunc("/taskmanager/htop", htop.UpdateHTop) http.HandleFunc("/taskmanager/htop", htop.UpdateHTop)
http.HandleFunc("/", handlers.UpdateMainPage) http.HandleFunc("/", mainPage.UpdateMainPage)
http.HandleFunc("/start-process", mainPage.StartProcess)
http.HandleFunc("/stop-process", mainPage.StopProcess)
http.HandleFunc("/restart-process", mainPage.RestartProcess)
http.HandleFunc("/taskmanager/kill", handlers.KillHandler) http.HandleFunc("/taskmanager/kill", handlers.KillHandler)
http.HandleFunc("/taskmanager/usage", handlers.UsageHandler) http.HandleFunc("/taskmanager/usage", handlers.UsageHandler)
log.Printf("Listening on http://localhost:%d/taskmanager/htop\n", *port) log.Printf("Listening on http://localhost:%d\n", *port)
log.Fatal(http.ListenAndServe(fmt.Sprintf(":%d", *port), nil)) log.Fatal(http.ListenAndServe(fmt.Sprintf(":%d", *port), nil))
} }

59
models/supervisor.go Normal file
View File

@@ -0,0 +1,59 @@
package models
import (
"encoding/json"
"fmt"
"os"
"sort"
)
type Supervisor struct {
Processes []*SupervisorProcess `json:"processes"`
}
func ReadTemplate(path string) (supervisor Supervisor, err error) {
content, err := os.ReadFile(path)
if err != nil {
return
}
err = json.Unmarshal(content, &supervisor)
return
}
func (s *Supervisor) StartProcesses() error {
sort.Slice(s.Processes, func(i, j int) bool {
return s.Processes[i].Priority < s.Processes[j].Priority
})
for _, p := range s.Processes {
p.Start()
}
return nil
}
func (s *Supervisor) GetProcesses() (processes []SupervisorProcess) {
for _, p := range s.Processes {
processes = append(processes, *p)
}
return
}
func (s *Supervisor) StartProcessByName(name string) error {
for _, p := range s.Processes {
if p.Name == name {
return p.Start()
}
}
return fmt.Errorf("process %s not found", name)
}
func (s *Supervisor) StopProcessByName(name string) error {
for _, p := range s.Processes {
if p.Name == name {
return p.Stop()
}
}
return fmt.Errorf("process %s not found", name)
}

View File

@@ -0,0 +1,92 @@
package models
import (
"context"
"fmt"
"os"
"os/exec"
"strings"
"time"
)
type SupervisorProcess struct {
Name string `json:"name"`
ExecutePath string `json:"executePath"`
WorkingDirectory string `json:"workingDirectory,omitempty"`
StartDelay int `json:"startDelay,omitempty"`
Priority int `json:"priority,omitempty"`
Arguments []string `json:"arguments,omitempty"`
process *exec.Cmd `json:"-"`
cancel *context.CancelFunc `json:"-"`
Running bool `json:"-"`
}
func (p *SupervisorProcess) Start() error {
if p.process != nil && p.process.Process != nil {
return fmt.Errorf("process %s already running", p.Name)
}
ctx, cancel := context.WithCancel(context.Background())
p.cancel = &cancel
var args []string
for _, arg := range p.Arguments {
fields := strings.Fields(arg)
args = append(args, fields...)
}
cmd := exec.CommandContext(ctx, p.ExecutePath, args...)
cmd.Dir = p.WorkingDirectory
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
p.process = cmd
time.Sleep(time.Duration(p.StartDelay) * time.Millisecond)
if err := cmd.Start(); err != nil {
return fmt.Errorf("failed to start process %s: %w", p.Name, err)
}
go func() {
defer func() {
p.Running = false
}()
p.Running = true
err := cmd.Wait()
if err != nil {
fmt.Printf("Process %s exited with error: %v\n", p.Name, err)
} else {
fmt.Printf("Process %s exited successfully\n", p.Name)
}
}()
return nil
}
func (p *SupervisorProcess) Stop() error {
defer func() {
p.Running = false
}()
if p.process == nil || p.process.Process == nil {
return fmt.Errorf("process %s is not running", p.Name)
}
// Cancel the context
if p.cancel != nil {
(*p.cancel)()
}
err := p.process.Process.Kill()
if err != nil {
return fmt.Errorf("failed to kill process %s: %w", p.Name, err)
}
p.process = nil
p.cancel = nil
fmt.Printf("Process %s stopped.\n", p.Name)
return nil
}

View File

@@ -4,30 +4,13 @@
<meta charset="UTF-8"> <meta charset="UTF-8">
<title>Process Dashboard</title> <title>Process Dashboard</title>
<script src="https://unpkg.com/htmx.org@1.9.5"></script> <script src="https://unpkg.com/htmx.org@1.9.5"></script>
<style>
body {
font-family: sans-serif;
margin: 2rem;
}
.top-link {
display: block;
margin-bottom: 1rem;
font-size: 1.2rem;
}
ul#process-list {
list-style: none;
padding: 0;
}
ul#process-list li {
background: #f4f4f4;
margin: 0.5rem 0;
padding: 0.75rem;
border-radius: 5px;
}
</style>
</head> </head>
<body> <body>
<!-- Placeholder List for Future Processes -->
<ul id="process-list">
<div class="top-link-container"> <div class="top-link-container">
<a href="/taskmanager/htop" class="top-link" target="_blank" rel="noopener noreferrer"> <a href="/taskmanager/htop" class="top-link" target="_blank" rel="noopener noreferrer">
📈 Task Manager 📈 Task Manager
@@ -35,28 +18,160 @@
</div> </div>
<!-- Header --> <!-- Header -->
<h2>Future Processes</h2> <h2>Processes</h2>
{{ range .Supervisor.GetProcesses }}
<li>
<span class="process-name">📦
{{ if .Running }}
🟢 {{ .Name }}
{{ else }}
🔴 {{ .Name }}
{{ end }}
</span>
<!-- Placeholder List for Future Processes --> <div class="dropdown">
<ul id="process-list"> <button class="dot-button"></button>
<li>📌 Process A (placeholder)</li> <div class="dropdown-content">
<li>📌 Process B (placeholder)</li>
<li>📌 Process C (placeholder)</li>
</ul>
<!-- Example Button for Future HTMX Action -->
<button <button
hx-get="/get-latest-processes" hx-post="/start-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list" hx-target="#process-list"
hx-swap="innerHTML"> hx-swap="innerHTML">
🔄 Load Latest Processes ▶️ Start
</button> </button>
<button
hx-post="/stop-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list"
hx-swap="innerHTML">
⏹️ Stop
</button>
<button
hx-post="/restart-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list"
hx-swap="innerHTML">
🔁 Restart
</button>
</div>
</div>
</li>
{{ end }}
</ul>
</body> </body>
<style> <style>
body {
font-family: "Segoe UI", Roboto, sans-serif;
margin: 2rem;
background-color: #f7f9fb;
color: #222;
}
.top-link-container { .top-link-container {
text-align: center; text-align: center;
margin-bottom: 1rem; margin-bottom: 2rem;
}
.top-link {
font-size: 1.2rem;
text-decoration: none;
color: #4a90e2;
font-weight: 500;
}
.top-link:hover {
text-decoration: underline;
}
h2 {
text-align: center;
color: #444;
margin-bottom: 1.5rem;
}
ul#process-list {
list-style: none;
padding: 0;
max-width: 600px;
margin: 0 auto;
}
ul#process-list li {
background: #fff;
margin: 0.75rem 0;
padding: 1rem 1.25rem;
border-radius: 10px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.05);
display: flex;
justify-content: space-between;
align-items: center;
transition: box-shadow 0.25s ease;
}
ul#process-list li:hover {
box-shadow: 0 6px 16px rgba(0, 0, 0, 0.08);
}
.process-name {
font-weight: 600;
font-size: 1rem;
color: #2c3e50;
}
.dot-button {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #777;
padding: 0 4px;
transition: color 0.2s;
}
.dot-button:hover {
color: #333;
}
.dropdown {
position: relative;
display: inline-block;
}
.dropdown-content {
display: none;
position: absolute;
right: 0;
top: 1.8rem;
background-color: #fff;
min-width: 130px;
box-shadow: 0 6px 12px rgba(0,0,0,0.1);
border-radius: 6px;
overflow: hidden;
z-index: 100;
}
.dropdown:hover .dropdown-content {
display: block;
}
.dropdown-content button {
width: 100%;
padding: 10px 12px;
text-align: left;
background: none;
border: none;
cursor: pointer;
font-size: 0.95rem;
color: #333;
transition: background 0.2s, color 0.2s;
}
.dropdown-content button:hover {
background-color: #f0f0f0;
color: #000;
} }
</style> </style>
</html> </html>

156
templates/processes.html Normal file
View File

@@ -0,0 +1,156 @@
<ul id="process-list">
{{ range .Supervisor.GetProcesses }}
<li>
<span class="process-name">📦
{{ if .Running }}
🟢 {{ .Name }}
{{ else }}
🔴 {{ .Name }}
{{ end }}
</span>
<div class="dropdown">
<button class="dot-button"></button>
<div class="dropdown-content">
<button
hx-post="/start-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list"
hx-swap="innerHTML">
▶️ Start
</button>
<button
hx-post="/stop-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list"
hx-swap="innerHTML">
⏹️ Stop
</button>
<button
hx-post="/restart-process"
hx-vals='{"name": "{{ .Name }}"}'
hx-trigger="click"
hx-target="#process-list"
hx-swap="innerHTML">
🔁 Restart
</button>
</div>
</div>
</li>
{{ end }}
</ul>
<style>
body {
font-family: "Segoe UI", Roboto, sans-serif;
margin: 2rem;
background-color: #f7f9fb;
color: #222;
}
.top-link-container {
text-align: center;
margin-bottom: 2rem;
}
.top-link {
font-size: 1.2rem;
text-decoration: none;
color: #4a90e2;
font-weight: 500;
}
.top-link:hover {
text-decoration: underline;
}
h2 {
text-align: center;
color: #444;
margin-bottom: 1.5rem;
}
ul#process-list {
list-style: none;
padding: 0;
max-width: 600px;
margin: 0 auto;
}
ul#process-list li {
background: #fff;
margin: 0.75rem 0;
padding: 1rem 1.25rem;
border-radius: 10px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.05);
display: flex;
justify-content: space-between;
align-items: center;
transition: box-shadow 0.25s ease;
}
ul#process-list li:hover {
box-shadow: 0 6px 16px rgba(0, 0, 0, 0.08);
}
.process-name {
font-weight: 600;
font-size: 1rem;
color: #2c3e50;
}
.dot-button {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #777;
padding: 0 4px;
transition: color 0.2s;
}
.dot-button:hover {
color: #333;
}
.dropdown {
position: relative;
display: inline-block;
}
.dropdown-content {
display: none;
position: absolute;
right: 0;
top: 1.8rem;
background-color: #fff;
min-width: 130px;
box-shadow: 0 6px 12px rgba(0,0,0,0.1);
border-radius: 6px;
overflow: hidden;
z-index: 100;
}
.dropdown:hover .dropdown-content {
display: block;
}
.dropdown-content button {
width: 100%;
padding: 10px 12px;
text-align: left;
background: none;
border: none;
cursor: pointer;
font-size: 0.95rem;
color: #333;
transition: background 0.2s, color 0.2s;
}
.dropdown-content button:hover {
background-color: #f0f0f0;
color: #000;
}
</style>
</html>