@@ -41,9 +41,10 @@ export async function getConfig(s: string, file: boolean): Promise<string> {
41
41
return configFile ;
42
42
}
43
43
44
- export async function isAvailable ( ) : Promise < boolean > {
44
+ export async function isAvailable ( standalone ?: boolean ) : Promise < boolean > {
45
+ const cmd = getCommand ( [ ] , standalone ) ;
45
46
return await exec
46
- . getExecOutput ( 'docker' , [ 'buildx' ] , {
47
+ . getExecOutput ( cmd . commandLine , cmd . args , {
47
48
ignoreReturnCode : true ,
48
49
silent : true
49
50
} )
@@ -52,12 +53,17 @@ export async function isAvailable(): Promise<boolean> {
52
53
return false ;
53
54
}
54
55
return res . exitCode == 0 ;
56
+ } )
57
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
58
+ . catch ( error => {
59
+ return false ;
55
60
} ) ;
56
61
}
57
62
58
- export async function getVersion ( ) : Promise < string > {
63
+ export async function getVersion ( standalone ?: boolean ) : Promise < string > {
64
+ const cmd = getCommand ( [ 'version' ] , standalone ) ;
59
65
return await exec
60
- . getExecOutput ( 'docker' , [ 'buildx' , 'version' ] , {
66
+ . getExecOutput ( cmd . commandLine , cmd . args , {
61
67
ignoreReturnCode : true ,
62
68
silent : true
63
69
} )
@@ -81,9 +87,10 @@ export function satisfies(version: string, range: string): boolean {
81
87
return semver . satisfies ( version , range ) || / ^ [ 0 - 9 a - f ] { 7 } $ / . exec ( version ) !== null ;
82
88
}
83
89
84
- export async function inspect ( name : string ) : Promise < Builder > {
90
+ export async function inspect ( name : string , standalone ?: boolean ) : Promise < Builder > {
91
+ const cmd = getCommand ( [ 'inspect' , name ] , standalone ) ;
85
92
return await exec
86
- . getExecOutput ( `docker` , [ 'buildx' , 'inspect' , name ] , {
93
+ . getExecOutput ( cmd . commandLine , cmd . args , {
87
94
ignoreReturnCode : true ,
88
95
silent : true
89
96
} )
@@ -133,7 +140,7 @@ export async function inspect(name: string): Promise<Builder> {
133
140
} ) ;
134
141
}
135
142
136
- export async function build ( inputBuildRef : string , dockerConfigHome : string ) : Promise < string > {
143
+ export async function build ( inputBuildRef : string , dest : string , standalone : boolean ) : Promise < string > {
137
144
// eslint-disable-next-line prefer-const
138
145
let [ repo , ref ] = inputBuildRef . split ( '#' ) ;
139
146
if ( ref . length == 0 ) {
@@ -152,8 +159,27 @@ export async function build(inputBuildRef: string, dockerConfigHome: string): Pr
152
159
toolPath = tc . find ( 'buildx' , vspec ) ;
153
160
if ( ! toolPath ) {
154
161
const outFolder = path . join ( context . tmpDir ( ) , 'out' ) . split ( path . sep ) . join ( path . posix . sep ) ;
162
+ let buildWithStandalone = false ;
163
+ const standaloneFound = await isAvailable ( true ) ;
164
+ const pluginFound = await isAvailable ( false ) ;
165
+ if ( standalone && standaloneFound ) {
166
+ core . debug ( `Buildx standalone found, build with it` ) ;
167
+ buildWithStandalone = true ;
168
+ } else if ( ! standalone && pluginFound ) {
169
+ core . debug ( `Buildx plugin found, build with it` ) ;
170
+ buildWithStandalone = false ;
171
+ } else if ( standaloneFound ) {
172
+ core . debug ( `Buildx plugin not found, but standalone found so trying to build with it` ) ;
173
+ buildWithStandalone = true ;
174
+ } else if ( pluginFound ) {
175
+ core . debug ( `Buildx standalone not found, but plugin found so trying to build with it` ) ;
176
+ buildWithStandalone = false ;
177
+ } else {
178
+ throw new Error ( `Neither buildx standalone or plugin have been found to build from ref` ) ;
179
+ }
180
+ const buildCmd = getCommand ( [ 'build' , '--target' , 'binaries' , '--build-arg' , 'BUILDKIT_CONTEXT_KEEP_GIT_DIR=1' , '--output' , `type=local,dest=${ outFolder } ` , inputBuildRef ] , buildWithStandalone ) ;
155
181
toolPath = await exec
156
- . getExecOutput ( 'docker' , [ 'buildx' , 'build' , '--target' , 'binaries' , '--build-arg' , 'BUILDKIT_CONTEXT_KEEP_GIT_DIR=1' , '--output' , `type=local,dest= ${ outFolder } ` , inputBuildRef ] , {
182
+ . getExecOutput ( buildCmd . commandLine , buildCmd . args , {
157
183
ignoreReturnCode : true
158
184
} )
159
185
. then ( res => {
@@ -164,10 +190,13 @@ export async function build(inputBuildRef: string, dockerConfigHome: string): Pr
164
190
} ) ;
165
191
}
166
192
167
- return setPlugin ( toolPath , dockerConfigHome ) ;
193
+ if ( standalone ) {
194
+ return setStandalone ( toolPath , dest ) ;
195
+ }
196
+ return setPlugin ( toolPath , dest ) ;
168
197
}
169
198
170
- export async function install ( inputVersion : string , dockerConfigHome : string ) : Promise < string > {
199
+ export async function install ( inputVersion : string , dest : string , standalone : boolean ) : Promise < string > {
171
200
const release : github . GitHubRelease | null = await github . getRelease ( inputVersion ) ;
172
201
if ( ! release ) {
173
202
throw new Error ( `Cannot find buildx ${ inputVersion } release` ) ;
@@ -185,10 +214,40 @@ export async function install(inputVersion: string, dockerConfigHome: string): P
185
214
toolPath = await download ( version ) ;
186
215
}
187
216
188
- return setPlugin ( toolPath , dockerConfigHome ) ;
217
+ if ( standalone ) {
218
+ return setStandalone ( toolPath , dest ) ;
219
+ }
220
+ return setPlugin ( toolPath , dest ) ;
221
+ }
222
+
223
+ async function setStandalone ( toolPath : string , dest : string ) : Promise < string > {
224
+ core . info ( 'Standalone mode' ) ;
225
+ const toolBinPath = path . join ( toolPath , context . osPlat == 'win32' ? 'docker-buildx.exe' : 'docker-buildx' ) ;
226
+
227
+ const binDir = path . join ( dest , 'bin' ) ;
228
+ core . debug ( `Bin dir is ${ binDir } ` ) ;
229
+ if ( ! fs . existsSync ( binDir ) ) {
230
+ fs . mkdirSync ( binDir , { recursive : true } ) ;
231
+ }
232
+
233
+ const filename : string = context . osPlat == 'win32' ? 'buildx.exe' : 'buildx' ;
234
+ const buildxPath : string = path . join ( binDir , filename ) ;
235
+ core . debug ( `Bin path is ${ buildxPath } ` ) ;
236
+ fs . copyFileSync ( toolBinPath , buildxPath ) ;
237
+
238
+ core . info ( 'Fixing perms' ) ;
239
+ fs . chmodSync ( buildxPath , '0755' ) ;
240
+
241
+ core . addPath ( binDir ) ;
242
+ core . info ( 'Added buildx to the path' ) ;
243
+
244
+ return buildxPath ;
189
245
}
190
246
191
247
async function setPlugin ( toolPath : string , dockerConfigHome : string ) : Promise < string > {
248
+ core . info ( 'Docker plugin mode' ) ;
249
+ const toolBinPath = path . join ( toolPath , context . osPlat == 'win32' ? 'docker-buildx.exe' : 'docker-buildx' ) ;
250
+
192
251
const pluginsDir : string = path . join ( dockerConfigHome , 'cli-plugins' ) ;
193
252
core . debug ( `Plugins dir is ${ pluginsDir } ` ) ;
194
253
if ( ! fs . existsSync ( pluginsDir ) ) {
@@ -198,7 +257,7 @@ async function setPlugin(toolPath: string, dockerConfigHome: string): Promise<st
198
257
const filename : string = context . osPlat == 'win32' ? 'docker-buildx.exe' : 'docker-buildx' ;
199
258
const pluginPath : string = path . join ( pluginsDir , filename ) ;
200
259
core . debug ( `Plugin path is ${ pluginPath } ` ) ;
201
- fs . copyFileSync ( path . join ( toolPath , filename ) , pluginPath ) ;
260
+ fs . copyFileSync ( toolBinPath , pluginPath ) ;
202
261
203
262
core . info ( 'Fixing perms' ) ;
204
263
fs . chmodSync ( pluginPath , '0755' ) ;
@@ -269,3 +328,10 @@ export async function getBuildKitVersion(containerID: string): Promise<string> {
269
328
return bkitimage . stdout . trim ( ) ;
270
329
} ) ;
271
330
}
331
+
332
+ export function getCommand ( args : Array < string > , standalone ?: boolean ) {
333
+ return {
334
+ commandLine : standalone ? 'buildx' : 'docker' ,
335
+ args : standalone ? args : [ 'buildx' , ...args ]
336
+ } ;
337
+ }
0 commit comments