| 
									
										
										
										
											2016-08-19 07:23:42 +08:00
										 |  |  | package cmd | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | 	"bufio" | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2016-09-24 04:32:51 +08:00
										 |  |  | 	"encoding/xml" | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 	"io" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | 	"net/http/httptest" | 
					
						
							| 
									
										
										
										
											2016-10-21 17:39:37 +08:00
										 |  |  | 	"reflect" | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/gorilla/mux" | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Implement a dummy flush writer.
 | 
					
						
							|  |  |  | type flushWriter struct { | 
					
						
							|  |  |  | 	io.Writer | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Flush writer is a dummy writer compatible with http.Flusher and http.ResponseWriter.
 | 
					
						
							|  |  |  | func (f *flushWriter) Flush()                            {} | 
					
						
							|  |  |  | func (f *flushWriter) Write(b []byte) (n int, err error) { return f.Writer.Write(b) } | 
					
						
							|  |  |  | func (f *flushWriter) Header() http.Header               { return http.Header{} } | 
					
						
							|  |  |  | func (f *flushWriter) WriteHeader(code int)              {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | func newFlushWriter(writer io.Writer) http.ResponseWriter { | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 	return &flushWriter{writer} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Tests write notification code.
 | 
					
						
							|  |  |  | func TestWriteNotification(t *testing.T) { | 
					
						
							|  |  |  | 	// Initialize a new test config.
 | 
					
						
							|  |  |  | 	root, err := newTestConfig("us-east-1") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("Unable to initialize test config %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(root) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var buffer bytes.Buffer | 
					
						
							|  |  |  | 	// Collection of test cases for each event writer.
 | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | 		writer http.ResponseWriter | 
					
						
							| 
									
										
										
										
											2016-08-05 13:01:58 +08:00
										 |  |  | 		event  map[string][]NotificationEvent | 
					
						
							|  |  |  | 		err    error | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		// Invalid input argument with writer `nil` - Test - 1
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			writer: nil, | 
					
						
							|  |  |  | 			event:  nil, | 
					
						
							|  |  |  | 			err:    errInvalidArgument, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Invalid input argument with event `nil` - Test - 2
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			writer: newFlushWriter(ioutil.Discard), | 
					
						
							|  |  |  | 			event:  nil, | 
					
						
							|  |  |  | 			err:    errInvalidArgument, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Unmarshal and write, validate last 5 bytes. - Test - 3
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			writer: newFlushWriter(&buffer), | 
					
						
							|  |  |  | 			event: map[string][]NotificationEvent{ | 
					
						
							|  |  |  | 				"Records": {newNotificationEvent(eventData{ | 
					
						
							|  |  |  | 					Type:   ObjectCreatedPut, | 
					
						
							|  |  |  | 					Bucket: "testbucket", | 
					
						
							|  |  |  | 					ObjInfo: ObjectInfo{ | 
					
						
							|  |  |  | 						Name: "key", | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					ReqParams: map[string]string{ | 
					
						
							|  |  |  | 						"ip": "10.1.10.1", | 
					
						
							|  |  |  | 					}}), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			err: nil, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Validates all the testcases for writing notification.
 | 
					
						
							|  |  |  | 	for _, testCase := range testCases { | 
					
						
							|  |  |  | 		err := writeNotification(testCase.writer, testCase.event) | 
					
						
							|  |  |  | 		if err != testCase.err { | 
					
						
							|  |  |  | 			t.Errorf("Unable to write notification %s", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Validates if the ending string has 'crlf'
 | 
					
						
							|  |  |  | 		if err == nil && !bytes.HasSuffix(buffer.Bytes(), crlf) { | 
					
						
							|  |  |  | 			buf := buffer.Bytes()[buffer.Len()-5 : 0] | 
					
						
							|  |  |  | 			t.Errorf("Invalid suffix found from the writer last 5 bytes %s, expected `\r\n`", string(buf)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Not printing 'buf' on purpose, validates look for string '10.1.10.1'.
 | 
					
						
							|  |  |  | 		if err == nil && !bytes.Contains(buffer.Bytes(), []byte("10.1.10.1")) { | 
					
						
							|  |  |  | 			// Enable when debugging)
 | 
					
						
							|  |  |  | 			// fmt.Println(string(buffer.Bytes()))
 | 
					
						
							|  |  |  | 			t.Errorf("Requested content couldn't be found, expected `10.1.10.1`") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 08:41:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestSendBucketNotification(t *testing.T) { | 
					
						
							|  |  |  | 	// Initialize a new test config.
 | 
					
						
							|  |  |  | 	root, err := newTestConfig("us-east-1") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("Unable to initialize test config %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(root) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	eventCh := make(chan []NotificationEvent) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create a Pipe with FlushWriter on the write-side and bufio.Scanner
 | 
					
						
							|  |  |  | 	// on the reader-side to receive notification over the listen channel in a
 | 
					
						
							|  |  |  | 	// synchronized manner.
 | 
					
						
							|  |  |  | 	pr, pw := io.Pipe() | 
					
						
							|  |  |  | 	fw := newFlushWriter(pw) | 
					
						
							|  |  |  | 	scanner := bufio.NewScanner(pr) | 
					
						
							|  |  |  | 	// Start a go-routine to wait for notification events.
 | 
					
						
							|  |  |  | 	go func(listenerCh <-chan []NotificationEvent) { | 
					
						
							|  |  |  | 		sendBucketNotification(fw, listenerCh) | 
					
						
							|  |  |  | 	}(eventCh) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Construct notification events to be passed on the events channel.
 | 
					
						
							|  |  |  | 	var events []NotificationEvent | 
					
						
							|  |  |  | 	evTypes := []EventName{ | 
					
						
							|  |  |  | 		ObjectCreatedPut, | 
					
						
							|  |  |  | 		ObjectCreatedPost, | 
					
						
							|  |  |  | 		ObjectCreatedCopy, | 
					
						
							|  |  |  | 		ObjectCreatedCompleteMultipartUpload, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for _, evType := range evTypes { | 
					
						
							|  |  |  | 		events = append(events, newNotificationEvent(eventData{ | 
					
						
							|  |  |  | 			Type: evType, | 
					
						
							|  |  |  | 		})) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Send notification events to the channel on which sendBucketNotification
 | 
					
						
							|  |  |  | 	// is waiting on.
 | 
					
						
							|  |  |  | 	eventCh <- events | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Read from the pipe connected to the ResponseWriter.
 | 
					
						
							|  |  |  | 	scanner.Scan() | 
					
						
							|  |  |  | 	notificationBytes := scanner.Bytes() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Close the read-end and send an empty notification event on the channel
 | 
					
						
							|  |  |  | 	// to signal sendBucketNotification to terminate.
 | 
					
						
							|  |  |  | 	pr.Close() | 
					
						
							|  |  |  | 	eventCh <- []NotificationEvent{} | 
					
						
							|  |  |  | 	close(eventCh) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Checking if the notification are the same as those sent over the channel.
 | 
					
						
							|  |  |  | 	var notifications map[string][]NotificationEvent | 
					
						
							|  |  |  | 	err = json.Unmarshal(notificationBytes, ¬ifications) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal("Failed to Unmarshal notification") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	records := notifications["Records"] | 
					
						
							|  |  |  | 	for i, rec := range records { | 
					
						
							|  |  |  | 		if rec.EventName == evTypes[i].String() { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		t.Errorf("Failed to receive %d event %s", i, evTypes[i].String()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 17:39:37 +08:00
										 |  |  | func TestGetBucketNotificationHandler(t *testing.T) { | 
					
						
							|  |  |  | 	ExecObjectLayerAPITest(t, testGetBucketNotificationHandler, []string{ | 
					
						
							|  |  |  | 		"GetBucketNotification", | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func testGetBucketNotificationHandler(obj ObjectLayer, instanceType, bucketName string, apiRouter http.Handler, | 
					
						
							|  |  |  | 	credentials credential, t *testing.T) { | 
					
						
							|  |  |  | 	// declare sample configs
 | 
					
						
							|  |  |  | 	filterRules := []filterRule{ | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Name:  "prefix", | 
					
						
							|  |  |  | 			Value: "minio", | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Name:  "suffix", | 
					
						
							|  |  |  | 			Value: "*.jpg", | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sampleSvcCfg := ServiceConfig{ | 
					
						
							|  |  |  | 		[]string{"s3:ObjectRemoved:*", "s3:ObjectCreated:*"}, | 
					
						
							|  |  |  | 		filterStruct{ | 
					
						
							|  |  |  | 			keyFilter{filterRules}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		"1", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sampleNotifCfg := notificationConfig{ | 
					
						
							|  |  |  | 		QueueConfigs: []queueConfig{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				ServiceConfig: sampleSvcCfg, | 
					
						
							|  |  |  | 				QueueARN:      "testqARN", | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rec := httptest.NewRecorder() | 
					
						
							|  |  |  | 	req, err := newTestSignedRequestV4("GET", getGetBucketNotificationURL("", bucketName), | 
					
						
							|  |  |  | 		0, nil, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	apiRouter.ServeHTTP(rec, req) | 
					
						
							|  |  |  | 	if rec.Code != http.StatusOK { | 
					
						
							|  |  |  | 		t.Fatalf("Unexpected http response %d", rec.Code) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err = persistNotificationConfig(bucketName, &sampleNotifCfg, obj); err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("Unable to save notification config %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rec = httptest.NewRecorder() | 
					
						
							|  |  |  | 	req, err = newTestSignedRequestV4("GET", getGetBucketNotificationURL("", bucketName), | 
					
						
							|  |  |  | 		0, nil, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	apiRouter.ServeHTTP(rec, req) | 
					
						
							|  |  |  | 	if rec.Code != http.StatusOK { | 
					
						
							|  |  |  | 		t.Fatalf("Unexpected http response %d", rec.Code) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	notificationBytes, err := ioutil.ReadAll(rec.Body) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("Unexpected error %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	nConfig := notificationConfig{} | 
					
						
							|  |  |  | 	if err = xml.Unmarshal(notificationBytes, &nConfig); err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("Unexpected XML received %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if sampleNotifCfg.QueueConfigs[0].QueueARN != nConfig.QueueConfigs[0].QueueARN { | 
					
						
							|  |  |  | 		t.Fatalf("Uexpected notification configs expected %#v, got %#v", sampleNotifCfg, nConfig) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(sampleNotifCfg.QueueConfigs[0].Events, nConfig.QueueConfigs[0].Events) { | 
					
						
							|  |  |  | 		t.Fatalf("Uexpected notification configs expected %#v, got %#v", sampleNotifCfg, nConfig) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | func TestListenBucketNotificationHandler(t *testing.T) { | 
					
						
							|  |  |  | 	ExecObjectLayerAPITest(t, testListenBucketNotificationHandler, []string{ | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 		"ListenBucketNotification", | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		"PutObject", | 
					
						
							| 
									
										
										
										
											2016-09-24 04:32:51 +08:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 17:39:37 +08:00
										 |  |  | func testListenBucketNotificationHandler(obj ObjectLayer, instanceType, bucketName string, apiRouter http.Handler, | 
					
						
							|  |  |  | 	credentials credential, t *testing.T) { | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	mux, ok := apiRouter.(*mux.Router) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		t.Fatal("Invalid mux router found") | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	registerS3PeerRPCRouter(mux) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	testServer := httptest.NewServer(apiRouter) | 
					
						
							|  |  |  | 	defer testServer.Close() | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	// setup port and minio addr
 | 
					
						
							|  |  |  | 	_, portStr, err := net.SplitHostPort(testServer.Listener.Addr().String()) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		t.Fatalf("Initialization error: %v", err) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	globalMinioPort, err = strconv.Atoi(portStr) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		t.Fatalf("Initialization error: %v", err) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 	globalMinioAddr = testServer.Listener.Addr().String() | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	// initialize the peer client(s)
 | 
					
						
							| 
									
										
										
										
											2016-10-19 03:49:24 +08:00
										 |  |  | 	initGlobalS3Peers([]storageEndPoint{}) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-01 07:56:36 +08:00
										 |  |  | 	invalidBucket := "Invalid\\Bucket" | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	noNotificationBucket := "nonotificationbucket" | 
					
						
							|  |  |  | 	// get random bucket name.
 | 
					
						
							|  |  |  | 	randBucket := getRandomBucketName() | 
					
						
							|  |  |  | 	for _, bucket := range []string{randBucket, noNotificationBucket} { | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 		err = obj.MakeBucket(bucket) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			// failed to create bucket, abort.
 | 
					
						
							|  |  |  | 			t.Fatalf("Failed to create bucket %s %s : %s", bucket, | 
					
						
							|  |  |  | 				instanceType, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-12 16:03:50 +08:00
										 |  |  | 	var testRec *httptest.ResponseRecorder | 
					
						
							|  |  |  | 	var testReq *http.Request | 
					
						
							|  |  |  | 	var tErr error | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	signatureMismatchError := getAPIError(ErrContentSHA256Mismatch) | 
					
						
							|  |  |  | 	tooBigPrefix := string(bytes.Repeat([]byte("a"), 1025)) | 
					
						
							|  |  |  | 	validEvents := []string{"s3:ObjectCreated:*", "s3:ObjectRemoved:*"} | 
					
						
							|  |  |  | 	invalidEvents := []string{"invalidEvent"} | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							|  |  |  | 		bucketName       string | 
					
						
							| 
									
										
										
										
											2016-10-13 02:02:15 +08:00
										 |  |  | 		prefixes         []string | 
					
						
							|  |  |  | 		suffixes         []string | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 		events           []string | 
					
						
							|  |  |  | 		expectedHTTPCode int | 
					
						
							|  |  |  | 		expectedAPIError string | 
					
						
							|  |  |  | 	}{ | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		{randBucket, []string{}, []string{}, invalidEvents, signatureMismatchError.HTTPStatusCode, "InvalidArgument"}, | 
					
						
							|  |  |  | 		{randBucket, []string{tooBigPrefix}, []string{}, validEvents, http.StatusBadRequest, "InvalidArgument"}, | 
					
						
							|  |  |  | 		{invalidBucket, []string{}, []string{}, validEvents, http.StatusBadRequest, "InvalidBucketName"}, | 
					
						
							|  |  |  | 		{randBucket, []string{}, []string{}, validEvents, signatureMismatchError.HTTPStatusCode, signatureMismatchError.Code}, | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i, test := range testCases { | 
					
						
							|  |  |  | 		testRec = httptest.NewRecorder() | 
					
						
							| 
									
										
										
										
											2016-10-01 05:32:13 +08:00
										 |  |  | 		testReq, tErr = newTestSignedRequestV4("GET", | 
					
						
							| 
									
										
										
										
											2016-10-13 02:02:15 +08:00
										 |  |  | 			getListenBucketNotificationURL("", test.bucketName, test.prefixes, test.suffixes, test.events), | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 			0, nil, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 		if tErr != nil { | 
					
						
							|  |  |  | 			t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, tErr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Set X-Amz-Content-SHA256 in header different from what was used to calculate Signature.
 | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		if test.expectedAPIError == "XAmzContentSHA256Mismatch" { | 
					
						
							|  |  |  | 			// Triggering a authentication failure.
 | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 			testReq.Header.Set("x-amz-content-sha256", "somethingElse") | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		apiRouter.ServeHTTP(testRec, testReq) | 
					
						
							|  |  |  | 		rspBytes, rErr := ioutil.ReadAll(testRec.Body) | 
					
						
							|  |  |  | 		if rErr != nil { | 
					
						
							|  |  |  | 			t.Errorf("Test %d: %s: Failed to read response body: <ERROR> %v", i+1, instanceType, rErr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var errXML APIErrorResponse | 
					
						
							|  |  |  | 		xErr := xml.Unmarshal(rspBytes, &errXML) | 
					
						
							|  |  |  | 		if xErr != nil { | 
					
						
							|  |  |  | 			t.Errorf("Test %d: %s: Failed to unmarshal error XML: <ERROR> %v", i+1, instanceType, xErr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if errXML.Code != test.expectedAPIError { | 
					
						
							|  |  |  | 			t.Errorf("Test %d: %s: Expected error code %s but received %s: <ERROR> %v", i+1, | 
					
						
							|  |  |  | 				instanceType, test.expectedAPIError, errXML.Code, err) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		if testRec.Code != test.expectedHTTPCode { | 
					
						
							|  |  |  | 			t.Errorf("Test %d: %s: expected HTTP code %d, but received %d: <ERROR> %v", | 
					
						
							|  |  |  | 				i+1, instanceType, test.expectedHTTPCode, testRec.Code, err) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Nil Object layer
 | 
					
						
							|  |  |  | 	nilAPIRouter := initTestAPIEndPoints(nil, []string{ | 
					
						
							| 
									
										
										
										
											2016-10-06 03:48:07 +08:00
										 |  |  | 		"ListenBucketNotification", | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 	testRec = httptest.NewRecorder() | 
					
						
							| 
									
										
										
										
											2016-10-01 05:32:13 +08:00
										 |  |  | 	testReq, tErr = newTestSignedRequestV4("GET", | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 		getListenBucketNotificationURL("", randBucket, []string{}, | 
					
						
							|  |  |  | 			[]string{"*.jpg"}, []string{ | 
					
						
							|  |  |  | 				"s3:ObjectCreated:*", | 
					
						
							|  |  |  | 				"s3:ObjectRemoved:*", | 
					
						
							|  |  |  | 			}), 0, nil, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	if tErr != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, tErr) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	nilAPIRouter.ServeHTTP(testRec, testReq) | 
					
						
							|  |  |  | 	if testRec.Code != http.StatusServiceUnavailable { | 
					
						
							|  |  |  | 		t.Errorf("Test %d: %s: expected HTTP code %d, but received %d: <ERROR> %v", | 
					
						
							|  |  |  | 			1, instanceType, http.StatusServiceUnavailable, testRec.Code, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 16:25:17 +08:00
										 |  |  | 	testRec = httptest.NewRecorder() | 
					
						
							|  |  |  | 	testReq, tErr = newTestSignedRequestV4("GET", | 
					
						
							|  |  |  | 		getListenBucketNotificationURL("", randBucket, []string{}, []string{}, validEvents), | 
					
						
							|  |  |  | 		0, nil, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 	if tErr != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, tErr) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	globalObjLayerMutex.Lock() | 
					
						
							|  |  |  | 	globalObjectAPI = obj | 
					
						
							|  |  |  | 	globalObjLayerMutex.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go apiRouter.ServeHTTP(testRec, testReq) | 
					
						
							|  |  |  | 	var wg sync.WaitGroup | 
					
						
							|  |  |  | 	wg.Add(1) | 
					
						
							|  |  |  | 	go func() { | 
					
						
							|  |  |  | 		defer wg.Done() | 
					
						
							|  |  |  | 		rec := httptest.NewRecorder() | 
					
						
							|  |  |  | 		buf := bytes.NewReader([]byte("hello, world")) | 
					
						
							|  |  |  | 		req, rerr := newTestSignedRequestV4("PUT", getPutObjectURL("", randBucket, "jeezus"), | 
					
						
							|  |  |  | 			int64(buf.Len()), buf, credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 		if rerr != nil { | 
					
						
							|  |  |  | 			t.Fatalf("%s: Failed to create HTTP testRequest for ListenBucketNotification: <ERROR> %v", instanceType, rerr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		apiRouter.ServeHTTP(rec, req) | 
					
						
							|  |  |  | 		if rec.Code != http.StatusOK { | 
					
						
							|  |  |  | 			t.Fatalf("Unexpected http reply %d should be %d", rec.Code, http.StatusOK) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 	wg.Wait() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bio := bufio.NewScanner(testRec.Body) | 
					
						
							|  |  |  | 	// Unmarshal each line, returns marshalled values.
 | 
					
						
							|  |  |  | 	for bio.Scan() { | 
					
						
							|  |  |  | 		var notificationInfo struct { | 
					
						
							|  |  |  | 			Records []NotificationEvent | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err = json.Unmarshal(bio.Bytes(), ¬ificationInfo); err != nil { | 
					
						
							|  |  |  | 			t.Fatalf("%s: Unable to marshal: <ERROR> %v", instanceType, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Send notifications on channel only if there are events received.
 | 
					
						
							|  |  |  | 		if len(notificationInfo.Records) == 0 { | 
					
						
							|  |  |  | 			t.Fatalf("%s: Expected notification events, received none", instanceType) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Look for any underlying errors.
 | 
					
						
							|  |  |  | 	if err = bio.Err(); err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Server connection closed prematurely %s", instanceType, err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-22 08:12:56 +08:00
										 |  |  | func testRemoveNotificationConfig(obj ObjectLayer, instanceType, bucketName string, apiRouter http.Handler, | 
					
						
							|  |  |  | 	credentials credential, t *testing.T) { | 
					
						
							| 
									
										
										
										
											2016-10-01 07:56:36 +08:00
										 |  |  | 	invalidBucket := "Invalid\\Bucket" | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 	// get random bucket name.
 | 
					
						
							| 
									
										
										
										
											2016-10-22 08:12:56 +08:00
										 |  |  | 	randBucket := bucketName | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	sampleNotificationBytes := []byte("<NotificationConfiguration><TopicConfiguration>" + | 
					
						
							|  |  |  | 		"<Event>s3:ObjectCreated:*</Event><Event>s3:ObjectRemoved:*</Event><Filter>" + | 
					
						
							|  |  |  | 		"<S3Key></S3Key></Filter><Id></Id><Topic>arn:minio:sns:us-east-1:1474332374:listen</Topic>" + | 
					
						
							|  |  |  | 		"</TopicConfiguration></NotificationConfiguration>") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Set sample bucket notification on randBucket.
 | 
					
						
							|  |  |  | 	testRec := httptest.NewRecorder() | 
					
						
							| 
									
										
										
										
											2016-10-01 05:32:13 +08:00
										 |  |  | 	testReq, tErr := newTestSignedRequestV4("PUT", getPutBucketNotificationURL("", randBucket), | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | 		int64(len(sampleNotificationBytes)), bytes.NewReader(sampleNotificationBytes), | 
					
						
							|  |  |  | 		credentials.AccessKeyID, credentials.SecretAccessKey) | 
					
						
							|  |  |  | 	if tErr != nil { | 
					
						
							|  |  |  | 		t.Fatalf("%s: Failed to create HTTP testRequest for PutBucketNotification: <ERROR> %v", instanceType, tErr) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	apiRouter.ServeHTTP(testRec, testReq) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							|  |  |  | 		bucketName  string | 
					
						
							|  |  |  | 		expectedErr error | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		{invalidBucket, BucketNameInvalid{Bucket: invalidBucket}}, | 
					
						
							|  |  |  | 		{randBucket, nil}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i, test := range testCases { | 
					
						
							|  |  |  | 		tErr := removeNotificationConfig(test.bucketName, obj) | 
					
						
							|  |  |  | 		if tErr != test.expectedErr { | 
					
						
							|  |  |  | 			t.Errorf("Test %d: %s expected error %v, but received %v", i+1, instanceType, test.expectedErr, tErr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestRemoveNotificationConfig(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2016-10-22 08:12:56 +08:00
										 |  |  | 	ExecObjectLayerAPITest(t, testRemoveNotificationConfig, []string{ | 
					
						
							|  |  |  | 		"PutBucketNotification", | 
					
						
							|  |  |  | 		"ListenBucketNotification", | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-09-28 16:08:03 +08:00
										 |  |  | } |