2 * Copyright (C) 2017 "IoT.bzh"
3 * Author Sebastien Douheret <sebastien@iot.bzh>
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
26 "github.com/iotbzh/xds-agent/lib/xaapiv1"
27 "github.com/urfave/cli"
30 func initCmdSdks(cmdDef *[]cli.Command) {
31 *cmdDef = append(*cmdDef, cli.Command{
33 Aliases: []string{"sdk"},
35 Usage: "SDKs commands group",
36 Subcommands: []cli.Command{
39 Usage: "Get a property of a SDK",
51 Aliases: []string{"ls"},
52 Usage: "List installed SDKs",
57 Usage: "display all existing sdks (installed + downloadable)",
61 Usage: "regexp to filter output (filtering done only on ID, Name, Version and Arch fields) ",
65 Usage: "display verbose output",
71 Aliases: []string{"i"},
72 Usage: "Install a SDK",
77 Usage: "sdk id to install",
82 Usage: "use this file to install SDK",
86 Usage: "enable debug mode (useful to investigate install issue)",
90 Usage: "force SDK installation when already installed",
96 Aliases: []string{"rm"},
97 Usage: "UnInstall an existing SDK",
98 Action: sdksUnInstall,
102 Usage: "sdk id to un-install",
103 EnvVar: "XDS_SDK_ID",
109 Aliases: []string{"a"},
110 Usage: "Abort an install action",
115 Usage: "sdk id to which abort action",
116 EnvVar: "XDS_SDK_ID",
124 func sdksList(ctx *cli.Context) error {
126 sdks := []xaapiv1.SDK{}
127 if err := _sdksListGet(&sdks); err != nil {
128 return cli.NewExitError(err.Error(), 1)
131 _displaySdks(sdks, ctx.Bool("verbose"), ctx.Bool("all"), ctx.String("filter"))
135 func sdksGet(ctx *cli.Context) error {
138 return cli.NewExitError("id parameter or option must be set", 1)
140 sdks := xaapiv1.SDK{}
141 url := XdsServerComputeURL("/sdks/" + id)
142 if err := HTTPCli.Get(url, &sdks); err != nil {
143 return cli.NewExitError(err.Error(), 1)
146 _displaySdks([]xaapiv1.SDK{sdks}, true, true, "")
150 func _displaySdks(sdks []xaapiv1.SDK, verbose bool, all bool, filter string) {
153 writer := NewTableWriter()
154 for _, s := range sdks {
155 if s.Status != xaapiv1.SdkStatusInstalled && !all {
159 re := regexp.MustCompile(filter)
160 if !(re.MatchString(s.ID) || re.MatchString(s.Name) ||
161 re.MatchString(s.Profile) || re.MatchString(s.Arch) ||
162 re.MatchString(s.Version)) {
171 fmt.Fprintln(writer, "ID\t"+s.ID)
172 fmt.Fprintln(writer, "Name\t"+s.Name)
173 fmt.Fprintln(writer, "Description\t"+s.Description)
174 fmt.Fprintln(writer, "Profile\t"+s.Profile)
175 fmt.Fprintln(writer, "Arch\t"+s.Arch)
176 fmt.Fprintln(writer, "Version\t"+s.Version)
177 fmt.Fprintln(writer, "Status\t"+s.Status)
178 fmt.Fprintln(writer, "Path\t"+s.Path)
179 fmt.Fprintln(writer, "Url\t"+s.URL)
184 fmt.Fprintf(writer, "List of available SDKs: \n")
186 fmt.Fprintf(writer, "List of installed SDKs: \n")
188 fmt.Fprintf(writer, "ID\t NAME\t STATUS\t VERSION\t ARCH\n")
190 fmt.Fprintf(writer, "%s\t %s\t %s\t %s\t %s\n", s.ID[:8], s.Name, s.Status, s.Version, s.Arch)
197 func _sdksListGet(sdks *[]xaapiv1.SDK) error {
198 url := XdsServerComputeURL("/sdks")
199 if err := HTTPCli.Get(url, &sdks); err != nil {
202 Log.Debugf("Result of %s: %v", url, sdks)
207 func sdksInstall(ctx *cli.Context) error {
209 file := ctx.String("file")
210 force := ctx.Bool("force")
212 if id == "" && file == "" {
213 return cli.NewExitError("id or file parameter or option must be set", 1)
216 // Process Socket IO events
217 type exitResult struct {
221 exitChan := make(chan exitResult, 1)
223 IOsk.On("disconnection", func(err error) {
224 Log.Debugf("WS disconnection event with err: %v\n", err)
229 exitChan <- exitResult{errMsg, 2}
232 IOsk.On(xaapiv1.EVTSDKInstall, func(ev xaapiv1.EventMsg) {
233 sdkEvt, _ := ev.DecodeSDKMsg()
235 if sdkEvt.Stdout != "" {
236 fmt.Printf("%s", sdkEvt.Stdout)
238 if sdkEvt.Stderr != "" {
239 fmt.Fprintf(os.Stderr, "%s", sdkEvt.Stderr)
243 exitChan <- exitResult{sdkEvt.Error, sdkEvt.Code}
247 evReg := xaapiv1.EventRegisterArgs{Name: xaapiv1.EVTSDKInstall}
248 if err := HTTPCli.Post("/events/register", &evReg, nil); err != nil {
249 return cli.NewExitError(err, 1)
252 url := XdsServerComputeURL("/sdks")
253 sdks := xaapiv1.SDKInstallArgs{
259 if ctx.Bool("debug") {
260 sdks.InstallArgs = []string{"--debug"}
263 newSdk := xaapiv1.SDK{}
264 if err := HTTPCli.Post(url, &sdks, &newSdk); err != nil {
265 return cli.NewExitError(err, 1)
267 Log.Debugf("Result of %s: %v", url, newSdk)
268 fmt.Printf("Installation of '%s' SDK successfully started.\n", newSdk.Name)
270 // TODO: trap CTRL+C and print question: "Installation of xxx is in progress, press 'a' to abort, 'b' to continue in background or 'c' to continue installation"
274 case res := <-exitChan:
276 Log.Debugln("Exit successfully")
277 fmt.Println("SDK ID " + newSdk.ID + " successfully installed.")
280 Log.Debugln("Exit with ERROR: ", res.error)
282 return cli.NewExitError(res.error, res.code)
286 func sdksUnInstall(ctx *cli.Context) error {
289 return cli.NewExitError("id parameter or option must be set", 1)
292 delSdk := xaapiv1.SDK{}
293 url := XdsServerComputeURL("/sdks/" + id)
294 if err := HTTPCli.Delete(url, &delSdk); err != nil {
295 return cli.NewExitError(err, 1)
298 Log.Debugf("Result of %s: %v", url, delSdk)
300 fmt.Println("SDK ID " + delSdk.ID + " successfully deleted.")
304 func sdksAbort(ctx *cli.Context) error {
307 return cli.NewExitError("id parameter or option must be set", 1)
310 sdks := xaapiv1.SDKInstallArgs{ID: id}
311 newSdk := xaapiv1.SDK{}
312 url := XdsServerComputeURL("/sdks/abortinstall")
313 if err := HTTPCli.Post(url, &sdks, &newSdk); err != nil {
314 return cli.NewExitError(err, 1)
317 Log.Debugf("Result of %s: %v", url, newSdk)