代码拉取完成,页面将自动刷新
同步操作将从 tupelo-shen/mysnapd 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
// -*- Mode: Go; indent-tabs-mode: t -*-
/*
* Copyright (C) 2015-2016 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 client
import (
"bufio"
"bytes"
"context"
"encoding/json"
"errors"
"fmt"
"net/url"
"strconv"
"strings"
"time"
"gitee.com/mysnapcore/mysnapd/snap"
)
// AppActivator is a thing that activates the app that is a service in the
// system.
type AppActivator struct {
Name string
// Type describes the type of the unit, either timer or socket
Type string
Active bool
Enabled bool
}
// AppInfo describes a single snap application.
type AppInfo struct {
Snap string `json:"snap,omitempty"`
Name string `json:"name"`
DesktopFile string `json:"desktop-file,omitempty"`
Daemon string `json:"daemon,omitempty"`
DaemonScope snap.DaemonScope `json:"daemon-scope,omitempty"`
Enabled bool `json:"enabled,omitempty"`
Active bool `json:"active,omitempty"`
CommonID string `json:"common-id,omitempty"`
Activators []AppActivator `json:"activators,omitempty"`
}
// IsService returns true if the application is a background daemon.
func (a *AppInfo) IsService() bool {
if a == nil {
return false
}
if a.Daemon == "" {
return false
}
return true
}
// AppOptions represent the options of the Apps call.
type AppOptions struct {
// If Service is true, only return apps that are services
// (app.IsService() is true); otherwise, return all.
Service bool
}
// Apps returns information about all matching apps. Each name can be
// either a snap or a snap.app. If names is empty, list all (that
// satisfy opts).
func (client *Client) Apps(names []string, opts AppOptions) ([]*AppInfo, error) {
q := make(url.Values)
if len(names) > 0 {
q.Add("names", strings.Join(names, ","))
}
if opts.Service {
q.Add("select", "service")
}
var appInfos []*AppInfo
_, err := client.doSync("GET", "/v2/apps", q, nil, nil, &appInfos)
return appInfos, err
}
// LogOptions represent the options of the Logs call.
type LogOptions struct {
N int // The maximum number of log lines to retrieve initially. If <0, no limit.
Follow bool // Whether to continue returning new lines as they appear
}
// A Log holds the information of a single syslog entry
type Log struct {
Timestamp time.Time `json:"timestamp"` // Timestamp of the event, in RFC3339 format to µs precision.
Message string `json:"message"` // The log message itself
SID string `json:"sid"` // The syslog identifier
PID string `json:"pid"` // The process identifier
}
// String will format the log entry with the timestamp in the local timezone
func (l Log) String() string {
return l.fmtLog(time.Local)
}
// StringInUTC will format the log entry with the timestamp in UTC
func (l Log) StringInUTC() string {
return l.fmtLog(time.UTC)
}
func (l Log) fmtLog(timezone *time.Location) string {
if timezone == nil {
timezone = time.Local
}
return fmt.Sprintf("%s %s[%s]: %s", l.Timestamp.In(timezone).Format(time.RFC3339), l.SID, l.PID, l.Message)
}
// Logs asks for the logs of a series of services, by name.
func (client *Client) Logs(names []string, opts LogOptions) (<-chan Log, error) {
query := url.Values{}
if len(names) > 0 {
query.Set("names", strings.Join(names, ","))
}
query.Set("n", strconv.Itoa(opts.N))
if opts.Follow {
query.Set("follow", strconv.FormatBool(opts.Follow))
}
rsp, err := client.raw(context.Background(), "GET", "/v2/logs", query, nil, nil)
if err != nil {
return nil, err
}
if rsp.StatusCode != 200 {
var r response
defer rsp.Body.Close()
if err := decodeInto(rsp.Body, &r); err != nil {
return nil, err
}
return nil, r.err(client, rsp.StatusCode)
}
ch := make(chan Log, 20)
go func() {
// logs come in application/json-seq, described in RFC7464: it's
// a series of <RS><arbitrary, valid JSON><LF>. Decoders are
// expected to skip invalid or truncated or empty records.
scanner := bufio.NewScanner(rsp.Body)
for scanner.Scan() {
buf := scanner.Bytes() // the scanner prunes the ending LF
if len(buf) < 1 {
// truncated record? skip
continue
}
idx := bytes.IndexByte(buf, 0x1E) // find the initial RS
if idx < 0 {
// no RS? skip
continue
}
buf = buf[idx+1:] // drop the initial RS
var log Log
if err := json.Unmarshal(buf, &log); err != nil {
// truncated/corrupted/binary record? skip
continue
}
ch <- log
}
close(ch)
rsp.Body.Close()
}()
return ch, nil
}
// ErrNoNames is returned by Start, Stop, or Restart, when the given
// list of things on which to operate is empty.
var ErrNoNames = errors.New(`"names" must not be empty`)
type appInstruction struct {
Action string `json:"action"`
Names []string `json:"names"`
StartOptions
StopOptions
RestartOptions
}
// StartOptions represent the different options of the Start call.
type StartOptions struct {
// Enable, as well as starting, the listed services. A
// disabled service does not start on boot.
Enable bool `json:"enable,omitempty"`
}
// Start services.
//
// It takes a list of names that can be snaps, of which all their
// services are started, or snap.service which are individual
// services to start; it shouldn't be empty.
func (client *Client) Start(names []string, opts StartOptions) (changeID string, err error) {
if len(names) == 0 {
return "", ErrNoNames
}
buf, err := json.Marshal(appInstruction{
Action: "start",
Names: names,
StartOptions: opts,
})
if err != nil {
return "", err
}
return client.doAsync("POST", "/v2/apps", nil, nil, bytes.NewReader(buf))
}
// StopOptions represent the different options of the Stop call.
type StopOptions struct {
// Disable, as well as stopping, the listed services. A
// service that is not disabled starts on boot.
Disable bool `json:"disable,omitempty"`
}
// Stop services.
//
// It takes a list of names that can be snaps, of which all their
// services are stopped, or snap.service which are individual
// services to stop; it shouldn't be empty.
func (client *Client) Stop(names []string, opts StopOptions) (changeID string, err error) {
if len(names) == 0 {
return "", ErrNoNames
}
buf, err := json.Marshal(appInstruction{
Action: "stop",
Names: names,
StopOptions: opts,
})
if err != nil {
return "", err
}
return client.doAsync("POST", "/v2/apps", nil, nil, bytes.NewReader(buf))
}
// RestartOptions represent the different options of the Restart call.
type RestartOptions struct {
// Reload the services, if possible (i.e. if the App has a
// ReloadCommand, invoque it), instead of restarting.
Reload bool `json:"reload,omitempty"`
}
// Restart services.
//
// It takes a list of names that can be snaps, of which all their
// services are restarted, or snap.service which are individual
// services to restart; it shouldn't be empty. If the service is not
// running, starts it.
func (client *Client) Restart(names []string, opts RestartOptions) (changeID string, err error) {
if len(names) == 0 {
return "", ErrNoNames
}
buf, err := json.Marshal(appInstruction{
Action: "restart",
Names: names,
RestartOptions: opts,
})
if err != nil {
return "", err
}
return client.doAsync("POST", "/v2/apps", nil, nil, bytes.NewReader(buf))
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。