2 * Copyright (C) 2017-2018 "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.
27 "gerrit.automotivelinux.org/gerrit/src/xds/xds-agent.git/lib/xaapiv1"
28 "github.com/urfave/cli"
31 func initCmdSdks(cmdDef *[]cli.Command) {
32 *cmdDef = append(*cmdDef, cli.Command{
34 Aliases: []string{"sdk"},
36 Usage: "SDKs commands group",
37 Subcommands: []cli.Command{
40 Usage: "Get a property of a SDK",
52 Aliases: []string{"ls"},
53 Usage: "List installed SDKs",
58 Usage: "display all existing sdks (installed + downloadable)",
62 Usage: "regexp to filter output (filtering done only on ID, Name, Version and Arch fields) ",
66 Usage: "display verbose output",
72 Aliases: []string{"i"},
73 Usage: "Install a SDK",
78 Usage: "sdk id to install",
83 Usage: "use this file to install SDK",
87 Usage: "enable debug mode (useful to investigate install issue)",
91 Usage: "force SDK installation when already installed",
95 Usage: "short output, only print create sdk id (useful from scripting)",
101 Aliases: []string{"rm"},
102 Usage: "UnInstall an existing SDK",
103 Action: sdksUnInstall,
107 Usage: "sdk id to un-install",
108 EnvVar: "XDS_SDK_ID",
112 Usage: "remove confirmation prompt before removal",
118 Aliases: []string{"a"},
119 Usage: "Abort an install action",
124 Usage: "sdk id to which abort action",
125 EnvVar: "XDS_SDK_ID",
133 func sdksList(ctx *cli.Context) error {
135 sdks := []xaapiv1.SDK{}
136 if err := _sdksListGet(&sdks); err != nil {
137 return cli.NewExitError(err.Error(), 1)
140 _displaySdks(sdks, ctx.Bool("verbose"), ctx.Bool("all"), ctx.String("filter"))
144 func sdksGet(ctx *cli.Context) error {
145 id := GetID(ctx, "XDS_SDK_ID")
147 return cli.NewExitError("id parameter or option must be set", 1)
149 sdks := xaapiv1.SDK{}
150 url := XdsServerComputeURL("/sdks/" + id)
151 if err := HTTPCli.Get(url, &sdks); err != nil {
152 return cli.NewExitError(err.Error(), 1)
155 _displaySdks([]xaapiv1.SDK{sdks}, true, true, "")
159 func _displaySdks(sdks []xaapiv1.SDK, verbose bool, all bool, filter string) {
162 writer := NewTableWriter()
163 for _, s := range sdks {
164 if s.Status != xaapiv1.SdkStatusInstalled &&
165 s.Status != xaapiv1.SdkStatusInstalling &&
170 re := regexp.MustCompile(filter)
171 if !(re.MatchString(s.ID) || re.MatchString(s.Name) ||
172 re.MatchString(s.Profile) || re.MatchString(s.Arch) ||
173 re.MatchString(s.Version)) {
182 fmt.Fprintln(writer, "ID\t"+s.ID)
183 fmt.Fprintln(writer, "Name\t"+s.Name)
184 fmt.Fprintln(writer, "Description\t"+s.Description)
185 fmt.Fprintln(writer, "Profile\t"+s.Profile)
186 fmt.Fprintln(writer, "Arch\t"+s.Arch)
187 fmt.Fprintln(writer, "Version\t"+s.Version)
188 fmt.Fprintln(writer, "Status\t"+s.Status)
189 fmt.Fprintln(writer, "Path\t"+s.Path)
190 fmt.Fprintln(writer, "Url\t"+formatURL(s.URL))
191 fmt.Fprintln(writer, "Image Url\t"+formatURL(s.ImageURL))
196 fmt.Fprintf(writer, "List of available SDKs: \n")
198 fmt.Fprintf(writer, "List of installed SDKs: \n")
200 fmt.Fprintf(writer, "ID\t NAME\t STATUS\t VERSION\t ARCH\n")
202 fmt.Fprintf(writer, "%s\t %s\t %s\t %s\t %s\n", s.ID[:8], s.Name, s.Status, s.Version, s.Arch)
210 type _SdkByID []xaapiv1.SDK
212 func (s _SdkByID) Len() int { return len(s) }
213 func (s _SdkByID) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
214 func (s _SdkByID) Less(i, j int) bool { return s[i].ID < s[j].ID }
216 // _sdksListGet Get the list of existing sdks
217 func _sdksListGet(sdks *[]xaapiv1.SDK) error {
218 url := XdsServerComputeURL("/sdks")
219 if err := HTTPCli.Get(url, &sdks); err != nil {
222 Log.Debugf("Result of %s: %v", url, sdks)
224 sort.Sort(_SdkByID(*sdks))
229 func sdksInstall(ctx *cli.Context) error {
230 id := GetID(ctx, "XDS_SDK_ID")
231 file := ctx.String("file")
232 force := ctx.Bool("force")
233 shortOut := ctx.Bool("short")
235 if id == "" && file == "" {
236 return cli.NewExitError("id or file parameter or option must be set", 1)
239 // Process Socket IO events
240 type exitResult struct {
244 exitChan := make(chan exitResult, 1)
246 IOSkClient.On("disconnection", func(err error) {
247 Log.Debugf("WS disconnection event with err: %v\n", err)
252 exitChan <- exitResult{errMsg, 2}
255 IOSkClient.On(xaapiv1.EVTSDKManagement, func(ev xaapiv1.EventMsg) {
256 sdkEvt, _ := ev.DecodeSDKMgtMsg()
258 if sdkEvt.Action != xaapiv1.SdkMgtActionInstall {
259 Log.Debugf("EVTSDKManagement (action %s) IGNORED", sdkEvt.Action)
263 if !shortOut && len(sdkEvt.Stdout) > 0 {
264 os.Stdout.Write([]byte(sdkEvt.Stdout))
266 if !shortOut && len(sdkEvt.Stderr) > 0 {
267 os.Stderr.Write([]byte(sdkEvt.Stderr))
271 exitChan <- exitResult{sdkEvt.Error, sdkEvt.Code}
275 IOSkClient.On(xaapiv1.EVTSDKStateChange, func(ev xaapiv1.EventMsg) {
276 sdk, _ := ev.DecodeSDKEvent()
277 Log.Debugf("EVTSDKStateChange: %v", sdk)
280 evReg := xaapiv1.EventRegisterArgs{Name: xaapiv1.EVTAll}
281 if err := HTTPCli.Post("/events/register", &evReg, nil); err != nil {
282 return cli.NewExitError(err, 1)
285 url := XdsServerComputeURL("/sdks")
286 sdks := xaapiv1.SDKInstallArgs{
292 if ctx.Bool("debug") {
293 sdks.InstallArgs = []string{"--debug"}
296 newSdk := xaapiv1.SDK{}
297 if err := HTTPCli.Post(url, &sdks, &newSdk); err != nil {
298 return cli.NewExitError(err, 1)
300 Log.Debugf("Result of %s: %v", url, newSdk)
302 fmt.Printf("Installation of '%s' SDK successfully started.\n", newSdk.Name)
305 // 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"
309 case res := <-IOSkClient.ServerDiscoChan:
310 Log.Debugf("XDS Server disconnected %v", res)
311 return cli.NewExitError(res.error, res.code)
313 case res := <-exitChan:
315 Log.Debugln("Exit successfully")
317 fmt.Println(newSdk.ID)
319 fmt.Println("SDK ID " + newSdk.ID + " successfully installed.")
323 Log.Debugln("Exit with ERROR: ", res.error)
325 return cli.NewExitError(res.error, res.code)
329 func sdksUnInstall(ctx *cli.Context) error {
330 id := GetID(ctx, "XDS_SDK_ID")
332 return cli.NewExitError("id parameter or option must be set", 1)
335 if !ctx.Bool("force") {
336 if !Confirm("Do you permanently remove SDK id '" + id + "' [yes/No] ? ") {
341 delSdk := xaapiv1.SDK{}
342 url := XdsServerComputeURL("/sdks/" + id)
343 if err := HTTPCli.Delete(url, &delSdk); err != nil {
344 return cli.NewExitError(err, 1)
347 Log.Debugf("Result of %s: %v", url, delSdk)
349 fmt.Println("SDK ID " + delSdk.ID + " successfully deleted.")
353 func sdksAbort(ctx *cli.Context) error {
354 id := GetID(ctx, "XDS_SDK_ID")
356 return cli.NewExitError("id parameter or option must be set", 1)
359 sdks := xaapiv1.SDKInstallArgs{ID: id}
360 newSdk := xaapiv1.SDK{}
361 url := XdsServerComputeURL("/sdks/abortinstall")
362 if err := HTTPCli.Post(url, &sdks, &newSdk); err != nil {
363 return cli.NewExitError(err, 1)
366 Log.Debugf("Result of %s: %v", url, newSdk)
370 func formatURL(u string) string {