Exemple #1
0
def get_data_dl_s3_client():

    user_id = get_jwt_field(get_cookie_vars(app.current_request.headers),
                            'urs-user-id')

    session = get_role_session(user_id=user_id)
    params = {}

    params['config'] = bc_Config(**get_bcconfig(user_id))
    client = session.client('s3', **params)
    return client
Exemple #2
0
def get_s3_resource():
    """

    :return: subclass of boto3.resources.base.ServiceResource
    """
    params = {}
    # Swift signature compatability
    if os.getenv('S3_SIGNATURE_VERSION'):
        params['config'] = bc_Config(
            signature_version=os.getenv('S3_SIGNATURE_VERSION'))
    s3 = botoresource('s3', **params)
    return s3
Exemple #3
0
def get_bc_config_client(user_id):
    params = {}
    now = time.time()

    if user_id not in bc_client_cache:
        # This a new user, generate a new bc_Config client
        params['config'] = bc_Config(**get_bcconfig(user_id))
        session = get_role_session(user_id=user_id)
        bc_client_cache[user_id] = {
            "client": session.client('s3', **params),
            "timestamp": now
        }
    elif now - bc_client_cache[user_id]["timestamp"] >= (50 * 60):
        # Replace the client if is more than 50 minutes old
        log.info(f"Replacing old bc_Config_client for user {user_id}")
        params['config'] = bc_Config(**get_bcconfig(user_id))
        session = get_role_session(user_id=user_id)
        bc_client_cache[user_id] = {
            "client": session.client('s3', **params),
            "timestamp": now
        }

    return bc_client_cache[user_id]["client"]
Exemple #4
0
def get_s3_resource():
    """

    :return: subclass of boto3.resources.base.ServiceResource
    """
    global s3_resource  #pylint: disable=global-statement
    if not s3_resource:
        params = {}
        # Swift signature compatability
        if os.getenv('S3_SIGNATURE_VERSION'):
            params['config'] = bc_Config(
                signature_version=os.getenv('S3_SIGNATURE_VERSION'))
        s3_resource = botoresource('s3', **params)
    return s3_resource
Exemple #5
0
def get_data_dl_s3_client():

    cookievars = get_cookie_vars(app.current_request.headers)
    if cookievars:
        user_id = cookievars['urs-user-id']
    else:
        user_id = None
    session = get_role_session(user_id=user_id)
    params = {}
    BCCONFIG = {'user_agent': "Egress App for userid={0}".format(user_id)}
    if os.getenv('S3_SIGNATURE_VERSION'):
        BCCONFIG['signature_version'] = os.getenv('S3_SIGNATURE_VERSION')
    params['config'] = bc_Config(**BCCONFIG)
    client = session.client('s3', **params)
    return client
Exemple #6
0
def try_download_from_bucket(bucket, filename, user_profile):

    # Attempt to pull userid from profile
    user_id = None
    if isinstance(user_profile, dict):
        if 'urs-user-id' in user_profile:
            user_id = user_profile['urs-user-id']
        elif 'uid' in user_profile:
            user_id = user_profile['uid']
    log.info("User Id for download is {0}".format(user_id))

    is_in_region = check_in_region_request(
        app.current_request.context['identity']['sourceIp'])
    creds = get_role_creds(user_id, is_in_region)

    session = get_role_session(creds=creds, user_id=user_id)

    try:
        bucket_region = get_bucket_region(session, bucket)
    except ClientError as e:
        log.error(
            f'ClientError while {user_id} tried downloading {bucket}/{filename}: {e}'
        )
        template_vars = {
            'contentstring': 'There was a problem accessing download data.',
            'title': 'Data Not Available'
        }
        headers = {}
        return make_html_response(template_vars, headers, 500, 'error.html')

    log.debug('this region: {}'.format(
        os.getenv('AWS_DEFAULT_REGION', 'env var doesnt exist')))
    if bucket_region != os.getenv('AWS_DEFAULT_REGION'):
        log.warning(
            "bucket {0} is in region {1}, we are in region {2}! " +
            "This is double egress in Proxy mode!".format(
                bucket, bucket_region, os.getenv('AWS_DEFAULT_REGION')))
    params = {}
    # now that we know where the bucket is, connect in THAT region
    params['config'] = bc_Config(**get_bcconfig(user_id))
    client = session.client('s3', bucket_region, **params)

    log.info("Attempting to download s3://{0}/{1}".format(bucket, filename))

    try:
        # Make sure this file exists, don't ACTUALLY download
        range_header = get_range_header_val()
        if not range_header:
            client.get_object(Bucket=bucket, Key=filename)
            redirheaders = {}
        else:
            client.get_object(Bucket=bucket, Key=filename, Range=range_header)
            redirheaders = {'Range': range_header}

        # Generate URL
        presigned_url = get_presigned_url(creds, bucket, filename,
                                          bucket_region, 24 * 3600, user_id)
        s3_host = urlparse(presigned_url).netloc
        log.debug("Presigned URL host was {0}".format(s3_host))

        log.info("Using REDIRECT because no PROXY in egresslambda")
        return make_redriect(presigned_url, redirheaders, 303)

    except ClientError as e:
        log.warning("Could not download s3://{0}/{1}: {2}".format(
            bucket, filename, e))

        # Watch for bad range request:
        if e.response['ResponseMetadata']['HTTPStatusCode'] == 416:
            return Response(body='Invalid Range', status_code=416, headers={})

        template_vars = {
            'contentstring': 'Could not find requested data.',
            'title': 'Data Not Available'
        }
        headers = {}
        return make_html_response(template_vars, headers, 404, 'error.html')
Exemple #7
0
def try_download_from_bucket(bucket, filename, user_profile):

    # Attempt to pull userid from profile
    user_id = None
    if isinstance(user_profile, dict):
        if 'urs-user-id' in user_profile:
            user_id = user_profile['urs-user-id']
        elif 'uid' in user_profile:
            user_id = user_profile['uid']
    log.info("User Id for download is {0}".format(user_id))

    is_in_region = check_in_region_request(
        app.current_request.context['identity']['sourceIp'])
    creds = get_role_creds(user_id, is_in_region)
    session = get_role_session(creds=creds, user_id=user_id)

    params = {}

    BCCONFIG = {
        "user_agent": "RAIN Egress App for userid={0}".format(user_id),
        "s3": {
            "addressing_style": "path"
        },
        "connect_timeout": 600,
        "read_timeout": 600,
        "retries": {
            "max_attempts": 10
        }
    }

    if os.getenv('S3_SIGNATURE_VERSION'):
        BCCONFIG['signature_version'] = os.getenv('S3_SIGNATURE_VERSION')

    # Figure out bucket region
    try:
        bucket_region = session.client(
            's3',
            **params).get_bucket_location(Bucket=bucket)['LocationConstraint']
        bucket_region = 'us-east-1' if not bucket_region else bucket_region
        log.debug("bucket {0} is in region {1}".format(bucket, bucket_region))
    except ClientError as e:
        # We hit here if the download role cannot access a bucket, or if it doesn't exist
        log.error("Coud not access download bucket {0}: {1}".format(bucket, e))

        template_vars = {
            'contentstring': 'There was a problem accessing download data.',
            'title': 'Data Not Available'
        }
        headers = {}
        return make_html_response(template_vars, headers, 500, 'error.html')

    log.debug('this region: {}'.format(
        os.getenv('AWS_DEFAULT_REGION', 'env var doesnt exist')))
    if bucket_region != os.getenv('AWS_DEFAULT_REGION'):
        log.warning(
            "bucket {0} is in region {1}, we are in region {2}! This is double egress in Proxy mode!"
            .format(bucket, bucket_region, os.getenv('AWS_DEFAULT_REGION')))

    # now that we know where the bucket is, connect in THAT region
    params['config'] = bc_Config(**BCCONFIG)
    client = session.client('s3', bucket_region, **params)

    log.info("Attempting to download s3://{0}/{1}".format(bucket, filename))

    try:
        # Make sure this file exists, don't ACTUALLY download
        range_header = get_range_header_val()
        if not range_header:
            client.get_object(Bucket=bucket, Key=filename)
            redirheaders = {}
        else:
            client.get_object(Bucket=bucket, Key=filename, Range=range_header)
            redirheaders = {'Range': range_header}

        # Generate URL
        presigned_url = get_presigned_url(creds, bucket, filename,
                                          bucket_region, 24 * 3600, user_id)
        s3_host = urlparse(presigned_url).netloc
        log.debug("Presigned URL host was {0}".format(s3_host))

        log.info("Using REDIRECT because no PROXY in egresslambda")
        return make_redriect(presigned_url, redirheaders, 303)

    except ClientError as e:
        log.warning("Could not download s3://{0}/{1}: {2}".format(
            bucket, filename, e))

        # Watch for bad range request:
        if e.response['ResponseMetadata']['HTTPStatusCode'] == 416:
            return Response(body='Invalid Range', status_code=416, headers={})

        template_vars = {
            'contentstring': 'Could not find requested data.',
            'title': 'Data Not Available'
        }
        headers = {}
        return make_html_response(template_vars, headers, 404, 'error.html')