feat: mute and unmute statuses

This commit adds support for muting and unmuting statuses. When viewing
a status the user can now see whether they've muted the status or not.
A status can only be muted by the user if they own it or are mentioned
in it.

PR: apollo/enbas#47
Resolves: apollo/enbas#46
This commit is contained in:
Dan Anglin 2024-08-16 14:42:57 +01:00
parent 42f54c6020
commit 3037af60ed
Signed by: dananglin
GPG key ID: 0C1D44CFBEE68638
7 changed files with 157 additions and 4 deletions

View file

@ -604,11 +604,29 @@ enbas show --type liked
### Mute a status
_Not yet supported_
Mutes a status in order to stop receiving future notifications for replies, likes, boosts, etc.
```
enbas mute --type status --status-id 01J56ZJAGEWG967GS1EK0TV3GA
```
| flag | type | required | description | default |
|------|------|----------|-------------|---------|
| `type` | string | true | The resource you want to mute.<br>Here this should be `status`. | |
| `status-id` | string | true | The ID of the status that you want to mute. | |
### Unmute a status
_Not yet supported_
Unmute a status in order to start receiving future notification from the status' thread.
```
enbas mute --type status --status-id 01J56ZJAGEWG967GS1EK0TV3GA
```
| flag | type | required | description | default |
|------|------|----------|-------------|---------|
| `type` | string | true | The resource you want to unmute.<br>Here this should be `status`. | |
| `status-id` | string | true | The ID of the status that you want to unmute. | |
### Vote in a poll within a status

View file

@ -241,7 +241,7 @@ func (g *Client) ReblogStatus(statusID string) error {
if err := g.sendRequest(params); err != nil {
return fmt.Errorf(
"received an error after sending the request to reblog the status; %w",
"received an error after sending the request to reblog the status: %w",
err,
)
}
@ -262,7 +262,49 @@ func (g *Client) UnreblogStatus(statusID string) error {
if err := g.sendRequest(params); err != nil {
return fmt.Errorf(
"received an error after sending the request to un-reblog the status; %w",
"received an error after sending the request to un-reblog the status: %w",
err,
)
}
return nil
}
func (g *Client) MuteStatus(statusID string) error {
url := g.Authentication.Instance + baseStatusesPath + "/" + statusID + "/mute"
params := requestParameters{
httpMethod: http.MethodPost,
url: url,
requestBody: nil,
contentType: "",
output: nil,
}
if err := g.sendRequest(params); err != nil {
return fmt.Errorf(
"received an error after sending the request to mute the status: %w",
err,
)
}
return nil
}
func (g *Client) UnmuteStatus(statusID string) error {
url := g.Authentication.Instance + baseStatusesPath + "/" + statusID + "/unmute"
params := requestParameters{
httpMethod: http.MethodPost,
url: url,
requestBody: nil,
contentType: "",
output: nil,
}
if err := g.sendRequest(params); err != nil {
return fmt.Errorf(
"received an error after sending the request to unmute the status: %w",
err,
)
}

View file

@ -347,6 +347,7 @@ type MuteExecutor struct {
accountName internalFlag.StringSliceValue
muteDuration internalFlag.TimeDurationValue
muteNotifications bool
statusID string
resourceType string
}
@ -367,6 +368,7 @@ func NewMuteExecutor(
exe.Var(&exe.accountName, "account-name", "The name of the account")
exe.Var(&exe.muteDuration, "mute-duration", "Specify how long the mute should last for. To mute indefinitely, set this to 0s")
exe.BoolVar(&exe.muteNotifications, "mute-notifications", false, "Set to true to mute notifications as well as posts")
exe.StringVar(&exe.statusID, "status-id", "", "The ID of the status")
exe.StringVar(&exe.resourceType, "type", "", "The type of resource you want to action on (e.g. account, status)")
return &exe
@ -591,6 +593,7 @@ type UnmuteExecutor struct {
printer *printer.Printer
config *config.Config
accountName internalFlag.StringSliceValue
statusID string
resourceType string
}
@ -608,6 +611,7 @@ func NewUnmuteExecutor(
exe.Usage = usage.ExecutorUsageFunc("unmute", "Umutes a specific resource (e.g. an account)", exe.FlagSet)
exe.Var(&exe.accountName, "account-name", "The name of the account")
exe.StringVar(&exe.statusID, "status-id", "", "The ID of the status")
exe.StringVar(&exe.resourceType, "type", "", "The type of resource you want to action on (e.g. account, status)")
return &exe

View file

@ -1,6 +1,7 @@
package executor
import (
"errors"
"fmt"
"codeflow.dananglin.me.uk/apollo/enbas/internal/client"
@ -9,6 +10,7 @@ import (
func (m *MuteExecutor) Execute() error {
funcMap := map[string]func(*client.Client) error{
resourceAccount: m.muteAccount,
resourceStatus: m.muteStatus,
}
doFunc, ok := funcMap[m.resourceType]
@ -43,3 +45,44 @@ func (m *MuteExecutor) muteAccount(gtsClient *client.Client) error {
return nil
}
func (m *MuteExecutor) muteStatus(gtsClient *client.Client) error {
if m.statusID == "" {
return FlagNotSetError{flagText: flagStatusID}
}
status, err := gtsClient.GetStatus(m.statusID)
if err != nil {
return fmt.Errorf("unable to retrieve the status: %w", err)
}
myAccountID, err := getAccountID(gtsClient, true, nil)
if err != nil {
return fmt.Errorf("unable to get your account ID: %w", err)
}
canMute := false
if status.Account.ID == myAccountID {
canMute = true
} else {
for _, mention := range status.Mentions {
if mention.ID == myAccountID {
canMute = true
break
}
}
}
if !canMute {
return errors.New("unable to mute the status because you are not the owner and you are not mentioned in it")
}
if err := gtsClient.MuteStatus(m.statusID); err != nil {
return fmt.Errorf("unable to mute the status: %w", err)
}
m.printer.PrintSuccess("Successfully muted the status.")
return nil
}

View file

@ -1,6 +1,7 @@
package executor
import (
"errors"
"fmt"
"codeflow.dananglin.me.uk/apollo/enbas/internal/client"
@ -9,6 +10,7 @@ import (
func (m *UnmuteExecutor) Execute() error {
funcMap := map[string]func(*client.Client) error{
resourceAccount: m.unmuteAccount,
resourceStatus: m.unmuteStatus,
}
doFunc, ok := funcMap[m.resourceType]
@ -38,3 +40,44 @@ func (m *UnmuteExecutor) unmuteAccount(gtsClient *client.Client) error {
return nil
}
func (m *UnmuteExecutor) unmuteStatus(gtsClient *client.Client) error {
if m.statusID == "" {
return FlagNotSetError{flagText: flagStatusID}
}
status, err := gtsClient.GetStatus(m.statusID)
if err != nil {
return fmt.Errorf("unable to retrieve the status: %w", err)
}
myAccountID, err := getAccountID(gtsClient, true, nil)
if err != nil {
return fmt.Errorf("unable to get your account ID: %w", err)
}
canUnmute := false
if status.Account.ID == myAccountID {
canUnmute = true
} else {
for _, mention := range status.Mentions {
if mention.ID == myAccountID {
canUnmute = true
break
}
}
}
if !canUnmute {
return errors.New("unable to unmute the status because you are not the owner and you are not mentioned in it")
}
if err := gtsClient.UnmuteStatus(m.statusID); err != nil {
return fmt.Errorf("unable to unmute the status: %w", err)
}
m.printer.PrintSuccess("Successfully unmuted the status.")
return nil
}

View file

@ -66,6 +66,7 @@ func (p Printer) PrintStatus(status model.Status, userAccountID string) {
builder.WriteString("\n" + p.fieldFormat("Boosted: ") + strconv.FormatBool(status.Reblogged))
builder.WriteString("\n" + p.fieldFormat("Liked: ") + strconv.FormatBool(status.Favourited))
builder.WriteString("\n" + p.fieldFormat("Bookmarked: ") + strconv.FormatBool(status.Bookmarked))
builder.WriteString("\n" + p.fieldFormat("Muted: ") + strconv.FormatBool(status.Muted))
// Status visibility
builder.WriteString("\n\n" + p.headerFormat("VISIBILITY:"))

View file

@ -335,6 +335,7 @@
{ "flag": "account-name" },
{ "flag": "mute-duration" },
{ "flag": "mute-notifications", "default": "false" },
{ "flag": "status-id", "fieldName": "statusID", "default": "" },
{ "flag": "type", "fieldName": "resourceType", "default": "" }
],
"summary": "Mutes a specific resource (e.g. an account)",
@ -427,6 +428,7 @@
"additionalFields": [],
"flags": [
{ "flag": "account-name" },
{ "flag": "status-id", "fieldName": "statusID", "default": "" },
{ "flag": "type", "fieldName": "resourceType", "default": "" }
],
"summary": "Umutes a specific resource (e.g. an account)",