2014-11-16 21:13:20 +01:00
// Copyright (C) 2014 The Syncthing Authors.
2014-10-04 15:48:33 +01:00
//
2015-03-07 21:36:35 +01:00
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
2017-02-09 07:52:18 +01:00
// You can obtain one at https://mozilla.org/MPL/2.0/.
2014-10-04 15:48:33 +01:00
package model
import (
2017-04-26 00:15:23 +00:00
"context"
2016-12-14 23:30:29 +00:00
"crypto/rand"
"io"
2014-10-11 00:27:17 +01:00
"os"
"path/filepath"
2014-10-04 15:48:33 +01:00
"testing"
2014-12-09 23:58:58 +00:00
"time"
2014-10-04 15:48:33 +01:00
2015-10-31 12:31:25 +01:00
"github.com/syncthing/syncthing/lib/db"
2016-12-14 23:30:29 +00:00
"github.com/syncthing/syncthing/lib/fs"
2017-01-17 07:33:48 +00:00
"github.com/syncthing/syncthing/lib/ignore"
2015-09-22 19:38:46 +02:00
"github.com/syncthing/syncthing/lib/protocol"
2015-08-06 11:29:25 +02:00
"github.com/syncthing/syncthing/lib/scanner"
"github.com/syncthing/syncthing/lib/sync"
2014-10-04 15:48:33 +01:00
)
2017-01-17 07:33:48 +00:00
func TestMain ( m * testing . M ) {
2016-11-30 21:23:24 +00:00
// We do this to make sure that the temp file required for the tests
// does not get removed during the tests. Also set the prefix so it's
// found correctly regardless of platform.
2017-01-17 07:33:48 +00:00
if ignore . TempPrefix != ignore . WindowsTempPrefix {
originalPrefix := ignore . TempPrefix
ignore . TempPrefix = ignore . WindowsTempPrefix
defer func ( ) {
ignore . TempPrefix = originalPrefix
} ( )
}
2014-12-09 23:58:58 +00:00
future := time . Now ( ) . Add ( time . Hour )
2017-01-17 07:33:48 +00:00
err := os . Chtimes ( filepath . Join ( "testdata" , ignore . TempName ( "file" ) ) , future , future )
2014-12-09 23:58:58 +00:00
if err != nil {
panic ( err )
}
2017-01-17 07:33:48 +00:00
os . Exit ( m . Run ( ) )
2014-12-09 23:58:58 +00:00
}
2014-10-04 15:48:33 +01:00
var blocks = [ ] protocol . BlockInfo {
{ Hash : [ ] uint8 { 0xfa , 0x43 , 0x23 , 0x9b , 0xce , 0xe7 , 0xb9 , 0x7c , 0xa6 , 0x2f , 0x0 , 0x7c , 0xc6 , 0x84 , 0x87 , 0x56 , 0xa , 0x39 , 0xe1 , 0x9f , 0x74 , 0xf3 , 0xdd , 0xe7 , 0x48 , 0x6d , 0xb3 , 0xf9 , 0x8d , 0xf8 , 0xe4 , 0x71 } } , // Zero'ed out block
{ Offset : 0 , Size : 0x20000 , Hash : [ ] uint8 { 0x7e , 0xad , 0xbc , 0x36 , 0xae , 0xbb , 0xcf , 0x74 , 0x43 , 0xe2 , 0x7a , 0x5a , 0x4b , 0xb8 , 0x5b , 0xce , 0xe6 , 0x9e , 0x1e , 0x10 , 0xf9 , 0x8a , 0xbc , 0x77 , 0x95 , 0x2 , 0x29 , 0x60 , 0x9e , 0x96 , 0xae , 0x6c } } ,
{ Offset : 131072 , Size : 0x20000 , Hash : [ ] uint8 { 0x3c , 0xc4 , 0x20 , 0xf4 , 0xb , 0x2e , 0xcb , 0xb9 , 0x5d , 0xce , 0x34 , 0xa8 , 0xc3 , 0x92 , 0xea , 0xf3 , 0xda , 0x88 , 0x33 , 0xee , 0x7a , 0xb6 , 0xe , 0xf1 , 0x82 , 0x5e , 0xb0 , 0xa9 , 0x26 , 0xa9 , 0xc0 , 0xef } } ,
{ Offset : 262144 , Size : 0x20000 , Hash : [ ] uint8 { 0x76 , 0xa8 , 0xc , 0x69 , 0xd7 , 0x5c , 0x52 , 0xfd , 0xdf , 0x55 , 0xef , 0x44 , 0xc1 , 0xd6 , 0x25 , 0x48 , 0x4d , 0x98 , 0x48 , 0x4d , 0xaa , 0x50 , 0xf6 , 0x6b , 0x32 , 0x47 , 0x55 , 0x81 , 0x6b , 0xed , 0xee , 0xfb } } ,
{ Offset : 393216 , Size : 0x20000 , Hash : [ ] uint8 { 0x44 , 0x1e , 0xa4 , 0xf2 , 0x8d , 0x1f , 0xc3 , 0x1b , 0x9d , 0xa5 , 0x18 , 0x5e , 0x59 , 0x1b , 0xd8 , 0x5c , 0xba , 0x7d , 0xb9 , 0x8d , 0x70 , 0x11 , 0x5c , 0xea , 0xa1 , 0x57 , 0x4d , 0xcb , 0x3c , 0x5b , 0xf8 , 0x6c } } ,
{ Offset : 524288 , Size : 0x20000 , Hash : [ ] uint8 { 0x8 , 0x40 , 0xd0 , 0x5e , 0x80 , 0x0 , 0x0 , 0x7c , 0x8b , 0xb3 , 0x8b , 0xf7 , 0x7b , 0x23 , 0x26 , 0x28 , 0xab , 0xda , 0xcf , 0x86 , 0x8f , 0xc2 , 0x8a , 0x39 , 0xc6 , 0xe6 , 0x69 , 0x59 , 0x97 , 0xb6 , 0x1a , 0x43 } } ,
{ Offset : 655360 , Size : 0x20000 , Hash : [ ] uint8 { 0x38 , 0x8e , 0x44 , 0xcb , 0x30 , 0xd8 , 0x90 , 0xf , 0xce , 0x7 , 0x4b , 0x58 , 0x86 , 0xde , 0xce , 0x59 , 0xa2 , 0x46 , 0xd2 , 0xf9 , 0xba , 0xaf , 0x35 , 0x87 , 0x38 , 0xdf , 0xd2 , 0xd , 0xf9 , 0x45 , 0xed , 0x91 } } ,
{ Offset : 786432 , Size : 0x20000 , Hash : [ ] uint8 { 0x32 , 0x28 , 0xcd , 0xf , 0x37 , 0x21 , 0xe5 , 0xd4 , 0x1e , 0x58 , 0x87 , 0x73 , 0x8e , 0x36 , 0xdf , 0xb2 , 0x70 , 0x78 , 0x56 , 0xc3 , 0x42 , 0xff , 0xf7 , 0x8f , 0x37 , 0x95 , 0x0 , 0x26 , 0xa , 0xac , 0x54 , 0x72 } } ,
{ Offset : 917504 , Size : 0x20000 , Hash : [ ] uint8 { 0x96 , 0x6b , 0x15 , 0x6b , 0xc4 , 0xf , 0x19 , 0x18 , 0xca , 0xbb , 0x5f , 0xd6 , 0xbb , 0xa2 , 0xc6 , 0x2a , 0xac , 0xbb , 0x8a , 0xb9 , 0xce , 0xec , 0x4c , 0xdb , 0x78 , 0xec , 0x57 , 0x5d , 0x33 , 0xf9 , 0x8e , 0xaf } } ,
}
2015-09-04 12:01:00 +02:00
var folders = [ ] string { "default" }
2015-12-21 16:47:26 -05:00
func setUpFile ( filename string , blockNumbers [ ] int ) protocol . FileInfo {
2014-10-11 00:27:17 +01:00
// Create existing file
2015-12-21 16:47:26 -05:00
existingBlocks := make ( [ ] protocol . BlockInfo , len ( blockNumbers ) )
for i := range blockNumbers {
existingBlocks [ i ] = blocks [ blockNumbers [ i ] ]
}
return protocol . FileInfo {
2016-07-04 10:40:29 +00:00
Name : filename ,
Blocks : existingBlocks ,
2014-10-04 15:48:33 +01:00
}
2015-12-21 16:47:26 -05:00
}
2014-10-04 15:48:33 +01:00
2015-12-21 16:47:26 -05:00
func setUpModel ( file protocol . FileInfo ) * Model {
2015-10-31 12:31:25 +01:00
db := db . OpenMemory ( )
2017-05-22 19:58:33 +00:00
model := NewModel ( defaultConfig , protocol . LocalDeviceID , "syncthing" , "dev" , db , nil )
2015-12-21 16:47:26 -05:00
model . AddFolder ( defaultFolderConfig )
2014-10-11 00:27:17 +01:00
// Update index
2016-05-19 00:19:26 +00:00
model . updateLocalsFromScanning ( "default" , [ ] protocol . FileInfo { file } )
2015-12-21 16:47:26 -05:00
return model
}
2014-10-04 15:48:33 +01:00
2017-04-01 09:58:06 +00:00
func setUpSendReceiveFolder ( model * Model ) * sendReceiveFolder {
f := & sendReceiveFolder {
2016-04-26 20:19:30 +00:00
folder : folder {
2017-04-20 00:20:34 +00:00
stateTracker : newStateTracker ( "default" ) ,
model : model ,
initialScanFinished : make ( chan struct { } ) ,
2017-04-26 00:15:23 +00:00
ctx : context . TODO ( ) ,
2016-04-26 20:19:30 +00:00
} ,
2016-12-14 23:30:29 +00:00
2017-04-01 09:04:11 +00:00
mtimeFS : fs . NewMtimeFS ( fs . DefaultFilesystem , db . NewNamespacedKV ( model . db , "mtime" ) ) ,
2015-06-26 13:31:30 +02:00
dir : "testdata" ,
2016-04-26 20:19:30 +00:00
queue : newJobQueue ( ) ,
2015-06-26 13:31:30 +02:00
errors : make ( map [ string ] string ) ,
errorsMut : sync . NewMutex ( ) ,
2014-10-04 15:48:33 +01:00
}
2017-04-01 09:58:06 +00:00
// Folders are never actually started, so no initial scan will be done
2017-04-20 00:20:34 +00:00
close ( f . initialScanFinished )
2017-04-01 09:58:06 +00:00
return f
2015-12-21 16:47:26 -05:00
}
// Layout of the files: (indexes from the above array)
// 12345678 - Required file
// 02005008 - Existing file (currently in the index)
// 02340070 - Temp file on the disk
func TestHandleFile ( t * testing . T ) {
// After the diff between required and existing we should:
// Copy: 2, 5, 8
// Pull: 1, 3, 4, 6, 7
2014-10-04 15:48:33 +01:00
2015-12-21 16:47:26 -05:00
existingBlocks := [ ] int { 0 , 2 , 0 , 0 , 5 , 0 , 0 , 8 }
existingFile := setUpFile ( "filex" , existingBlocks )
requiredFile := existingFile
requiredFile . Blocks = blocks [ 1 : ]
m := setUpModel ( existingFile )
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2014-10-08 23:41:23 +01:00
copyChan := make ( chan copyBlocksState , 1 )
2014-10-04 15:48:33 +01:00
2016-04-26 20:19:30 +00:00
f . handleFile ( requiredFile , copyChan , nil )
2014-10-04 15:48:33 +01:00
// Receive the results
toCopy := <- copyChan
2014-10-08 23:41:23 +01:00
if len ( toCopy . blocks ) != 8 {
t . Errorf ( "Unexpected count of copy blocks: %d != 8" , len ( toCopy . blocks ) )
2014-10-04 15:48:33 +01:00
}
2016-04-15 10:59:41 +00:00
for _ , block := range blocks [ 1 : ] {
found := false
for _ , toCopyBlock := range toCopy . blocks {
if string ( toCopyBlock . Hash ) == string ( block . Hash ) {
found = true
break
}
}
if ! found {
t . Errorf ( "Did not find block %s" , block . String ( ) )
2014-10-04 15:48:33 +01:00
}
}
}
func TestHandleFileWithTemp ( t * testing . T ) {
// After diff between required and existing we should:
// Copy: 2, 5, 8
// Pull: 1, 3, 4, 6, 7
// After dropping out blocks already on the temp file we should:
// Copy: 5, 8
// Pull: 1, 6
2015-12-21 16:47:26 -05:00
existingBlocks := [ ] int { 0 , 2 , 0 , 0 , 5 , 0 , 0 , 8 }
existingFile := setUpFile ( "file" , existingBlocks )
2014-10-04 15:48:33 +01:00
requiredFile := existingFile
requiredFile . Blocks = blocks [ 1 : ]
2015-12-21 16:47:26 -05:00
m := setUpModel ( existingFile )
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2014-10-08 23:41:23 +01:00
copyChan := make ( chan copyBlocksState , 1 )
2014-10-04 15:48:33 +01:00
2016-04-26 20:19:30 +00:00
f . handleFile ( requiredFile , copyChan , nil )
2014-10-04 15:48:33 +01:00
// Receive the results
toCopy := <- copyChan
2014-10-08 23:41:23 +01:00
if len ( toCopy . blocks ) != 4 {
t . Errorf ( "Unexpected count of copy blocks: %d != 4" , len ( toCopy . blocks ) )
2014-10-04 15:48:33 +01:00
}
2016-04-15 10:59:41 +00:00
for _ , idx := range [ ] int { 1 , 5 , 6 , 8 } {
found := false
block := blocks [ idx ]
for _ , toCopyBlock := range toCopy . blocks {
if string ( toCopyBlock . Hash ) == string ( block . Hash ) {
found = true
break
}
}
if ! found {
t . Errorf ( "Did not find block %s" , block . String ( ) )
2014-10-04 15:48:33 +01:00
}
}
}
2014-10-11 00:27:17 +01:00
func TestCopierFinder ( t * testing . T ) {
// After diff between required and existing we should:
// Copy: 1, 2, 3, 4, 6, 7, 8
// Since there is no existing file, nor a temp file
// After dropping out blocks found locally:
// Pull: 1, 5, 6, 8
2017-01-17 07:33:48 +00:00
tempFile := filepath . Join ( "testdata" , ignore . TempName ( "file2" ) )
2014-10-11 00:27:17 +01:00
err := os . Remove ( tempFile )
if err != nil && ! os . IsNotExist ( err ) {
t . Error ( err )
}
2015-12-21 16:47:26 -05:00
existingBlocks := [ ] int { 0 , 2 , 3 , 4 , 0 , 0 , 7 , 0 }
2017-01-17 07:33:48 +00:00
existingFile := setUpFile ( ignore . TempName ( "file" ) , existingBlocks )
2014-10-11 00:27:17 +01:00
requiredFile := existingFile
requiredFile . Blocks = blocks [ 1 : ]
requiredFile . Name = "file2"
2015-12-21 16:47:26 -05:00
m := setUpModel ( existingFile )
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2014-10-11 00:27:17 +01:00
copyChan := make ( chan copyBlocksState )
pullChan := make ( chan pullBlockState , 4 )
finisherChan := make ( chan * sharedPullerState , 1 )
// Run a single fetcher routine
2016-04-26 20:19:30 +00:00
go f . copierRoutine ( copyChan , pullChan , finisherChan )
2014-10-11 00:27:17 +01:00
2016-04-26 20:19:30 +00:00
f . handleFile ( requiredFile , copyChan , finisherChan )
2014-10-11 00:27:17 +01:00
pulls := [ ] pullBlockState { <- pullChan , <- pullChan , <- pullChan , <- pullChan }
finish := <- finisherChan
select {
case <- pullChan :
2016-12-14 23:30:29 +00:00
t . Fatal ( "Pull channel has data to be read" )
2014-10-11 00:27:17 +01:00
case <- finisherChan :
t . Fatal ( "Finisher channel has data to be read" )
default :
}
2016-04-15 10:59:41 +00:00
// Verify that the right blocks went into the pull list.
// They are pulled in random order.
for _ , idx := range [ ] int { 1 , 5 , 6 , 8 } {
found := false
block := blocks [ idx ]
for _ , pulledBlock := range pulls {
if string ( pulledBlock . block . Hash ) == string ( block . Hash ) {
found = true
break
}
}
if ! found {
t . Errorf ( "Did not find block %s" , block . String ( ) )
2014-10-11 00:27:17 +01:00
}
2016-04-15 10:59:41 +00:00
if string ( finish . file . Blocks [ idx - 1 ] . Hash ) != string ( blocks [ idx ] . Hash ) {
t . Errorf ( "Block %d mismatch: %s != %s" , idx , finish . file . Blocks [ idx - 1 ] . String ( ) , blocks [ idx ] . String ( ) )
2014-10-11 00:27:17 +01:00
}
}
// Verify that the fetched blocks have actually been written to the temp file
2017-04-26 00:15:23 +00:00
blks , err := scanner . HashFile ( context . TODO ( ) , fs . DefaultFilesystem , tempFile , protocol . BlockSize , nil , false )
2014-10-11 00:27:17 +01:00
if err != nil {
t . Log ( err )
}
for _ , eq := range [ ] int { 2 , 3 , 4 , 7 } {
if string ( blks [ eq - 1 ] . Hash ) != string ( blocks [ eq ] . Hash ) {
t . Errorf ( "Block %d mismatch: %s != %s" , eq , blks [ eq - 1 ] . String ( ) , blocks [ eq ] . String ( ) )
}
}
finish . fd . Close ( )
os . Remove ( tempFile )
}
2014-10-22 15:24:11 +01:00
2016-12-14 23:30:29 +00:00
func TestWeakHash ( t * testing . T ) {
2017-01-17 07:33:48 +00:00
tempFile := filepath . Join ( "testdata" , ignore . TempName ( "weakhash" ) )
2016-12-14 23:30:29 +00:00
var shift int64 = 10
var size int64 = 1 << 20
expectBlocks := int ( size / protocol . BlockSize )
expectPulls := int ( shift / protocol . BlockSize )
if shift > 0 {
expectPulls ++
}
cleanup := func ( ) {
for _ , path := range [ ] string { tempFile , "testdata/weakhash" } {
os . Remove ( path )
}
}
cleanup ( )
defer cleanup ( )
f , err := os . Create ( "testdata/weakhash" )
if err != nil {
t . Error ( err )
}
defer f . Close ( )
_ , err = io . CopyN ( f , rand . Reader , size )
if err != nil {
t . Error ( err )
}
info , err := f . Stat ( )
if err != nil {
t . Error ( err )
}
// Create two files, second file has `shifted` bytes random prefix, yet
// both are of the same length, for example:
// File 1: abcdefgh
// File 2: xyabcdef
f . Seek ( 0 , os . SEEK_SET )
2017-04-26 00:15:23 +00:00
existing , err := scanner . Blocks ( context . TODO ( ) , f , protocol . BlockSize , size , nil , true )
2016-12-14 23:30:29 +00:00
if err != nil {
t . Error ( err )
}
f . Seek ( 0 , os . SEEK_SET )
remainder := io . LimitReader ( f , size - shift )
prefix := io . LimitReader ( rand . Reader , shift )
nf := io . MultiReader ( prefix , remainder )
2017-04-26 00:15:23 +00:00
desired , err := scanner . Blocks ( context . TODO ( ) , nf , protocol . BlockSize , size , nil , true )
2016-12-14 23:30:29 +00:00
if err != nil {
t . Error ( err )
}
existingFile := protocol . FileInfo {
Name : "weakhash" ,
Blocks : existing ,
Size : size ,
ModifiedS : info . ModTime ( ) . Unix ( ) ,
ModifiedNs : int32 ( info . ModTime ( ) . Nanosecond ( ) ) ,
}
desiredFile := protocol . FileInfo {
Name : "weakhash" ,
Size : size ,
Blocks : desired ,
ModifiedS : info . ModTime ( ) . Unix ( ) + 1 ,
}
// Setup the model/pull environment
m := setUpModel ( existingFile )
2016-12-16 22:23:35 +00:00
fo := setUpSendReceiveFolder ( m )
2016-12-14 23:30:29 +00:00
copyChan := make ( chan copyBlocksState )
pullChan := make ( chan pullBlockState , expectBlocks )
finisherChan := make ( chan * sharedPullerState , 1 )
// Run a single fetcher routine
go fo . copierRoutine ( copyChan , pullChan , finisherChan )
// Test 1 - no weak hashing, file gets fully repulled (`expectBlocks` pulls).
2017-01-04 21:04:13 +00:00
fo . WeakHashThresholdPct = 101
2016-12-14 23:30:29 +00:00
fo . handleFile ( desiredFile , copyChan , finisherChan )
var pulls [ ] pullBlockState
for len ( pulls ) < expectBlocks {
select {
case pull := <- pullChan :
pulls = append ( pulls , pull )
2016-12-16 12:05:27 +00:00
case <- time . After ( 10 * time . Second ) :
t . Errorf ( "timed out, got %d pulls expected %d" , len ( pulls ) , expectPulls )
2016-12-14 23:30:29 +00:00
}
}
finish := <- finisherChan
select {
case <- pullChan :
t . Fatal ( "Pull channel has data to be read" )
case <- finisherChan :
t . Fatal ( "Finisher channel has data to be read" )
default :
}
finish . fd . Close ( )
if err := os . Remove ( tempFile ) ; err != nil && ! os . IsNotExist ( err ) {
t . Error ( err )
}
// Test 2 - using weak hash, expectPulls blocks pulled.
2017-01-04 21:04:13 +00:00
fo . WeakHashThresholdPct = - 1
2016-12-14 23:30:29 +00:00
fo . handleFile ( desiredFile , copyChan , finisherChan )
pulls = pulls [ : 0 ]
for len ( pulls ) < expectPulls {
select {
case pull := <- pullChan :
pulls = append ( pulls , pull )
2016-12-16 12:05:27 +00:00
case <- time . After ( 10 * time . Second ) :
t . Errorf ( "timed out, got %d pulls expected %d" , len ( pulls ) , expectPulls )
2016-12-14 23:30:29 +00:00
}
}
finish = <- finisherChan
finish . fd . Close ( )
expectShifted := expectBlocks - expectPulls
if finish . copyOriginShifted != expectShifted {
t . Errorf ( "did not copy %d shifted" , expectShifted )
}
}
2014-10-22 15:24:11 +01:00
// Test that updating a file removes it's old blocks from the blockmap
func TestCopierCleanup ( t * testing . T ) {
2015-01-18 02:12:06 +01:00
iterFn := func ( folder , file string , index int32 ) bool {
2014-10-22 15:24:11 +01:00
return true
}
// Create a file
2015-12-21 16:47:26 -05:00
file := setUpFile ( "test" , [ ] int { 0 } )
m := setUpModel ( file )
2014-10-22 15:24:11 +01:00
file . Blocks = [ ] protocol . BlockInfo { blocks [ 1 ] }
2015-03-25 22:37:23 +01:00
file . Version = file . Version . Update ( protocol . LocalDeviceID . Short ( ) )
2014-10-22 15:24:11 +01:00
// Update index (removing old blocks)
2016-05-19 00:19:26 +00:00
m . updateLocalsFromScanning ( "default" , [ ] protocol . FileInfo { file } )
2014-10-22 15:24:11 +01:00
2015-09-04 12:01:00 +02:00
if m . finder . Iterate ( folders , blocks [ 0 ] . Hash , iterFn ) {
2014-10-22 15:24:11 +01:00
t . Error ( "Unexpected block found" )
}
2015-09-04 12:01:00 +02:00
if ! m . finder . Iterate ( folders , blocks [ 1 ] . Hash , iterFn ) {
2014-10-22 15:24:11 +01:00
t . Error ( "Expected block not found" )
}
file . Blocks = [ ] protocol . BlockInfo { blocks [ 0 ] }
2015-03-25 22:37:23 +01:00
file . Version = file . Version . Update ( protocol . LocalDeviceID . Short ( ) )
2014-10-22 15:24:11 +01:00
// Update index (removing old blocks)
2016-05-19 00:19:26 +00:00
m . updateLocalsFromScanning ( "default" , [ ] protocol . FileInfo { file } )
2014-10-22 15:24:11 +01:00
2015-09-04 12:01:00 +02:00
if ! m . finder . Iterate ( folders , blocks [ 0 ] . Hash , iterFn ) {
2014-10-22 15:24:11 +01:00
t . Error ( "Unexpected block found" )
}
2015-09-04 12:01:00 +02:00
if m . finder . Iterate ( folders , blocks [ 1 ] . Hash , iterFn ) {
2014-10-22 15:24:11 +01:00
t . Error ( "Expected block not found" )
}
}
2014-10-24 23:20:08 +01:00
2014-12-28 23:11:32 +00:00
// Make sure that the copier routine hashes the content when asked, and pulls
// if it fails to find the block.
2014-10-24 23:20:08 +01:00
func TestLastResortPulling ( t * testing . T ) {
// Add a file to index (with the incorrect block representation, as content
// doesn't actually match the block list)
2015-12-21 16:47:26 -05:00
file := setUpFile ( "empty" , [ ] int { 0 } )
m := setUpModel ( file )
2014-10-24 23:20:08 +01:00
// Pretend that we are handling a new file of the same content but
// with a different name (causing to copy that particular block)
file . Name = "newfile"
2015-01-18 02:12:06 +01:00
iterFn := func ( folder , file string , index int32 ) bool {
2014-10-24 23:20:08 +01:00
return true
}
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2014-10-24 23:20:08 +01:00
copyChan := make ( chan copyBlocksState )
pullChan := make ( chan pullBlockState , 1 )
finisherChan := make ( chan * sharedPullerState , 1 )
2014-12-28 23:11:32 +00:00
// Run a single copier routine
2016-04-26 20:19:30 +00:00
go f . copierRoutine ( copyChan , pullChan , finisherChan )
2014-10-24 23:20:08 +01:00
2016-04-26 20:19:30 +00:00
f . handleFile ( file , copyChan , finisherChan )
2014-10-24 23:20:08 +01:00
// Copier should hash empty file, realise that the region it has read
// doesn't match the hash which was advertised by the block map, fix it
// and ask to pull the block.
<- pullChan
// Verify that it did fix the incorrect hash.
2015-09-04 12:01:00 +02:00
if m . finder . Iterate ( folders , blocks [ 0 ] . Hash , iterFn ) {
2014-10-24 23:20:08 +01:00
t . Error ( "Found unexpected block" )
}
2015-09-04 12:01:00 +02:00
if ! m . finder . Iterate ( folders , scanner . SHA256OfNothing , iterFn ) {
2014-10-24 23:20:08 +01:00
t . Error ( "Expected block not found" )
}
( <- finisherChan ) . fd . Close ( )
2017-01-17 07:33:48 +00:00
os . Remove ( filepath . Join ( "testdata" , ignore . TempName ( "newfile" ) ) )
2014-10-24 23:20:08 +01:00
}
2015-01-07 23:12:12 +00:00
func TestDeregisterOnFailInCopy ( t * testing . T ) {
2015-12-21 16:47:26 -05:00
file := setUpFile ( "filex" , [ ] int { 0 , 2 , 0 , 0 , 5 , 0 , 0 , 8 } )
2017-01-17 07:33:48 +00:00
defer os . Remove ( "testdata/" + ignore . TempName ( "filex" ) )
2015-01-07 23:12:12 +00:00
2015-10-31 12:31:25 +01:00
db := db . OpenMemory ( )
2015-01-07 23:12:12 +00:00
2017-05-22 19:58:33 +00:00
m := NewModel ( defaultConfig , protocol . LocalDeviceID , "syncthing" , "dev" , db , nil )
2015-03-04 23:33:48 +00:00
m . AddFolder ( defaultFolderConfig )
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2015-01-07 23:12:12 +00:00
// queue.Done should be called by the finisher routine
2016-08-06 13:05:59 +00:00
f . queue . Push ( "filex" , 0 , time . Time { } )
2016-04-26 20:19:30 +00:00
f . queue . Pop ( )
2015-01-07 23:12:12 +00:00
2016-04-26 20:19:30 +00:00
if f . queue . lenProgress ( ) != 1 {
2015-01-07 23:12:12 +00:00
t . Fatal ( "Expected file in progress" )
}
copyChan := make ( chan copyBlocksState )
pullChan := make ( chan pullBlockState )
finisherBufferChan := make ( chan * sharedPullerState )
finisherChan := make ( chan * sharedPullerState )
2016-04-26 20:19:30 +00:00
go f . copierRoutine ( copyChan , pullChan , finisherBufferChan )
go f . finisherRoutine ( finisherChan )
2015-01-07 23:12:12 +00:00
2016-04-26 20:19:30 +00:00
f . handleFile ( file , copyChan , finisherChan )
2015-01-07 23:12:12 +00:00
2015-04-28 18:34:55 +03:00
// Receive a block at puller, to indicate that at least a single copier
2015-01-07 23:12:12 +00:00
// loop has been performed.
toPull := <- pullChan
// Wait until copier is trying to pass something down to the puller again
time . Sleep ( 100 * time . Millisecond )
// Close the file
toPull . sharedPullerState . fail ( "test" , os . ErrNotExist )
// Unblock copier
<- pullChan
select {
case state := <- finisherBufferChan :
// At this point the file should still be registered with both the job
// queue, and the progress emitter. Verify this.
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 1 || f . queue . lenProgress ( ) != 1 || f . queue . lenQueued ( ) != 0 {
2015-01-07 23:12:12 +00:00
t . Fatal ( "Could not find file" )
}
// Pass the file down the real finisher, and give it time to consume
finisherChan <- state
time . Sleep ( 100 * time . Millisecond )
2015-10-14 14:38:13 +09:00
state . mut . Lock ( )
stateFd := state . fd
state . mut . Unlock ( )
if stateFd != nil {
2015-01-07 23:12:12 +00:00
t . Fatal ( "File not closed?" )
}
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 0 || f . queue . lenProgress ( ) != 0 || f . queue . lenQueued ( ) != 0 {
t . Fatal ( "Still registered" , f . model . progressEmitter . lenRegistry ( ) , f . queue . lenProgress ( ) , f . queue . lenQueued ( ) )
2015-01-07 23:12:12 +00:00
}
// Doing it again should have no effect
finisherChan <- state
time . Sleep ( 100 * time . Millisecond )
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 0 || f . queue . lenProgress ( ) != 0 || f . queue . lenQueued ( ) != 0 {
t . Fatal ( "Still registered" , f . model . progressEmitter . lenRegistry ( ) , f . queue . lenProgress ( ) , f . queue . lenQueued ( ) )
2015-01-07 23:12:12 +00:00
}
case <- time . After ( time . Second ) :
t . Fatal ( "Didn't get anything to the finisher" )
}
}
func TestDeregisterOnFailInPull ( t * testing . T ) {
2015-12-21 16:47:26 -05:00
file := setUpFile ( "filex" , [ ] int { 0 , 2 , 0 , 0 , 5 , 0 , 0 , 8 } )
2017-01-17 07:33:48 +00:00
defer os . Remove ( "testdata/" + ignore . TempName ( "filex" ) )
2015-01-07 23:12:12 +00:00
2015-10-31 12:31:25 +01:00
db := db . OpenMemory ( )
2017-05-22 19:58:33 +00:00
m := NewModel ( defaultConfig , protocol . LocalDeviceID , "syncthing" , "dev" , db , nil )
2015-03-04 23:33:48 +00:00
m . AddFolder ( defaultFolderConfig )
2015-01-07 23:12:12 +00:00
2016-12-16 22:23:35 +00:00
f := setUpSendReceiveFolder ( m )
2015-01-07 23:12:12 +00:00
// queue.Done should be called by the finisher routine
2016-08-06 13:05:59 +00:00
f . queue . Push ( "filex" , 0 , time . Time { } )
2016-04-26 20:19:30 +00:00
f . queue . Pop ( )
2015-01-07 23:12:12 +00:00
2016-04-26 20:19:30 +00:00
if f . queue . lenProgress ( ) != 1 {
2015-01-07 23:12:12 +00:00
t . Fatal ( "Expected file in progress" )
}
copyChan := make ( chan copyBlocksState )
pullChan := make ( chan pullBlockState )
finisherBufferChan := make ( chan * sharedPullerState )
finisherChan := make ( chan * sharedPullerState )
2016-04-26 20:19:30 +00:00
go f . copierRoutine ( copyChan , pullChan , finisherBufferChan )
go f . pullerRoutine ( pullChan , finisherBufferChan )
go f . finisherRoutine ( finisherChan )
2015-01-07 23:12:12 +00:00
2016-04-26 20:19:30 +00:00
f . handleFile ( file , copyChan , finisherChan )
2015-01-07 23:12:12 +00:00
2015-11-11 21:20:34 -05:00
// Receive at finisher, we should error out as puller has nowhere to pull
2015-01-07 23:12:12 +00:00
// from.
select {
case state := <- finisherBufferChan :
// At this point the file should still be registered with both the job
// queue, and the progress emitter. Verify this.
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 1 || f . queue . lenProgress ( ) != 1 || f . queue . lenQueued ( ) != 0 {
2015-01-07 23:12:12 +00:00
t . Fatal ( "Could not find file" )
}
// Pass the file down the real finisher, and give it time to consume
finisherChan <- state
time . Sleep ( 100 * time . Millisecond )
2015-10-14 14:38:13 +09:00
state . mut . Lock ( )
stateFd := state . fd
state . mut . Unlock ( )
if stateFd != nil {
2015-01-07 23:12:12 +00:00
t . Fatal ( "File not closed?" )
}
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 0 || f . queue . lenProgress ( ) != 0 || f . queue . lenQueued ( ) != 0 {
t . Fatal ( "Still registered" , f . model . progressEmitter . lenRegistry ( ) , f . queue . lenProgress ( ) , f . queue . lenQueued ( ) )
2015-01-07 23:12:12 +00:00
}
// Doing it again should have no effect
finisherChan <- state
time . Sleep ( 100 * time . Millisecond )
2016-04-26 20:19:30 +00:00
if f . model . progressEmitter . lenRegistry ( ) != 0 || f . queue . lenProgress ( ) != 0 || f . queue . lenQueued ( ) != 0 {
t . Fatal ( "Still registered" , f . model . progressEmitter . lenRegistry ( ) , f . queue . lenProgress ( ) , f . queue . lenQueued ( ) )
2015-01-07 23:12:12 +00:00
}
case <- time . After ( time . Second ) :
t . Fatal ( "Didn't get anything to the finisher" )
}
}