@@ -18,6 +18,7 @@ package fs
18
18
19
19
import (
20
20
"context"
21
+ "errors"
21
22
"os"
22
23
"path/filepath"
23
24
"strings"
@@ -102,9 +103,6 @@ func Changes(ctx context.Context, a, b string, changeFn ChangeFunc) error {
102
103
if a == "" {
103
104
logrus .Debugf ("Using single walk diff for %s" , b )
104
105
return addDirChanges (ctx , changeFn , b )
105
- } else if diffOptions := detectDirDiff (b , a ); diffOptions != nil {
106
- logrus .Debugf ("Using single walk diff for %s from %s" , diffOptions .diffDir , a )
107
- return diffDirChanges (ctx , changeFn , a , diffOptions )
108
106
}
109
107
110
108
logrus .Debugf ("Using double walk diff for %s from %s" , b , a )
@@ -134,24 +132,53 @@ func addDirChanges(ctx context.Context, changeFn ChangeFunc, root string) error
134
132
})
135
133
}
136
134
135
+ // DiffChangeSource is the source of diff directory.
136
+ type DiffSource int
137
+
138
+ const (
139
+ // DiffSourceOverlayFS indicates that a diff directory is from
140
+ // OverlayFS.
141
+ DiffSourceOverlayFS DiffSource = iota
142
+ )
143
+
137
144
// diffDirOptions is used when the diff can be directly calculated from
138
145
// a diff directory to its base, without walking both trees.
139
146
type diffDirOptions struct {
140
- diffDir string
141
- skipChange func (string ) (bool , error )
142
- deleteChange func (string , string , os.FileInfo ) (string , error )
147
+ skipChange func (string , os.FileInfo ) (bool , error )
148
+ deleteChange func (string , string , os.FileInfo , ChangeFunc ) (bool , error )
143
149
}
144
150
145
- // diffDirChanges walks the diff directory and compares changes against the base.
146
- func diffDirChanges (ctx context.Context , changeFn ChangeFunc , base string , o * diffDirOptions ) error {
151
+ // DiffDirChanges walks the diff directory and compares changes against the base.
152
+ //
153
+ // NOTE: If all the children of a dir are removed, or that dir are recreated
154
+ // after remove, we will mark non-existing `.wh..opq` file as deleted. It's
155
+ // unlikely to create explicit whiteout files for all the children and all
156
+ // descendants. And based on OCI spec, it's not possible to create a file or
157
+ // dir with a name beginning with `.wh.`. So, after `.wh..opq` file has been
158
+ // deleted, the ChangeFunc, the receiver will add whiteout prefix to create a
159
+ // opaque whiteout `.wh..wh..opq`.
160
+ //
161
+ // REF: https://github.com/opencontainers/image-spec/blob/v1.0/layer.md#whiteouts
162
+ func DiffDirChanges (ctx context.Context , baseDir , diffDir string , source DiffSource , changeFn ChangeFunc ) error {
163
+ var o * diffDirOptions
164
+
165
+ switch source {
166
+ case DiffSourceOverlayFS :
167
+ o = & diffDirOptions {
168
+ deleteChange : overlayFSWhiteoutConvert ,
169
+ }
170
+ default :
171
+ return errors .New ("unknown diff change source" )
172
+ }
173
+
147
174
changedDirs := make (map [string ]struct {})
148
- return filepath .Walk (o . diffDir , func (path string , f os.FileInfo , err error ) error {
175
+ return filepath .Walk (diffDir , func (path string , f os.FileInfo , err error ) error {
149
176
if err != nil {
150
177
return err
151
178
}
152
179
153
180
// Rebase path
154
- path , err = filepath .Rel (o . diffDir , path )
181
+ path , err = filepath .Rel (diffDir , path )
155
182
if err != nil {
156
183
return err
157
184
}
@@ -163,38 +190,45 @@ func diffDirChanges(ctx context.Context, changeFn ChangeFunc, base string, o *di
163
190
return nil
164
191
}
165
192
166
- // TODO: handle opaqueness, start new double walker at this
167
- // location to get deletes, and skip tree in single walker
168
-
169
193
if o .skipChange != nil {
170
- if skip , err := o .skipChange (path ); skip {
194
+ if skip , err := o .skipChange (path , f ); skip {
171
195
return err
172
196
}
173
197
}
174
198
175
199
var kind ChangeKind
176
200
177
- deletedFile , err := o .deleteChange (o .diffDir , path , f )
178
- if err != nil {
179
- return err
201
+ deletedFile := false
202
+
203
+ if o .deleteChange != nil {
204
+ deletedFile , err = o .deleteChange (diffDir , path , f , changeFn )
205
+ if err != nil {
206
+ return err
207
+ }
208
+
209
+ _ , err = os .Stat (filepath .Join (baseDir , path ))
210
+ if err != nil {
211
+ if ! os .IsNotExist (err ) {
212
+ return err
213
+ }
214
+ deletedFile = false
215
+ }
180
216
}
181
217
182
218
// Find out what kind of modification happened
183
- if deletedFile != "" {
184
- path = deletedFile
219
+ if deletedFile {
185
220
kind = ChangeKindDelete
186
- f = nil
187
221
} else {
188
222
// Otherwise, the file was added
189
223
kind = ChangeKindAdd
190
224
191
- // ...Unless it already existed in a base , in which case, it's a modification
192
- stat , err := os .Stat (filepath .Join (base , path ))
225
+ // ...Unless it already existed in a baseDir , in which case, it's a modification
226
+ stat , err := os .Stat (filepath .Join (baseDir , path ))
193
227
if err != nil && ! os .IsNotExist (err ) {
194
228
return err
195
229
}
196
230
if err == nil {
197
- // The file existed in the base , so that's a modification
231
+ // The file existed in the baseDir , so that's a modification
198
232
199
233
// However, if it's a directory, maybe it wasn't actually modified.
200
234
// If you modify /foo/bar/baz, then /foo will be part of the changed files only because it's the parent of bar
@@ -215,17 +249,22 @@ func diffDirChanges(ctx context.Context, changeFn ChangeFunc, base string, o *di
215
249
if f .IsDir () {
216
250
changedDirs [path ] = struct {}{}
217
251
}
252
+
218
253
if kind == ChangeKindAdd || kind == ChangeKindDelete {
219
254
parent := filepath .Dir (path )
255
+
220
256
if _ , ok := changedDirs [parent ]; ! ok && parent != "/" {
221
- pi , err := os .Stat (filepath .Join (o . diffDir , parent ))
257
+ pi , err := os .Stat (filepath .Join (diffDir , parent ))
222
258
if err := changeFn (ChangeKindModify , parent , pi , err ); err != nil {
223
259
return err
224
260
}
225
261
changedDirs [parent ] = struct {}{}
226
262
}
227
263
}
228
264
265
+ if kind == ChangeKindDelete {
266
+ f = nil
267
+ }
229
268
return changeFn (kind , path , f , nil )
230
269
})
231
270
}
0 commit comments