-
Notifications
You must be signed in to change notification settings - Fork 32
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
56ff1ff
commit e99b5ef
Showing
24 changed files
with
1,977 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
build/ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,125 @@ | ||
from __future__ import absolute_import | ||
|
||
import os | ||
import sys | ||
|
||
CF_SHARED_CDN = "d3jpl91pxevbkh.cloudfront.net" | ||
OLD_AKAMAI_SHARED_CDN = "cloudinary-a.akamaihd.net" | ||
AKAMAI_SHARED_CDN = "res.cloudinary.com" | ||
SHARED_CDN = AKAMAI_SHARED_CDN | ||
CL_BLANK = "" | ||
|
||
VERSION = "1.0.21" | ||
USER_AGENT = "cld-python-" + VERSION | ||
|
||
from cloudinary import utils | ||
from cloudinary.compat import urlparse, parse_qs | ||
|
||
def import_django_settings(): | ||
try: | ||
import django.conf | ||
from django.core.exceptions import ImproperlyConfigured | ||
try: | ||
if 'CLOUDINARY' in dir(django.conf.settings): | ||
return django.conf.settings.CLOUDINARY | ||
else: | ||
return None | ||
except ImproperlyConfigured: | ||
return None | ||
except ImportError: | ||
return None | ||
|
||
class Config(object): | ||
def __init__(self): | ||
django_settings = import_django_settings() | ||
if django_settings: | ||
self.update(**django_settings) | ||
elif os.environ.get("CLOUDINARY_CLOUD_NAME"): | ||
self.update( | ||
cloud_name = os.environ.get("CLOUDINARY_CLOUD_NAME"), | ||
api_key = os.environ.get("CLOUDINARY_API_KEY"), | ||
api_secret = os.environ.get("CLOUDINARY_API_SECRET"), | ||
secure_distribution = os.environ.get("CLOUDINARY_SECURE_DISTRIBUTION"), | ||
private_cdn = os.environ.get("CLOUDINARY_PRIVATE_CDN") == 'true' | ||
) | ||
elif os.environ.get("CLOUDINARY_URL"): | ||
uri = urlparse(os.environ.get("CLOUDINARY_URL").replace("cloudinary://", "http://")) | ||
for k, v in parse_qs(uri.query).items(): | ||
self.__dict__[k] = v[0] | ||
self.update( | ||
cloud_name = uri.hostname, | ||
api_key = uri.username, | ||
api_secret = uri.password, | ||
private_cdn = uri.path != '' | ||
) | ||
if uri.path != '': | ||
self.update(secure_distribution = uri.path[1:]) | ||
def __getattr__(self, i): | ||
if i in self.__dict__: | ||
return self.__dict__[i] | ||
else: | ||
return None | ||
|
||
def update(self, **keywords): | ||
for k, v in keywords.items(): | ||
self.__dict__[k] = v | ||
|
||
_config = Config() | ||
|
||
def config(**keywords): | ||
global _config | ||
_config.update(**keywords) | ||
return _config | ||
|
||
def reset_config(): | ||
global _config | ||
_config = Config() | ||
|
||
class CloudinaryImage(object): | ||
def __init__(self, public_id = None, format = None, version = None, | ||
signature = None, url_options = {}, metadata = None, type = None): | ||
|
||
self.metadata = metadata | ||
metadata = metadata or {} | ||
self.public_id = public_id or metadata.get('public_id') | ||
self.format = format or metadata.get('format') | ||
self.version = version or metadata.get('version') | ||
self.signature = signature or metadata.get('signature') | ||
self.type = type or metadata.get('type') or "upload" | ||
self.url_options = url_options | ||
|
||
def __unicode__(self): | ||
return self.public_id | ||
|
||
def validate(self): | ||
expected = utils.api_sign_request({"public_id": self.public_id, "version": self.version}, config().api_secret) | ||
return self.signature == expected | ||
|
||
@property | ||
def url(self): | ||
return self.build_url(**self.url_options) | ||
|
||
def __build_url(self, **options): | ||
combined_options = dict(format = self.format, version = self.version, type = self.type) | ||
combined_options.update(options) | ||
return utils.cloudinary_url(self.public_id, **combined_options) | ||
|
||
def build_url(self, **options): | ||
return self.__build_url(**options)[0] | ||
|
||
def image(self, **options): | ||
src, attrs = self.__build_url(**options) | ||
responsive = attrs.pop("responsive", False) | ||
hidpi = attrs.pop("hidpi", False) | ||
if responsive or hidpi: | ||
attrs["data-src"] = src | ||
classes = "cld-responsive" if responsive else "cld-hidpi" | ||
if "class" in attrs: classes += " " + attrs["class"] | ||
attrs["class"] = classes | ||
src = attrs.pop("responsive_placeholder", config().responsive_placeholder) | ||
if src == "blank": src = CL_BLANK | ||
|
||
attrs = sorted(attrs.items()) | ||
if src: attrs.insert(0, ("src", src)) | ||
|
||
return u"<img {0}/>".format(' '.join([u"{0}='{1}'".format(key, value) for key, value in attrs if value])) |
Binary file not shown.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,243 @@ | ||
# Copyright Cloudinary | ||
import json | ||
import base64 | ||
import sys | ||
import email.utils | ||
import socket | ||
import cloudinary | ||
from cloudinary.compat import urllib2, urlencode, to_string, to_bytes, PY3, HTTPError | ||
from cloudinary import utils | ||
|
||
class Error(Exception): pass | ||
class NotFound(Error): pass | ||
class NotAllowed(Error): pass | ||
class AlreadyExists(Error): pass | ||
class RateLimited(Error): pass | ||
class BadRequest(Error): pass | ||
class GeneralError(Error): pass | ||
class AuthorizationRequired(Error): pass | ||
|
||
EXCEPTION_CODES = { | ||
400: BadRequest, | ||
401: AuthorizationRequired, | ||
403: NotAllowed, | ||
404: NotFound, | ||
409: AlreadyExists, | ||
420: RateLimited | ||
} | ||
|
||
class Response(dict): | ||
def __init__(self, result, response): | ||
self.update(result) | ||
self.rate_limit_allowed = int(response.headers["x-featureratelimit-limit"]) | ||
self.rate_limit_reset_at = email.utils.parsedate(response.headers["x-featureratelimit-reset"]) | ||
self.rate_limit_remaining = int(response.headers["x-featureratelimit-remaining"]) | ||
|
||
def ping(**options): | ||
return call_api("get", ["ping"], {}, **options) | ||
|
||
def usage(**options): | ||
return call_api("get", ["usage"], {}, **options) | ||
|
||
def resource_types(**options): | ||
return call_api("get", ["resources"], {}, **options) | ||
|
||
def resources(**options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", None) | ||
uri = ["resources", resource_type] | ||
if type: uri.append(type) | ||
return call_api("get", uri, only(options, "next_cursor", "max_results", "prefix", "tags", "context", "moderations", "direction", "start_at"), **options) | ||
|
||
def resources_by_tag(tag, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
uri = ["resources", resource_type, "tags", tag] | ||
return call_api("get", uri, only(options, "next_cursor", "max_results", "tags", "context", "moderations", "direction"), **options) | ||
|
||
def resources_by_moderation(kind, status, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
uri = ["resources", resource_type, "moderations", kind, status] | ||
return call_api("get", uri, only(options, "next_cursor", "max_results", "tags", "context", "moderations", "direction"), **options) | ||
|
||
def resources_by_ids(public_ids, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type] | ||
params = [("public_ids[]", public_id) for public_id in public_ids] | ||
optional = list(only(options, "tags", "moderations", "context").items()) | ||
return call_api("get", uri, params + optional, **options) | ||
|
||
def resource(public_id, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type, public_id] | ||
return call_api("get", uri, only(options, "exif", "faces", "colors", "image_metadata", "pages", "phash", "coordinates", "max_results"), **options) | ||
|
||
def update(public_id, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type, public_id] | ||
upload_options = only(options, "moderation_status", "raw_convert", "ocr", "categorization", "detection", "similarity_search", "background_removal") | ||
if "tags" in options: upload_options["tags"] = ",".join(utils.build_array(options["tags"])) | ||
if "face_coordinates" in options: upload_options["face_coordinates"] = utils.encode_double_array(options.get("face_coordinates")) | ||
if "custom_coordinates" in options: upload_options["custom_coordinates"] = utils.encode_double_array(options.get("custom_coordinates")) | ||
if "context" in options: upload_options["context"] = utils.encode_dict(options.get("context")) | ||
if "auto_tagging" in options: upload_options["auto_tagging"] = float(options.get("auto_tagging")) | ||
return call_api("post", uri, upload_options, **options) | ||
|
||
def delete_resources(public_ids, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type] | ||
params = [("public_ids[]", public_id) for public_id in public_ids] | ||
optional = list(only(options, "keep_original", "next_cursor", "invalidate").items()) | ||
return call_api("delete", uri, params + optional, **options) | ||
|
||
def delete_resources_by_prefix(prefix, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type] | ||
return call_api("delete", uri, dict(only(options, "keep_original", "next_cursor", "invalidate"), prefix=prefix), **options) | ||
|
||
def delete_all_resources(**options): | ||
resource_type = options.pop("resource_type", "image") | ||
type = options.pop("type", "upload") | ||
uri = ["resources", resource_type, type] | ||
optional = list(only(options, "keep_original", "next_cursor").items()) | ||
return call_api("delete", uri, dict(only(options, "keep_original", "next_cursor", "invalidate"), all=True), **options) | ||
|
||
def delete_resources_by_tag(tag, **options): | ||
resource_type = options.pop("resource_type", "image") | ||
uri = ["resources", resource_type, "tags", tag] | ||
return call_api("delete", uri, only(options, "keep_original", "next_cursor", "invalidate"), **options) | ||
|
||
def delete_derived_resources(derived_resource_ids, **options): | ||
uri = ["derived_resources"] | ||
params = [("derived_resource_ids[]", derived_resource_id) for derived_resource_id in derived_resource_ids] | ||
|
||
return call_api("delete", uri, params, **options) | ||
|
||
def tags(**options): | ||
resource_type = options.pop("resource_type", "image") | ||
uri = ["tags", resource_type] | ||
return call_api("get", uri, only(options, "next_cursor", "max_results", "prefix"), **options) | ||
|
||
def transformations(**options): | ||
uri = ["transformations"] | ||
return call_api("get", uri, only(options, "next_cursor", "max_results"), **options) | ||
|
||
def transformation(transformation, **options): | ||
uri = ["transformations", transformation_string(transformation)] | ||
return call_api("get", uri, only(options, "max_results"), **options) | ||
|
||
def delete_transformation(transformation, **options): | ||
uri = ["transformations", transformation_string(transformation)] | ||
return call_api("delete", uri, {}, **options) | ||
|
||
# updates - currently only supported update is the "allowed_for_strict" boolean flag and unsafe_update | ||
def update_transformation(transformation, **options): | ||
uri = ["transformations", transformation_string(transformation)] | ||
updates = only(options, "allowed_for_strict") | ||
if "unsafe_update" in options: | ||
updates["unsafe_update"] = transformation_string(options.get("unsafe_update")) | ||
if len(updates) == 0: raise Exception("No updates given") | ||
|
||
return call_api("put", uri, updates, **options) | ||
|
||
def create_transformation(name, definition, **options): | ||
uri = ["transformations", name] | ||
return call_api("post", uri, {"transformation": transformation_string(definition)}, **options) | ||
|
||
def upload_presets(**options): | ||
uri = ["upload_presets"] | ||
return call_api("get", uri, only(options, "next_cursor", "max_results"), **options) | ||
|
||
def upload_preset(name, **options): | ||
uri = ["upload_presets", name] | ||
return call_api("get", uri, only(options, "max_results"), **options) | ||
|
||
def delete_upload_preset(name, **options): | ||
uri = ["upload_presets", name] | ||
return call_api("delete", uri, {}, **options) | ||
|
||
def update_upload_preset(name, **options): | ||
uri = ["upload_presets", name] | ||
params = utils.build_upload_params(**options) | ||
params = utils.cleanup_params(params) | ||
params.update(only(options, "unsigned", "disallow_public_id")) | ||
return call_api("put", uri, params, **options) | ||
|
||
def create_upload_preset(**options): | ||
uri = ["upload_presets"] | ||
params = utils.build_upload_params(**options) | ||
params = utils.cleanup_params(params) | ||
params.update(only(options, "unsigned", "disallow_public_id", "name")) | ||
return call_api("post", uri, params, **options) | ||
|
||
def root_folders(**options): | ||
return call_api("get", ["folders"], {}, **options) | ||
|
||
def subfolders(of_folder_path, **options): | ||
return call_api("get", ["folders", of_folder_path], {}, **options) | ||
|
||
def call_api(method, uri, params, **options): | ||
prefix = options.pop("upload_prefix", cloudinary.config().upload_prefix) or "https://api.cloudinary.com" | ||
cloud_name = options.pop("cloud_name", cloudinary.config().cloud_name) | ||
if not cloud_name: raise Exception("Must supply cloud_name") | ||
api_key = options.pop("api_key", cloudinary.config().api_key) | ||
if not api_key: raise Exception("Must supply api_key") | ||
api_secret = options.pop("api_secret", cloudinary.config().api_secret) | ||
if not cloud_name: raise Exception("Must supply api_secret") | ||
|
||
data = to_bytes(urlencode(params)) | ||
api_url = "/".join([prefix, "v1_1", cloud_name] + uri) | ||
request = urllib2.Request(api_url, data) | ||
# Add authentication | ||
byte_value = to_bytes('%s:%s' % (api_key, api_secret)) | ||
encoded_value = base64.encodebytes(byte_value) if PY3 else base64.encodestring(byte_value) | ||
base64string = to_string(encoded_value).replace('\n', '') | ||
request.add_header("Authorization", "Basic %s" % base64string) | ||
request.add_header("User-Agent", cloudinary.USER_AGENT) | ||
request.get_method = lambda: method.upper() | ||
|
||
kw = {} | ||
if 'timeout' in options: | ||
kw['timeout'] = options['timeout'] | ||
try: | ||
response = urllib2.urlopen(request, **kw) | ||
body = response.read() | ||
except HTTPError: | ||
e = sys.exc_info()[1] | ||
exception_class = EXCEPTION_CODES.get(e.code) | ||
if exception_class: | ||
response = e | ||
body = response.read() | ||
else: | ||
raise GeneralError("Server returned unexpected status code - %d - %s" % (e.code, e.read())) | ||
except socket.error: | ||
e = sys.exc_info()[1] | ||
raise GeneralError("Socket Error: %s" % (str(e))) | ||
|
||
try: | ||
body = to_string(body) | ||
result = json.loads(body) | ||
except Exception: | ||
# Error is parsing json | ||
e = sys.exc_info()[1] | ||
raise GeneralError("Error parsing server response (%d) - %s. Got - %s" % (response.code, body, e)) | ||
|
||
if "error" in result: | ||
exception_class = exception_class or Exception | ||
raise exception_class(result["error"]["message"]) | ||
|
||
return Response(result, response) | ||
|
||
def only(hash, *keys): | ||
result = {} | ||
for key in keys: | ||
if key in hash: result[key] = hash[key] | ||
return result | ||
|
||
def transformation_string(transformation): | ||
return transformation if isinstance(transformation, str) else cloudinary.utils.generate_transformation_string(**transformation)[0] | ||
|
Binary file not shown.
Oops, something went wrong.