1 Star 0 Fork 1

mysnapcore / mysnapd

forked from tupelo-shen / mysnapd 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
rest_api.go 9.66 KB
一键复制 编辑 原始数据 按行查看 历史
tupelo-shen 提交于 2022-11-08 21:39 . fix: usersession commit
// -*- Mode: Go; indent-tabs-mode: t -*-
/*
* Copyright (C) 2019 Canonical Ltd
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3 as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
package agent
import (
"encoding/json"
"fmt"
"mime"
"net/http"
"path/filepath"
"strings"
"sync"
"time"
"github.com/mvo5/goconfigparser"
"gitee.com/mysnapcore/mysnapd/desktop/notification"
"gitee.com/mysnapcore/mysnapd/dirs"
"gitee.com/mysnapcore/mysnapd/i18n"
"gitee.com/mysnapcore/mysnapd/systemd"
"gitee.com/mysnapcore/mysnapd/usersession/client"
)
var restApi = []*Command{
rootCmd,
sessionInfoCmd,
serviceControlCmd,
pendingRefreshNotificationCmd,
}
var (
rootCmd = &Command{
Path: "/",
GET: nil,
}
sessionInfoCmd = &Command{
Path: "/v1/session-info",
GET: sessionInfo,
}
serviceControlCmd = &Command{
Path: "/v1/service-control",
POST: postServiceControl,
}
pendingRefreshNotificationCmd = &Command{
Path: "/v1/notifications/pending-refresh",
POST: postPendingRefreshNotification,
}
finishRefreshNotificationCmd = &Command{
Path: "/v1/notifications/finish-refresh",
POST: postRefreshFinishedNotification,
}
)
func sessionInfo(c *Command, r *http.Request) Response {
m := map[string]interface{}{
"version": c.s.Version,
}
return SyncResponse(m)
}
type serviceInstruction struct {
Action string `json:"action"`
Services []string `json:"services"`
}
func serviceStart(inst *serviceInstruction, sysd systemd.Systemd) Response {
// Refuse to start non-snap services
for _, service := range inst.Services {
if !strings.HasPrefix(service, "snap.") {
return InternalError("cannot start non-snap service %v", service)
}
}
startErrors := make(map[string]string)
var started []string
for _, service := range inst.Services {
if err := sysd.Start([]string{service}); err != nil {
startErrors[service] = err.Error()
break
}
started = append(started, service)
}
// If we got any failures, attempt to stop the services we started.
stopErrors := make(map[string]string)
if len(startErrors) != 0 {
for _, service := range started {
if err := sysd.Stop([]string{service}); err != nil {
stopErrors[service] = err.Error()
}
}
}
if len(startErrors) == 0 {
return SyncResponse(nil)
}
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: "some user services failed to start",
Kind: errorKindServiceControl,
Value: map[string]interface{}{
"start-errors": startErrors,
"stop-errors": stopErrors,
},
},
})
}
func serviceStop(inst *serviceInstruction, sysd systemd.Systemd) Response {
// Refuse to stop non-snap services
for _, service := range inst.Services {
if !strings.HasPrefix(service, "snap.") {
return InternalError("cannot stop non-snap service %v", service)
}
}
stopErrors := make(map[string]string)
for _, service := range inst.Services {
if err := sysd.Stop([]string{service}); err != nil {
stopErrors[service] = err.Error()
}
}
if len(stopErrors) == 0 {
return SyncResponse(nil)
}
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: "some user services failed to stop",
Kind: errorKindServiceControl,
Value: map[string]interface{}{
"stop-errors": stopErrors,
},
},
})
}
func serviceDaemonReload(inst *serviceInstruction, sysd systemd.Systemd) Response {
if len(inst.Services) != 0 {
return InternalError("daemon-reload should not be called with any services")
}
if err := sysd.DaemonReload(); err != nil {
return InternalError("cannot reload daemon: %v", err)
}
return SyncResponse(nil)
}
var serviceInstructionDispTable = map[string]func(*serviceInstruction, systemd.Systemd) Response{
"start": serviceStart,
"stop": serviceStop,
"daemon-reload": serviceDaemonReload,
}
var systemdLock sync.Mutex
type noopReporter struct{}
func (noopReporter) Notify(string) {}
func validateJSONRequest(r *http.Request) (valid bool, errResp Response) {
contentType := r.Header.Get("Content-Type")
mediaType, params, err := mime.ParseMediaType(contentType)
if err != nil {
return false, BadRequest("cannot parse content type: %v", err)
}
if mediaType != "application/json" {
return false, BadRequest("unknown content type: %s", contentType)
}
charset := strings.ToUpper(params["charset"])
if charset != "" && charset != "UTF-8" {
return false, BadRequest("unknown charset in content type: %s", contentType)
}
return true, nil
}
func postServiceControl(c *Command, r *http.Request) Response {
if ok, resp := validateJSONRequest(r); !ok {
return resp
}
decoder := json.NewDecoder(r.Body)
var inst serviceInstruction
if err := decoder.Decode(&inst); err != nil {
return BadRequest("cannot decode request body into service instruction: %v", err)
}
impl := serviceInstructionDispTable[inst.Action]
if impl == nil {
return BadRequest("unknown action %s", inst.Action)
}
// Prevent multiple systemd actions from being carried out simultaneously
systemdLock.Lock()
defer systemdLock.Unlock()
sysd := systemd.New(systemd.UserMode, noopReporter{})
return impl(&inst, sysd)
}
func postPendingRefreshNotification(c *Command, r *http.Request) Response {
if ok, resp := validateJSONRequest(r); !ok {
return resp
}
decoder := json.NewDecoder(r.Body)
// pendingSnapRefreshInfo holds information about pending snap refresh provided by snapd.
var refreshInfo client.PendingSnapRefreshInfo
if err := decoder.Decode(&refreshInfo); err != nil {
return BadRequest("cannot decode request body into pending snap refresh info: %v", err)
}
// Note that since the connection is shared, we are not closing it.
if c.s.bus == nil {
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: "cannot connect to the session bus",
},
})
}
// TODO: this message needs to be crafted better as it's the only thing guaranteed to be delivered.
summary := fmt.Sprintf(i18n.G("Pending update of %q snap"), refreshInfo.InstanceName)
var urgencyLevel notification.Urgency
var body, icon string
var hints []notification.Hint
plzClose := i18n.G("Close the app to avoid disruptions")
if daysLeft := int(refreshInfo.TimeRemaining.Truncate(time.Hour).Hours() / 24); daysLeft > 0 {
urgencyLevel = notification.LowUrgency
body = fmt.Sprintf("%s (%s)", plzClose, fmt.Sprintf(
i18n.NG("%d day left", "%d days left", daysLeft), daysLeft))
} else if hoursLeft := int(refreshInfo.TimeRemaining.Truncate(time.Minute).Minutes() / 60); hoursLeft > 0 {
urgencyLevel = notification.NormalUrgency
body = fmt.Sprintf("%s (%s)", plzClose, fmt.Sprintf(
i18n.NG("%d hour left", "%d hours left", hoursLeft), hoursLeft))
} else if minutesLeft := int(refreshInfo.TimeRemaining.Truncate(time.Minute).Minutes()); minutesLeft > 0 {
urgencyLevel = notification.CriticalUrgency
body = fmt.Sprintf("%s (%s)", plzClose, fmt.Sprintf(
i18n.NG("%d minute left", "%d minutes left", minutesLeft), minutesLeft))
} else {
summary = fmt.Sprintf(i18n.G("Snap %q is refreshing now!"), refreshInfo.InstanceName)
urgencyLevel = notification.CriticalUrgency
}
hints = append(hints, notification.WithUrgency(urgencyLevel))
// The notification is provided by snapd session agent.
hints = append(hints, notification.WithDesktopEntry("io.snapcraft.SessionAgent"))
// But if we have a desktop file of the busy application, use that apps's icon.
if refreshInfo.BusyAppDesktopEntry != "" {
parser := goconfigparser.New()
desktopFilePath := filepath.Join(dirs.SnapDesktopFilesDir, refreshInfo.BusyAppDesktopEntry+".desktop")
if err := parser.ReadFile(desktopFilePath); err == nil {
icon, _ = parser.Get("Desktop Entry", "Icon")
}
}
msg := &notification.Message{
AppName: refreshInfo.BusyAppName,
Title: summary,
Icon: icon,
Body: body,
Hints: hints,
}
// TODO: silently ignore error returned when the notification server does not exist.
// TODO: track returned notification ID and respond to actions, if supported.
if err := c.s.notificationMgr.SendNotification(notification.ID(refreshInfo.InstanceName), msg); err != nil {
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: fmt.Sprintf("cannot send notification message: %v", err),
},
})
}
return SyncResponse(nil)
}
func postRefreshFinishedNotification(c *Command, r *http.Request) Response {
if ok, resp := validateJSONRequest(r); !ok {
return resp
}
decoder := json.NewDecoder(r.Body)
var finishRefresh client.FinishedSnapRefreshInfo
if err := decoder.Decode(&finishRefresh); err != nil {
return BadRequest("cannot decode request body into finish refresh notification info: %v", err)
}
// Note that since the connection is shared, we are not closing it.
if c.s.bus == nil {
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: "cannot connect to the session bus",
},
})
}
if err := c.s.notificationMgr.CloseNotification(notification.ID(finishRefresh.InstanceName)); err != nil {
return SyncResponse(&resp{
Type: ResponseTypeError,
Status: 500,
Result: &errorResult{
Message: fmt.Sprintf("cannot send close notification message: %v", err),
},
})
}
return SyncResponse(nil)
}
Go
1
https://gitee.com/mysnapcore/mysnapd.git
git@gitee.com:mysnapcore/mysnapd.git
mysnapcore
mysnapd
mysnapd
v0.1.0

搜索帮助