def test_no_base_url_subpath(self): endpoint = os.getenv("MINIO_STORAGE_ENDPOINT", "minio:9000") assert endpoint != "" media_storage = MinioMediaStorage() name = "23/23/aaa/bbb/22" url = media_storage.url(name) self.assertEqual(url, f"http://{endpoint}/foo/23/23/aaa/bbb/22")
def test_media_policy_rw(self): ms = MinioMediaStorage() self.maxDiff = 50000 self.assertPolicyEqual( Policy.read_write.bucket(ms.bucket_name, json_encode=False), json.loads(ms.client.get_bucket_policy(ms.bucket_name)), )
def test_media_storage_ignore_bucket_check(self): try: MinioMediaStorage() except OSError: self.assertTrue(False) else: pass
def setUp(self): self.media_storage = MinioMediaStorage() self.static_storage = MinioStaticStorage() self.new_file = self.media_storage.save("test-file", ContentFile(b"yep")) self.second_file = self.media_storage.save("test-file", ContentFile(b"nope"))
def test_presigned_base_url_subpath(self): # The url generated here probably doenst work in a real situation media_storage = MinioMediaStorage() name = "1/555/666/777" url = media_storage.url(name) self.assertIn("X-Amz-Signature", url) self.assertIn("http://example11.com/foo", url) self.assertIn(name, url)
def setUp(self): self.media_storage = MinioMediaStorage() self.static_storage = MinioStaticStorage() self.new_file = self.media_storage.save("test-file", ContentFile(b"yep")) self.media_storage.client.set_bucket_policy( self.media_storage.bucket_name, '', Policy.READ_WRITE) self.static_storage.client.set_bucket_policy( self.static_storage.bucket_name, '', Policy.READ_WRITE)
def test_media_policy_write(self): ms = MinioMediaStorage() self.maxDiff = 50000 self.assertPolicyEqual( Policy.write.bucket(ms.bucket_name, json_encode=False), json.loads(ms.client.get_bucket_policy(ms.bucket_name)), ) fn = ms.save("somefile", ContentFile(b"test")) self.assertEqual(ms.open(fn).read(), b"test") self.assertEqual(requests.get(ms.url(fn)).status_code, 403) self.assertEqual( requests.get(f"{ms.client._endpoint_url}/{ms.bucket_name}").status_code, 403 )
def test_media_policy_auto_true(self): ms = MinioMediaStorage() self.maxDiff = 50000 self.assertPolicyEqual( Policy.get.bucket(ms.bucket_name, json_encode=False), json.loads(ms.client.get_bucket_policy(ms.bucket_name)), ) fn = ms.save("somefile", ContentFile(b"test")) self.assertEqual(ms.open(fn).read(), b"test") url = ms.url(fn) self.assertEqual(requests.get(url).status_code, 200) self.assertEqual( requests.get("{}/{}".format(ms.client._endpoint_url, ms.bucket_name)).status_code, 403 )
def test_public_url_generation(self): media_storage = MinioMediaStorage() media_test_file_name = media_storage.save("weird & ÜRΛ", ContentFile(b"irrelevant")) url = media_storage.url(media_test_file_name) res = requests.get(url) self.assertEqual(res.content, b"irrelevant") static_storage = MinioStaticStorage() static_test_file_name = static_storage.save("weird & ÜRΛ", ContentFile(b"irrelevant")) url = static_storage.url(static_test_file_name) res = requests.get(url) self.assertEqual(res.content, b"irrelevant")
class ZipDumpFile(models.Model): dump_file = models.ForeignKey(DumpFile, on_delete=models.CASCADE) file = models.FileField(upload_to=zipfile_upload_to, storage=MinioMediaStorage()) @classmethod def create_from_dump_file(cls, dump_file: DumpFile, dump_file_path: str): with TempZipDumpFile(dump_file) as zip_file: zip_file.write(dump_file_path, arcname=f'{dump_file.file_name}.{dump_file.file_type}') zip_file.close() with open(zip_file.filename, 'rb') as f: return cls.objects.create(file=File(f), dump_file=dump_file) def delete(self, using=None, keep_parents=False): if self.file: minio = create_minio_client_from_settings() minio.remove_object(settings.MINIO_STORAGE_MEDIA_BUCKET_NAME, self.file.name) super(ZipDumpFile, self).delete(using, keep_parents)
def test_no_base_url(self): endpoint = os.getenv("MINIO_STORAGE_ENDPOINT", "minio:9000") assert endpoint != "" media_storage = MinioMediaStorage() url = media_storage.url("22") self.assertEqual(url, "http://{}/foo/22".format(endpoint))
def test_file_upload_fail_incorrect_keys(self): with self.assertRaises(InvalidAccessKeyId): MinioMediaStorage()
def test_auto_create_no_policy(self): ms = MinioMediaStorage() with self.assertRaises(minio.error.NoSuchBucketPolicy): ms.client.get_bucket_policy(ms.bucket_name)
def test_media_storage_ignore_bucket_check(self): MinioMediaStorage()
def test_media_storage_cannot_be_initialized_without_bucket(self): with self.assertRaises(OSError): MinioMediaStorage()
class DumpFile(models.Model): file = models.FileField(upload_to=dumpfile_upload_to, storage=MinioMediaStorage()) FILENAME_FULL = 'series-tiempo' FILENAME_VALUES = 'series-tiempo-valores' FILENAME_METADATA = 'series-tiempo-metadatos' FILENAME_SOURCES = 'series-tiempo-fuentes' FILENAME_CHOICES = ( (FILENAME_FULL, 'Series de tiempo (valores + metadatos)'), (FILENAME_VALUES, 'Series de tiempo (valores)'), (FILENAME_METADATA, 'Series de tiempo (metadatos)'), (FILENAME_SOURCES, 'Series de tiempo (fuentes)'), ) file_name = models.CharField(max_length=64, choices=FILENAME_CHOICES) TYPE_CSV = 'csv' TYPE_XLSX = 'xlsx' TYPE_ZIP = 'zip' TYPE_SQL = 'sqlite' TYPE_DTA = 'dta' TYPE_CHOICES = ( (TYPE_CSV, 'CSV'), (TYPE_XLSX, 'XLSX'), (TYPE_ZIP, 'ZIP'), (TYPE_SQL, 'SQL'), (TYPE_DTA, 'DTA'), ) ZIP_FILES = ( (FILENAME_FULL, TYPE_CSV), (FILENAME_VALUES, TYPE_CSV), (FILENAME_FULL, TYPE_SQL), (FILENAME_VALUES, TYPE_DTA), ) file_type = models.CharField(max_length=12, choices=TYPE_CHOICES, default=TYPE_CSV) task = models.ForeignKey(GenerateDumpTask, on_delete=models.CASCADE) node = models.ForeignKey(Node, on_delete=models.PROTECT, blank=True, null=True) def delete(self, using=None, keep_parents=False): if self.file: minio = create_minio_client_from_settings() minio.remove_object(settings.MINIO_STORAGE_MEDIA_BUCKET_NAME, self.file.name) super(DumpFile, self).delete(using, keep_parents) def get_path(self): directory = f'{self.node.catalog_id}/' if self.node else '' return f'{constants.DUMP_DIR}/{directory}{self.get_file_name()}' def get_file_name(self): return f'{self.file_name}.{self.file_type}' def __str__(self): return f'{self.get_file_name()} ({self.node or "global"})' @classmethod def get_from_path(cls, filename: str, node: str = None) -> 'DumpFile': """Devuelve la última instancia de clase que corresponda al archivo con filename de formato tipo <catalog>/<file_name>.<extension>. Lanza DoesNotExist si no se encuentra un dump correspondiente. """ try: node = Node.objects.get(catalog_id=node) if node else None except Node.DoesNotExist: raise cls.DoesNotExist try: name, extension = filename.split('.') except ValueError: raise cls.DoesNotExist if extension == cls.TYPE_ZIP: split = name.rfind('-') orig_extension = name[split + 1:] name = name[:split] dump = cls.objects.filter(file_name=name, file_type=orig_extension, node=node).last() try: return ZipDumpFile.objects.get(dump_file=dump) except ZipDumpFile.DoesNotExist: raise cls.DoesNotExist dump = cls.objects.filter(file_name=name, file_type=extension, node=node).last() if dump is None: raise cls.DoesNotExist return dump @classmethod def get_last_of_type(cls, file_type: str, node: str = None) -> list: """Devuelve el último dump generado del formato file_type especificado. Si se pasa un parámetro node, devuelve los últimos dumps para ese node. Si no, se devuelve los últimos dumps globales. """ dumps_qs = cls.objects.filter(node__catalog_id=node) dumps = [] for dump_name, _ in cls.FILENAME_CHOICES: dump_file = dumps_qs.filter(file_name=dump_name, file_type=file_type).last() if dump_file is not None: dumps.append(dump_file) return dumps
def test_presigned_no_base_url(self): endpoint = os.getenv("MINIO_STORAGE_ENDPOINT", "minio:9000") assert endpoint != "" media_storage = MinioMediaStorage() url = media_storage.url("22") self.assertRegex(url, rf"^http://{endpoint}/foo/22\?")
def test_no_base_url_subpath(self): media_storage = MinioMediaStorage() name = "23/23/aaa/bbb/22" url = media_storage.url(name) self.assertEqual(url, 'http://localhost:9000/foo/23/23/aaa/bbb/22')
def test_quote_base_url(self): media_storage = MinioMediaStorage() url = media_storage.url(self.NAME) self.assertEqual(url, "{}/{}".format(self.BASE, self.ENCODED))
def test_presigned_base_url_subpath(self): media_storage = MinioMediaStorage() url = media_storage.url("1/555/666/777") self.assertIn("X-Amz-Signature", url) self.assertRegex(url, r"^http://example11.com/foo/1/555/666/777\?")
def test_base_url_subpath(self): media_storage = MinioMediaStorage() url = media_storage.url("1/2/3/4") self.assertEqual(url, "https://example23.com/foo/1/2/3/4")
def test_quote_base_url(self): media_storage = MinioMediaStorage() url = media_storage.url(self.NAME) self.assertEqual(url, f"{self.BASE}/{self.ENCODED}")
def test_file_upload_fail_incorrect_keys(self): with self.assertRaises(ResponseError): MinioMediaStorage()
def test_presigned_base_url(self): # The url generated here probably doenst work in a real situation media_storage = MinioMediaStorage() url = media_storage.url("1") self.assertIn("X-Amz-Signature", url) self.assertIn("http://example11.com/foo", url)
def test_base_url(self): media_storage = MinioMediaStorage() url = media_storage.url("1") self.assertEqual(url, 'https://example23.com/foo/1')
def test_quote_url(self): media_storage = MinioMediaStorage() url = media_storage.url(self.NAME) self.assertTrue(url.endswith(self.ENCODED)) self.assertTrue(len(url) > len(self.ENCODED))
def setUp(self): self.media_storage = MinioMediaStorage() self.static_storage = MinioStaticStorage() self.new_file = self.media_storage.save("a new & original file", ContentFile(b"yep"))
def test_quote_base_url_presigned(self): media_storage = MinioMediaStorage() url = media_storage.url(self.NAME) prefix = "{}/{}".format(self.BASE, self.ENCODED) self.assertTrue(url.startswith(prefix)) self.assertTrue(len(url) > len(prefix))
def test_no_base_url(self): media_storage = MinioMediaStorage() url = media_storage.url("22") self.assertEqual(url, 'http://localhost:9000/foo/22')