Add basic partition viewer

This commit is contained in:
Syed Daanish
2024-12-03 17:19:04 +03:00
parent 8d7bb207a7
commit 5c66d54c6d

324
main.go
View File

@@ -1,12 +1,38 @@
package main package main
import ( import (
"bytes"
"errors"
"fmt"
"log" "log"
"math"
"os/exec"
"regexp"
"strconv"
"strings"
"unicode/utf8"
"github.com/gdamore/tcell/v2" "github.com/gdamore/tcell/v2"
) )
var start int
var pline int
func main() { func main() {
/*
disk := "nvme0n1"
parts := getPartitions()
fmt.Println(parts)
for _, val := range parts {
x, y := get_used_p(disk+"p"+val[0], val[4])
if y != nil {
fmt.Println(disk+"p"+val[0], val[4], "--")
} else {
fmt.Println(disk+"p"+val[0], val[4], fmt.Sprintf("%.2f%%", x))
}
}
return
//*/
screen, err := tcell.NewScreen() screen, err := tcell.NewScreen()
if err != nil { if err != nil {
log.Fatalf("Error creating screen: %v", err) log.Fatalf("Error creating screen: %v", err)
@@ -19,12 +45,7 @@ func main() {
} }
screen.Clear() screen.Clear()
refresh(screen)
style := tcell.StyleDefault.Background(tcell.ColorBlack).Foreground(tcell.ColorWhite)
screen.SetContent(10, 5, 'H', nil, style)
screen.SetContent(11, 5, 'i', nil, style)
draw_box(screen, 10, 10, 40, 20, "")
draw_box(screen, 45, 10, 85, 20, "═║╔╚╗╝")
screen.Show() screen.Show()
for { for {
@@ -39,29 +60,286 @@ func main() {
} }
case *tcell.EventResize: case *tcell.EventResize:
screen.Sync() screen.Sync()
screen.Clear()
refresh(screen)
screen.Show()
} }
} }
} }
func draw_box(s tcell.Screen, x1, y1, x2, y2 int, chars_i string) { func refresh(screen tcell.Screen) {
start = 0
pline = 12
// style := tcell.StyleDefault.Background(tcell.GetColor("#00000000")).Foreground(tcell.GetColor("#FF8040"))
width, _ := screen.Size()
// screen.SetContent(10, 5, 'H', nil, style)
// screen.SetContent(11, 5, 'i', nil, style)
// draw_box(screen, 0, 0, width-1, height-1, style, "═║╔╚╗╝")
// draw_box(screen, 45, 10, 85, 20, style, "═║╔╚╗╝")
disk := "nvme0n1"
parts := getPartitions()
empty_parts := 0
for _, val := range parts {
size, err := strconv.ParseFloat(strings.TrimRight(val[3], "GB"), 64)
if err != nil {
fmt.Println("Error parsing used space:", err)
return
}
if int(math.Round((size/512)*100)) == 0 {
empty_parts++
}
}
for _, val := range parts {
x, y := get_used_p(disk+"p"+val[0], val[4])
if y != nil {
// fmt.Println(disk+"p"+val[0], val[4], "--")
size, err := strconv.ParseFloat(strings.TrimRight(val[3], "GB"), 64)
if err != nil {
fmt.Println("Error parsing used space:", err)
return
}
print_it(screen, fmt.Sprint(((size / 512) * 100)))
draw_it(screen, ((size / 512) * 100), 0, width, val[4])
} else {
// fmt.Println(disk+"p"+val[0], val[4], fmt.Sprintf("%.2f%%", x))
// print_it(screen, fmt.Sprintf("%.2f%%", x)+":hello:"+fmt.Sprintf("%d", int(math.Round(x))), style)
size, err := strconv.ParseFloat(strings.TrimRight(val[3], "GB"), 64)
if err != nil {
fmt.Println("Error parsing used space:", err)
return
}
print_it(screen, fmt.Sprint(float64(width)/100))
draw_it(screen, ((size/512)*100)-float64(empty_parts)/(float64(width)/86), int(math.Round(x)), width, val[4])
}
}
}
func getPartitions() [][]string {
// Run the parted command
cmd := exec.Command("sudo", "parted", "-s", "/dev/nvme0n1", "unit", "GB", "print") // Replace "/dev/sda" with the target disk
output, err := cmd.CombinedOutput()
if err != nil {
log.Fatalf("Error executing command: %v", err)
}
// Convert output to string
outputStr := string(output)
// Regular expression to extract partition information
// Example format: " 1 1049MB 100GB 100GB ext4"
re := regexp.MustCompile(`\s*(\d+)\s+(\S+)\s+(\S+)\s+(\S+)\s+(\S+)`)
matches := re.FindAllStringSubmatch(outputStr, -1)
x := [][]string{}
// Iterate through matches and extract partition details
for _, match := range matches {
// match[0] is the full match, match[1] is the partition number, and match[2] to match[5] are the other columns
vals := []string{match[1], match[2], match[3], match[4], match[5]}
x = append(x, vals)
}
return x
}
func get_used_p(part, tip string) (float64, error) { // `tip` is turkish for type
if tip == "ntfs" {
cmd1 := exec.Command("sudo", "ntfsinfo", "-m", "/dev/"+part)
output1, err := cmd1.Output()
if err != nil {
log.Fatal(err)
}
// Run the second command (grep Free) with input from the first command
cmd2 := exec.Command("grep", "Free")
cmd2.Stdin = bytes.NewReader(output1)
output2, err2 := cmd2.Output()
if err2 != nil {
log.Fatal(err2)
}
// Run the third command (awk) with input from the second command
cmd3 := exec.Command("awk", "{match($4, /\\(([0-9]+(\\.[0-9]+)?)%\\)/, arr); print arr[1]}")
cmd3.Stdin = bytes.NewReader(output2)
finalOutput, err3 := cmd3.Output()
if err3 != nil {
log.Fatal(err3)
}
// Print the final output
val, err4 := strconv.ParseFloat(strings.TrimSpace(string(finalOutput)), 64)
if err4 != nil {
log.Fatal(err4)
}
return math.Round((100-val)*100) / 100, nil
} else if tip == "ext4" {
// echo "scale=5; 100 - (" (sudo tune2fs -l /dev/nvme0n1p5 | grep "Free blocks" | awk '{ print $3 }') "/" (sudo tune2fs -l /dev/nvme0n1p5 | grep "Block count" | awk '{ print $3 }') ") * 100" | bc
cmd1 := exec.Command("sudo", "tune2fs", "-l", "/dev/"+part)
output1, err := cmd1.Output()
if err != nil {
log.Fatal(err)
}
//fmt.Println(string(output1))
lines := strings.Split(string(output1), "\n")
needed := ""
// Find the line starting with "Free blocks:"
for _, line := range lines {
if strings.HasPrefix(line, "Free blocks:") {
parts := strings.Split(line, ":")
if len(parts) == 2 {
number := strings.TrimSpace(parts[1]) // Remove any leading/trailing spaces
needed = number // Print the number
}
break
}
}
// Print the final output
val_free, err4 := strconv.ParseFloat(strings.TrimSpace(string(needed)), 64)
cmd4 := exec.Command("sudo", "tune2fs", "-l", "/dev/"+part)
output4, err5 := cmd4.Output()
if err5 != nil {
log.Fatal(err5)
}
lines2 := strings.Split(string(output4), "\n")
needed2 := ""
// Find the line starting with "Free blocks:"
for _, line := range lines2 {
if strings.HasPrefix(line, "Block count:") {
parts := strings.Split(line, ":")
if len(parts) == 2 {
number := strings.TrimSpace(parts[1]) // Remove any leading/trailing spaces
needed2 = number // Print the number
}
break
}
}
val_total, err4 := strconv.ParseFloat(strings.TrimSpace(string(needed2)), 64)
if err4 != nil {
log.Fatal(err4)
}
//fmt.Println(val_free, val_total, val_free/val_total)
return math.Round((100-(val_free/val_total)*100)*100) / 100, nil
} else {
return 0, errors.New("Cannot parse type: " + tip)
}
}
func draw_it(s tcell.Screen, percent float64, filled, width int, tip string) {
chars_i := "🬂▐🬕🬲🬨🬷▌🬭"
// chars_i := "─│╭╰╮╯"
x := start + int((float64(percent)/100)*float64(width-2))
if x >= width {
x = width - 1
} else if (x - start) < 2 {
x = start + 1
}
colors := map[string]string{
"ntfs": "#90F2D1",
"fat32": "#46a046",
"ext4": "#314e6c",
"linux-swap(v1)": "#c1665a",
}
color, exists := colors[tip]
var style tcell.Style
var style_box tcell.Style
if exists {
style = tcell.StyleDefault.Background(tcell.GetColor("#cfc174")).Foreground(tcell.GetColor(color))
style_box = tcell.StyleDefault.Background(tcell.GetColor("#00000000")).Foreground(tcell.GetColor(color))
} else {
style = tcell.StyleDefault.Background(tcell.GetColor("#cfc174")).Foreground(tcell.GetColor("#555555"))
style_box = tcell.StyleDefault.Background(tcell.GetColor("#00000000")).Foreground(tcell.GetColor("#555555"))
}
// print_it(s, tip, style)
draw_box(s, start, 2, x, 5, style_box, chars_i)
for y := 2; y <= 5; y++ {
for x1 := start + 1; x1 < start+int((float64(filled)/100)*((float64(percent)/100)*float64(width))); x1++ {
char, _, _, _ := s.GetContent(x1, y)
if char == '\x00' {
char = ' '
}
s.SetContent(x1, y, char, nil, style)
}
if filled > 0 {
char, _, _, _ := s.GetContent(start, y)
s.SetContent(start, y, char, nil, style)
}
if filled > 99 {
char, _, _, _ := s.GetContent(x, y)
s.SetContent(x, y, char, nil, style)
}
// s.SetContent(start+1, y, ' ', nil, style.Reverse(true))
// s.SetContent(x-1, y, ' ', nil, style.Reverse(true))
}
start = x + 2
}
func print_it(s tcell.Screen, str string) {
for i, char := range str {
s.SetContent(10+i, pline, char, nil, tcell.StyleDefault.Background(tcell.GetColor("#00000000")).Foreground(tcell.GetColor("#FF8040")))
}
pline++
}
func draw_box(s tcell.Screen, x1, y1, x2, y2 int, style tcell.Style, chars_i string) {
if chars_i == "" { if chars_i == "" {
chars_i = "─│╭╰╮╯" // "━┃┏┗┓┛" chars_i = "─│╭╰╮╯" // "━┃┏┗┓┛"
} }
chars := []rune(chars_i) // print_it(s, fmt.Sprintf("Result: %d", utf8.RuneCountInString(chars_i)), style)
for x := x1; x <= x2; x++ { if utf8.RuneCountInString(chars_i) == 6 {
s.SetContent(x, y1, chars[0], nil, tcell.StyleDefault) chars := []rune(chars_i)
for x := x1; x <= x2; x++ {
s.SetContent(x, y1, chars[0], nil, style)
}
for x := x1; x <= x2; x++ {
s.SetContent(x, y2, chars[0], nil, style)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x1, x, chars[1], nil, style)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x2, x, chars[1], nil, style)
}
s.SetContent(x1, y1, chars[2], nil, style)
s.SetContent(x1, y2, chars[3], nil, style)
s.SetContent(x2, y1, chars[4], nil, style)
s.SetContent(x2, y2, chars[5], nil, style)
} else {
chars := []rune(chars_i)
for x := x1; x <= x2; x++ {
s.SetContent(x, y1, chars[0], nil, style)
}
for x := x1; x <= x2; x++ {
s.SetContent(x, y2, chars[7], nil, style)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x1, x, chars[6], nil, style)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x2, x, chars[1], nil, style)
}
s.SetContent(x1, y1, chars[2], nil, style)
s.SetContent(x1, y2, chars[3], nil, style)
s.SetContent(x2, y1, chars[4], nil, style)
s.SetContent(x2, y2, chars[5], nil, style)
} }
for x := x1; x <= x2; x++ {
s.SetContent(x, y2, chars[0], nil, tcell.StyleDefault)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x1, x, chars[1], nil, tcell.StyleDefault)
}
for x := y1 + 1; x < y2; x++ {
s.SetContent(x2, x, chars[1], nil, tcell.StyleDefault)
}
s.SetContent(x1, y1, chars[2], nil, tcell.StyleDefault)
s.SetContent(x1, y2, chars[3], nil, tcell.StyleDefault)
s.SetContent(x2, y1, chars[4], nil, tcell.StyleDefault)
s.SetContent(x2, y2, chars[5], nil, tcell.StyleDefault)
} }