bedrocktool/subcommands/skins/skins.go

269 lines
6.7 KiB
Go
Raw Normal View History

2022-09-04 14:26:32 +00:00
package skins
2022-03-05 11:59:36 +00:00
import (
2022-03-05 12:59:11 +00:00
"bytes"
2022-03-05 11:59:36 +00:00
"context"
"encoding/json"
"flag"
"fmt"
"image"
"image/png"
2022-03-05 12:59:11 +00:00
"io"
2022-03-05 11:59:36 +00:00
"os"
2022-03-05 12:59:11 +00:00
"path"
2022-03-05 11:59:36 +00:00
"strings"
2022-09-04 14:53:21 +00:00
"github.com/bedrock-tool/bedrocktool/utils"
2022-09-04 14:26:32 +00:00
2022-08-25 15:56:03 +00:00
"github.com/flytam/filenamify"
"github.com/google/subcommands"
2022-03-05 11:59:36 +00:00
"github.com/sandertv/gophertunnel/minecraft"
"github.com/sandertv/gophertunnel/minecraft/protocol"
"github.com/sandertv/gophertunnel/minecraft/protocol/packet"
2022-09-04 00:24:58 +00:00
"github.com/sirupsen/logrus"
2022-03-05 11:59:36 +00:00
)
type Skin struct {
protocol.Skin
2022-03-05 11:59:36 +00:00
}
2022-08-17 18:04:13 +00:00
// WriteGeometry writes the geometry json for the skin to output_path
func (skin *Skin) WriteGeometry(output_path string) error {
2022-08-17 18:04:13 +00:00
f, err := os.Create(output_path)
2022-03-05 12:59:11 +00:00
if err != nil {
return fmt.Errorf("failed to write Geometry %s: %s", output_path, err)
2022-03-05 12:59:11 +00:00
}
defer f.Close()
io.Copy(f, bytes.NewReader(skin.SkinGeometry))
return nil
2022-03-05 12:59:11 +00:00
}
2022-08-17 18:04:13 +00:00
// WriteCape writes the cape as a png at output_path
func (skin *Skin) WriteCape(output_path string) error {
2022-08-17 18:04:13 +00:00
f, err := os.Create(output_path)
2022-03-05 12:59:11 +00:00
if err != nil {
return fmt.Errorf("failed to write Cape %s: %s", output_path, err)
2022-03-05 12:59:11 +00:00
}
defer f.Close()
cape_tex := image.NewRGBA(image.Rect(0, 0, int(skin.CapeImageWidth), int(skin.CapeImageHeight)))
cape_tex.Pix = skin.CapeData
if err := png.Encode(f, cape_tex); err != nil {
return fmt.Errorf("error writing skin: %s", err)
2022-03-05 12:59:11 +00:00
}
return nil
2022-03-05 12:59:11 +00:00
}
2022-08-17 18:04:13 +00:00
// WriteAnimations writes skin animations to the folder
func (skin *Skin) WriteAnimations(output_path string) error {
2022-09-04 00:24:58 +00:00
logrus.Warnf("%s has animations (unimplemented)", output_path)
2022-08-17 18:04:13 +00:00
return nil
2022-03-05 12:59:11 +00:00
}
2022-08-17 18:04:13 +00:00
// WriteTexture writes the main texture for this skin to a file
func (skin *Skin) WriteTexture(output_path string) error {
2022-08-17 18:04:13 +00:00
f, err := os.Create(output_path)
if err != nil {
return fmt.Errorf("error writing Texture: %s", err)
2022-03-05 13:39:46 +00:00
}
defer f.Close()
skin_tex := image.NewRGBA(image.Rect(0, 0, int(skin.SkinImageWidth), int(skin.SkinImageHeight)))
skin_tex.Pix = skin.SkinData
if err := png.Encode(f, skin_tex); err != nil {
return fmt.Errorf("error writing Texture: %s", err)
}
return nil
}
2022-08-23 20:37:04 +00:00
func (skin *Skin) WriteTint(output_path string) error {
f, err := os.Create(output_path)
if err != nil {
return fmt.Errorf("failed to write Tint %s: %s", output_path, err)
}
defer f.Close()
err = json.NewEncoder(f).Encode(skin.PieceTintColours)
if err != nil {
return fmt.Errorf("failed to write Tint %s: %s", output_path, err)
}
return nil
}
func (skin *Skin) WriteMeta(output_path string) error {
f, err := os.Create(output_path)
if err != nil {
return fmt.Errorf("failed to write Tint %s: %s", output_path, err)
}
defer f.Close()
d, err := json.MarshalIndent(struct {
SkinID string
PlayFabID string
PremiumSkin bool
PersonaSkin bool
CapeID string
SkinColour string
ArmSize string
Trusted bool
PersonaPieces []protocol.PersonaPiece
}{
skin.SkinID,
skin.PlayFabID,
skin.PremiumSkin,
skin.PersonaSkin,
skin.CapeID,
skin.SkinColour,
skin.ArmSize,
skin.Trusted,
skin.PersonaPieces,
}, "", " ")
if err != nil {
return err
}
f.Write(d)
return nil
}
2022-08-17 18:04:13 +00:00
// Write writes all data for this skin to a folder
func (skin *Skin) Write(output_path, name string) error {
2022-08-25 15:56:03 +00:00
name, _ = filenamify.FilenamifyV2(name)
skin_dir := path.Join(output_path, name)
2022-08-17 18:04:13 +00:00
2022-08-23 20:37:04 +00:00
have_geometry, have_cape, have_animations, have_tint := len(skin.SkinGeometry) > 0, len(skin.CapeData) > 0, len(skin.Animations) > 0, len(skin.PieceTintColours) > 0
2022-08-17 18:04:13 +00:00
2022-09-03 22:56:40 +00:00
os.MkdirAll(skin_dir, 0o755)
2022-08-23 20:37:04 +00:00
if have_geometry {
if err := skin.WriteGeometry(path.Join(skin_dir, "geometry.json")); err != nil {
return err
}
2022-08-23 20:37:04 +00:00
}
if have_cape {
if err := skin.WriteCape(path.Join(skin_dir, "cape.png")); err != nil {
return err
}
2022-08-23 20:37:04 +00:00
}
if have_animations {
if err := skin.WriteAnimations(skin_dir); err != nil {
return err
}
}
if have_tint {
if err := skin.WriteTint(path.Join(skin_dir, "tint.json")); err != nil {
return err
}
2022-03-05 11:59:36 +00:00
}
2022-08-23 20:37:04 +00:00
if err := skin.WriteMeta(path.Join(skin_dir, "metadata.json")); err != nil {
return err
}
return skin.WriteTexture(path.Join(skin_dir, "skin.png"))
}
2022-08-17 18:04:13 +00:00
// puts the skin at output_path if the filter matches it
// internally converts the struct so it can use the extra methods
func write_skin(output_path, name string, skin protocol.Skin, filter string) {
if !strings.HasPrefix(name, filter) {
return
}
2022-09-04 00:24:58 +00:00
logrus.Infof("Writing skin for %s\n", name)
_skin := &Skin{skin}
if err := _skin.Write(output_path, name); err != nil {
fmt.Fprintf(os.Stderr, "Error writing skin: %s\n", err)
}
}
2022-09-03 22:56:40 +00:00
var (
skin_players = make(map[string]string)
skin_player_counts = make(map[string]int)
processed_skins = make(map[string]bool)
)
2022-03-18 18:22:31 +00:00
2022-08-17 18:04:13 +00:00
func process_packet_skins(conn *minecraft.Conn, out_path string, pk packet.Packet, filter string) {
2022-03-18 18:22:31 +00:00
switch _pk := pk.(type) {
case *packet.PlayerSkin:
name := skin_players[_pk.UUID.String()]
if name == "" {
name = _pk.UUID.String()
}
skin_player_counts[name]++
name = fmt.Sprintf("%s_%d", name, skin_player_counts[name])
2022-08-17 18:04:13 +00:00
write_skin(out_path, name, _pk.Skin, filter)
2022-03-18 18:22:31 +00:00
case *packet.PlayerList:
if _pk.ActionType == 1 { // remove
return
}
for _, player := range _pk.Entries {
2022-09-04 14:26:32 +00:00
name := utils.CleanupName(player.Username)
2022-03-18 18:22:31 +00:00
if name == "" {
name = player.UUID.String()
}
2022-03-27 14:09:46 +00:00
if _, ok := processed_skins[name]; ok {
continue
}
2022-08-17 18:04:13 +00:00
write_skin(out_path, name, player.Skin, filter)
2022-03-18 18:22:31 +00:00
skin_players[player.UUID.String()] = name
2022-03-27 14:09:46 +00:00
processed_skins[name] = true
if conn != nil {
2022-09-04 14:26:32 +00:00
(&utils.ProxyContext{Client: conn}).SendPopup(fmt.Sprintf("%s Skin was Saved", name))
2022-03-27 14:09:46 +00:00
}
2022-03-18 18:22:31 +00:00
}
}
}
type SkinCMD struct {
server_address string
filter string
}
2022-07-29 16:12:06 +00:00
func (*SkinCMD) Name() string { return "skins" }
func (*SkinCMD) Synopsis() string { return "download all skins from players on a server" }
2022-07-29 16:12:06 +00:00
func (c *SkinCMD) SetFlags(f *flag.FlagSet) {
f.StringVar(&c.server_address, "address", "", "remote server address")
f.StringVar(&c.filter, "filter", "", "player name filter prefix")
}
2022-09-03 22:56:40 +00:00
func (c *SkinCMD) Usage() string {
return c.Name() + ": " + c.Synopsis() + "\n"
}
2022-03-05 11:59:36 +00:00
func (c *SkinCMD) Execute(ctx context.Context, f *flag.FlagSet, _ ...interface{}) subcommands.ExitStatus {
2022-09-04 14:26:32 +00:00
address, hostname, err := utils.ServerInput(c.server_address)
2022-08-13 14:30:46 +00:00
if err != nil {
fmt.Fprint(os.Stderr, err)
return 1
2022-08-13 14:30:46 +00:00
}
2022-09-04 14:26:32 +00:00
serverConn, err := utils.ConnectServer(ctx, address, nil, false, nil)
2022-03-05 11:59:36 +00:00
if err != nil {
fmt.Fprint(os.Stderr, err)
return 1
2022-03-05 11:59:36 +00:00
}
2022-03-18 18:22:31 +00:00
defer serverConn.Close()
2022-03-05 11:59:36 +00:00
2022-04-16 11:02:32 +00:00
out_path := fmt.Sprintf("skins/%s", hostname)
2022-03-05 11:59:36 +00:00
if err := serverConn.DoSpawnContext(ctx); err != nil {
fmt.Fprint(os.Stderr, err)
return 1
2022-03-05 11:59:36 +00:00
}
2022-09-04 00:24:58 +00:00
logrus.Info("Connected")
logrus.Info("Press ctrl+c to exit")
2022-03-05 11:59:36 +00:00
2022-09-03 22:56:40 +00:00
os.MkdirAll(out_path, 0o755)
2022-03-05 11:59:36 +00:00
for {
pk, err := serverConn.ReadPacket()
if err != nil {
return 1
}
2022-08-17 18:04:13 +00:00
process_packet_skins(nil, out_path, pk, c.filter)
2022-03-05 11:59:36 +00:00
}
}
func init() {
2022-09-04 14:26:32 +00:00
utils.RegisterCommand(&SkinCMD{})
}