Skip to content

Commit

Permalink
Added virtual URL style support and working tests.
Browse files Browse the repository at this point in the history
  • Loading branch information
dcardOrtus committed Sep 19, 2023
1 parent 6a6cd41 commit bdcfcd3
Show file tree
Hide file tree
Showing 14 changed files with 724 additions and 29 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -22,3 +22,4 @@ modules/**

# log files
logs/**
.idea/
6 changes: 5 additions & 1 deletion ModuleConfig.cfc
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,8 @@ component {
defaultBlockPublicAcls : false,
defaultIgnorePublicAcls : false,
defaultBlockPublicPolicy : false,
defaultRestrictPublicBuckets : false
defaultRestrictPublicBuckets : false,
urlStyle : "path"
};
}

Expand Down Expand Up @@ -83,6 +84,9 @@ component {
.initArg(
name = "defaultRestrictPublicBuckets",
value = variables.settings.defaultRestrictPublicBuckets
).initArg(
name = "urlStyle",
value = variables.settings.urlStyle
);
binder.map( "Sv4Util@s3sdk" ).to( "#moduleMapping#.models.AmazonS3" );

Expand Down
80 changes: 54 additions & 26 deletions models/AmazonS3.cfc
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@
* s3_accessKey : The Amazon access key
* s3_secretKey : The Amazon secret key
* s3_encryption_charset : encryptyion charset (Optional, defaults to utf-8)
* s3_ssl : Whether to use ssl on all cals or not (Optional, defaults to false)
* s3_ssl : Whether to use ssl on all calls or not (Optional, defaults to false)
*/
component accessors="true" singleton {

Expand Down Expand Up @@ -59,7 +59,7 @@ component accessors="true" singleton {
property name="defaultIgnorePublicAcls";
property name="defaultBlockPublicPolicy";
property name="defaultRestrictPublicBuckets";

property name="urlStyle";

// STATIC Contsants
this.ACL_PRIVATE = "private";
Expand Down Expand Up @@ -109,6 +109,7 @@ component accessors="true" singleton {
* @defaultIgnorePublicAcls Specifies whether Amazon S3 should block public bucket policies for this bucket. Setting this element to TRUE causes Amazon S3 to reject calls to PUT Bucket policy if the specified bucket policy allows public access.
* @defaultBlockPublicPolicy Specifies whether Amazon S3 should ignore public ACLs for this bucket and objects in this bucket. Setting this element to TRUE causes Amazon S3 to ignore all public ACLs on this bucket and objects in this bucket.
* @defaultRestrictPublicBuckets Specifies whether Amazon S3 should restrict public bucket policies for this bucket. Setting this element to TRUE restricts access to this bucket to only AWS service principals and authorized users within this account if the bucket has a public policy.
* @urlStyle Specifies the format of the URL whether it is the `path` format or `virtual` format. Defaults to path. For more information see https://docs.aws.amazon.com/AmazonS3/latest/userguide/VirtualHosting.html
*
* @return An AmazonS3 instance.
*/
Expand Down Expand Up @@ -139,7 +140,8 @@ component accessors="true" singleton {
boolean defaultBlockPublicAcls = false,
boolean defaultIgnorePublicAcls = false,
boolean defaultBlockPublicPolicy = false,
boolean defaultRestrictPublicBuckets = false
boolean defaultRestrictPublicBuckets = false,
string urlStyle = "path"
){
if ( arguments.awsDomain == "amazonaws.com" && arguments.awsRegion == "" ) {
arguments.awsRegion = "us-east-1";
Expand Down Expand Up @@ -176,6 +178,7 @@ component accessors="true" singleton {
variables.defaultIgnorePublicAcls = arguments.defaultIgnorePublicAcls;
variables.defaultBlockPublicPolicy = arguments.defaultBlockPublicPolicy;
variables.defaultRestrictPublicBuckets = arguments.defaultRestrictPublicBuckets;
variables.urlStyle = arguments.urlStyle;

// Construct the SSL Domain
setSSL( arguments.ssl );
Expand Down Expand Up @@ -261,21 +264,30 @@ component accessors="true" singleton {
/**
* This function builds variables.UrlEndpoint and variables.URLEndpointHostname according to credentials and ssl configuration, usually called after init() for you automatically.
*/
AmazonS3 function buildUrlEndpoint(){
AmazonS3 function buildUrlEndpoint(string bucketName){
// Build accordingly
var URLEndPointProtocol = ( variables.ssl ) ? "https://" : "http://";

var hostnameComponents = [];
if ( variables.awsDomain contains "amazonaws.com" ) {
hostnameComponents.append( "s3" );
}
if ( len( variables.awsRegion ) ) {
hostnameComponents.append( variables.awsRegion );
}
hostnameComponents.append( variables.awsDomain );
variables.URLEndpointHostname = arrayToList( hostnameComponents, "." );
variables.URLEndpoint = URLEndpointProtocol & variables.URLEndpointHostname;

if(variables.urlStyle=="path") {
if ( variables.awsDomain contains "amazonaws.com" ) {
hostnameComponents.append( "s3" );
}
if ( len( variables.awsRegion ) ) {
hostnameComponents.append( variables.awsRegion );
}
} else if(variables.urlStyle=="virtual"){
if(!isNull(arguments.bucketName)){
hostnameComponents.append(arguments.bucketName);
}
hostnameComponents.append("s3");
if (len(variables.awsRegion)) {
hostnameComponents.append(variables.awsRegion);
}
}
hostnameComponents.append(variables.awsDomain);
variables.URLEndpointHostname = arrayToList(hostnameComponents, '.');
variables.URLEndpoint = URLEndpointProtocol & variables.URLEndpointHostname;
return this;
}

Expand Down Expand Up @@ -858,7 +870,7 @@ component accessors="true" singleton {

var finalized = s3Request(
method = "POST",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
timeout = arguments.HTTPTimeout,
parameters = { "uploadId" : uploadId },
body = finalizeBody
Expand All @@ -875,7 +887,7 @@ component accessors="true" singleton {
} catch ( any e ) {
s3Request(
method = "DELETE",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
timeout = arguments.HTTPTimeout,
parameters = { "uploadId" : uploadId }
);
Expand Down Expand Up @@ -1034,7 +1046,7 @@ component accessors="true" singleton {

var results = s3Request(
method = "PUT",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
body = arguments.data,
timeout = arguments.HTTPTimeout,
headers = headers
Expand Down Expand Up @@ -1065,7 +1077,7 @@ component accessors="true" singleton {
var headers = applyEncryptionHeaders( {}, arguments );
var results = s3Request(
method = "HEAD",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
headers = headers
);

Expand Down Expand Up @@ -1096,7 +1108,7 @@ component accessors="true" singleton {
requireBucketName( arguments.bucketName );
var results = s3Request(
method = "GET",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
parameters = { "acl" : "" },
throwOnError = throwOnError
);
Expand Down Expand Up @@ -1139,7 +1151,7 @@ component accessors="true" singleton {
requireBucketName( arguments.bucketName );
var results = s3Request(
method = "HEAD",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
throwOnError = false
);
var status_code = results.responseHeader.status_code ?: 0;
Expand Down Expand Up @@ -1292,7 +1304,7 @@ component accessors="true" singleton {

return s3Request(
method = "POST",
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
timeout = arguments.HTTPTimeout,
headers = headers,
parameters = { "uploads" : true },
Expand Down Expand Up @@ -1320,14 +1332,15 @@ component accessors="true" singleton {
required string uri,
string encryptionKey = variables.defaultEncryptionKey
){
buildUrlEndpoint( arguments.bucketName);
requireBucketName( arguments.bucketName );

var headers = applyEncryptionHeaders( {}, arguments );

var results = s3Request(
method = "GET",
headers = headers,
resource = arguments.bucketName & "/" & arguments.uri
resource = buildKeyName( arguments.uri,arguments.bucketName)
);
return results;
}
Expand Down Expand Up @@ -1365,7 +1378,7 @@ component accessors="true" singleton {
var results = s3Request(
method = "GET",
headers = headers,
resource = arguments.bucketName & "/" & arguments.uri,
resource = buildKeyName( arguments.uri,arguments.bucketName),
filename = arguments.filepath,
timeout = arguments.HTTPTimeout,
getAsBinary = arguments.getAsBinary,
Expand Down Expand Up @@ -1393,7 +1406,7 @@ component accessors="true" singleton {
boolean function deleteObject( required string bucketName = variables.defaultBucketName, required string uri ){
requireBucketName( arguments.bucketName );

var results = s3Request( method = "DELETE", resource = arguments.bucketName & "/" & arguments.uri );
var results = s3Request( method = "DELETE", resource = buildKeyName( arguments.uri,arguments.bucketName));

return results.responseheader.status_code == 204;
}
Expand Down Expand Up @@ -1858,8 +1871,12 @@ component accessors="true" singleton {
}

/**
* Determine mime type from the file extension
* */
* Determines mime type from the file extension
*
* @filePath The path to the file stored in S3.
*
* @return string
*/
string function getFileMimeType( required string filePath ){
var contentType = "binary/octet-stream";
if ( len( arguments.filePath ) ) {
Expand All @@ -1879,4 +1896,15 @@ component accessors="true" singleton {
return contentType;
}


/**
* Creates the s3 key name based on the format (path or virtual) from the bucket name and the object key
*
* @url The key for the file in question
* @bucketName The name of the bucket to use. Not needed if the urlStyle is `virtual`
**/
function buildKeyName(required string uri, string bucketName="" ){
return variables.urlStyle=="path" ? arguments.bucketName & (arguments.bucketName.len() ? "/" : "" ) & arguments.uri : arguments.uri;
}
}
8 changes: 6 additions & 2 deletions readme.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,8 @@ This SDK will be installed into a directory called `s3sdk` and then the SDK can
* @debug Used to turn debugging on or off outside of logbox. Defaults to false.
* @defaultEncryptionAlgorithm The default server side encryption algorithm to use. Usually "AES256". Not needed if using custom defaultEncryptionKey
* @defaultEncryptionKey The default base64 encoded AES 356 bit key for server side encryption.
*
* @urlStyle Specifies the format of the URL whether it is the `path` format or `virtual` format. Defaults to path. For more information see https://docs.aws.amazon.com/AmazonS3/latest/userguide/VirtualHosting.html
*
* @return An AmazonS3 instance.
*/
public AmazonS3 function init(
Expand All @@ -75,6 +76,7 @@ public AmazonS3 function init(
boolean debug= false,
string defaultEncryptionAlgorithm = "",
string defaultEncryptionKey = "",
string urlStyle = "path"
)
```

Expand Down Expand Up @@ -125,7 +127,9 @@ moduleSettings = {
// SSL mode or not on cfhttp calls and when generating put/get authenticated URLs: Defaults to true
ssl = true,
// Throw exceptions when s3 requests fail, else it swallows them up.
throwOnRequestError : true
throwOnRequestError : true,
// What format of endpoint to use whether path or virtual
urlStyle = "path"
}
};
```
Expand Down
52 changes: 52 additions & 0 deletions test-harness/tests/specs/models/AmazonS3/buildKeyName.cfc
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
/**
* My BDD Test
*/
component extends="coldbox.system.testing.BaseTestCase"{

/*********************************** LIFE CYCLE Methods ***********************************/
this.unloadColdbox=false;
// executes before all suites+specs in the run() method
function beforeAll(){
super.beforeAll();
}

// executes after all suites+specs in the run() method
//function afterAll(){

//}

/*********************************** BDD SUITES ***********************************/

function run( testResults, testBox ){
// all your suites go here.
describe( "The buildKeyName function should...", function(){
beforeEach(function(){
uri = mockData($num=1,$type="words:1")[1];
bucketName = mockData($num=1,$type="words:1")[1];

testObj = getInstance("AmazonS3@s3sdk");
});
it( "If the urlStyle is path and a bucket is submitted, return bucket\uri", function(){
testObj.setUrlStyle("path");
testme = testObj.buildKeyName(uri,bucketName);
expect(testme).tobe("#bucketName#/#uri#");
});
it( "If the urlStyle is path and a bucket is not submitted, return uri", function(){
testObj.setUrlStyle("path");
testme = testObj.buildKeyName(uri);
expect(testme).tobe(uri);
});
it( "If the urlStyle is path and the bucket is an empty string, return uri", function(){
testObj.setUrlStyle("path");
testme = testObj.buildKeyName(uri,"");
expect(testme).tobe(uri);
});
it( "If the urlStyle is virtual, return uri", function(){
testObj.setUrlStyle("virtual");
testme = testObj.buildKeyName(uri,bucketname);
expect(testme).tobe(uri);
});
});
}
}

Loading

0 comments on commit bdcfcd3

Please sign in to comment.