Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (kevin13@chol.net)
홈페이지
첨부 파일

sysnet 첨부 파일을 Azure Storage 에 마이그레이션

보통, 웬만한 웹 사이트는 게시판을 가지고 있기 마련이고, 따라서 첨부 파일이 있을 것입니다. 업로드 된 파일들을 DB에 저장하는 경우도 있겠지만 대개의 경우 파일 시스템에 저장할 텐데요.

sysnet 웹 사이트도 Azure 클라우드 서비스로 마이그레이션 할 때 '웹 애플리케이션 코드' 이외에 다음과 같이 2가지의 데이터 이전이 필요했습니다.

  • Database
  • 첨부 파일

Database 는 지난 번 글에서 이미 설명했고,

sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
; http://www.sysnet.pe.kr/2/0/1303

이제 "첨부되었던 파일"들을 Azure로 올려야 하는데요. 이러한 파일 데이터를 위해서 Azure에서는 (아마존에서는 S3서비스에 해당하는) "Storage 서비스"를 제공해 주고 있는데, 실제로 Azure에 Storage 서비스를 하나 생성하면 다음과 같이 3개의 서비스 종류가 한꺼번에 제공됩니다.

blob_storage_explorer_1.png

그리고 이를 이용하려면 하단의 "MANAGE KEYS"버튼을 눌러서 아래와 같이 계정 키를 알아내야 합니다.

blob_storage_explorer_2.png

3가지 서비스 종류에서 '첨부 파일'을 처리하기에 적합한 유형이 바로 "Blobs" 저장소입니다. (나중에 설명하겠지만, 이 곳에 파일을 저장하면 웹 브라우저에 경로를 직접 입력하여 접근하는 것도 가능합니다.)

Azure에 올라갈 웹 사이트에 첨부 파일 기능이 있다면 다음과 같이 크게 3가지 단계를 반영해 주어야 합니다.

  1. 게시판에서 업로드되는 파일을 Azure Blob Storage에 저장하도록 변경
  2. 게시판에서 다운로드되는 파일을 Azure Blob Storage로부터 가져오도록 변경
  3. 기존 업로드된 파일들을 Azure Blob Storage에 미리 적재

이 중에서 1번과 2번 단계는 (나중에 알아보겠지만) 웹 애플리케이션의 코드 변경을 가져오는 반면, 3번 단계는 별도 조치를 취해주어야 하는데, 이를 위해 다시 2가지 정도의 방법이 있습니다.

첫번째는 직접 코딩을 하는 방법이 있는데요. 이건 아래의 글을 참고해서 만들어 주시면 됩니다.

블랍(Blob) 저장소 서비스 사용하기 
; http://taeyo.net/Columns/View.aspx?SEQ=417&PSEQ=33

두번째는, "Azure Storage Explorer"를 이용하는 방법이 있습니다.

Azure Storage Explorer (소스 코드 제공됨)
; http://azurestorageexplorer.codeplex.com/

설치 후, 실행해서 "Add Account" 버튼을 눌러 계정 정보를 입력해야 하는데,

blob_storage_explorer_3.png

"Storage account name"은 Azure 관리화면에서 "STORAGE" 서비스로 생성했던 이름을 넣고, "Storage account key"는 "MANAGE KEYS" 버튼으로 알아냈던 "PRIMARY ACCESS KEY" 또는 "SECONDARY ACCESS KEY"를 입력해 주면 됩니다.

제 경우에는, 이미 웹 사이트를 하나 Cloud Services 에 배포했기 때문에 다음과 같이 "wad-control-container" 컨테이너가 있는 것을 볼 수 있습니다.

blob_storage_explorer_4.png

"첨부되었던 파일"들을 보관하기 위해 새로운 컨테이너를 만들어야 하는데, "Container" / "New" 버튼을 눌러주면 됩니다.

blob_storage_explorer_5.png

Container를 생성했으면 이제 원하는 파일을 "Upload"버튼을 이용하여 할 수 있습니다. (다중 파일 선택이 가능합니다.)

blob_storage_explorer_6.png

업로드가 완료되면 다음과 같이 목록에 나타나고,

blob_storage_explorer_7.png

이전에 "attachments" 컨테이너를 "Public Container"로 지정했기 때문에 웹 브라우저를 이용하여 직접 다운로드하는 것이 가능합니다.

http://[STROAGE 서비스 이름].blob.core.windows.net/[컨테이너 이름]/[파일명]

정확한 URL은 아래와 같이 "View" 버튼을 이용하여 (또는 해당 파일을 더블 클릭하거나) 나오는 속성 창에서 "AbsoluteUri" 속성값을 이용해서도 구할 수 있습니다.

blob_storage_explorer_8.png

비록 "Azure Storage Explorer"가 편리하긴 하지만, 단점이 하나 있는데요. 파일이 많은 경우 - 가령 2,000 개 정도만 있어도 해당 컨테이너를 선택하면 파일 목록을 보는데 30초 넘게 기다려야 합니다. 개인적으로 이 부분이 다소 불편했는데, 가만 보니 "Azure Storage Explorer" 는 해당 파일에 대해서 '속성'값들을 전부 조회하기 때문에 느려지는 것 같았습니다. 그래서, 저 나름대로 다시 간단한 프로그램을 만들었는데요.

캡쳐 화면은 아래와 같고, 사용법은 직관적으로 알 수 있으니 생략하지만... 코드는 이 글에 첨부해 두었으니 참고하실 분은 다운로드 받으면 되겠습니다. ^^

blob_storage_explorer_9.png




그 외에, 게시판에서 첨부 파일을 업로드/다운로드 하는 경우에 대해 코드를 변경해 주어야 하는데요. 이 부분은 블랍(Blob) 저장소 서비스 사용하기 글만 잘 읽어보면 구현하는 데 별 어려움이 없습니다. 예를 들어, 다운로드는 다음과 같이 Response.OutputStream에 직접 쓰기를 해주거나,

Response.Clear();
Response.ContentType = "application/octet-stream";
string encodedFilename = HttpContext.Current.Server.UrlEncode(orgFileName);
Response.AddHeader("Content-Disposition", "attachment; filename=" + encodedFilename);

CloudStorageAccount storageAccount = 
    CloudStorageAccount.Parse(ConfigurationManager.AppSettings["AzureStorage"]);

CloudBlobClient blobClient = storageAccount.CreateCloudBlobClient();
CloudBlobContainer blobContainer = blobClient.GetContainerReference("attachments");

CloudBlob blobItem = blobContainer.GetBlobReference(orgFileName);

blobItem.DownloadToStream(Response.OutputStream);
Response.End();

또는 위에서 설명한데로 "http://[STROAGE 서비스 이름].blob.core.windows.net/[컨테이너 이름]/[파일명]" 경로를 직접 웹 브라우저 측에 A 링크로 노출시켜도 됩니다.

마지막으로, 업로드의 경우에는 PostedFile 개체의 InputStream을 직접 CloudBlob.UploadFromStream에 전달해 주면 됩니다.

CloudStorageAccount storageAccount = 
    CloudStorageAccount.Parse(ConfigurationManager.AppSettings["AzureStorage"]);

CloudBlobClient blobClient = storageAccount.CreateCloudBlobClient();
CloudBlobContainer blobContainer = blobClient.GetContainerReference("attachments");

CloudBlob blobItem = blobContainer.GetBlobReference("...[파일명]...");
blobItem.UploadFromStream(upFile.PostedFile.InputStream);

어떠세요? 생각했던 것보다 더욱 간단하지요. 물론, 복잡한 응용 프로그램이라면 코드를 좀더 변경할 부분들이 많겠지만 sysnet 웹 사이트의 경우에는 위와 같이 업/다운로드 처리를 하는 코드 작업이 주요했고 이 외에는 거의 변경된 것이 없습니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]



[최초 등록일: ]
[최종 수정일: 6/30/2012 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:kevin13@chol.com

비밀번호

댓글 쓴 사람
 



2014-03-01 03시22분
[runner] 블롭 파일을 웹앱을 경유해서 다운로드 시키려 할 때 memorystream에 download 했다가 response로 보내는


 Stream stream = new MemoryStream();
    CloudBlockBlob blockBlob = blobContainer.GetBlockBlobReference(blobName);
if (blockBlob.Exists())
    {
        blockBlob.DownloadToStream(stream);
        long streamlen = stream.Length; <-- This shows 0 bytes
        stream.Position = 0;
    }


이런 예제들이 많이 검색되던데 , 더 직접 response Stream 에 보내는 방법이 있었군요! 좋은 정보 알아 갑니다.
[손님]
2015-11-19 06시40분
본문에서 소개한 Azure Storage Explorer보다 더 나은 버전이 preview로 공개되었습니다.

Microsoft Azure Storage Explorer
; http://storageexplorer.com/

또한 소스코드도 github에 오픈소스로 공개된 상태입니다.

Project Deco - Azure Storage Explorer for OS X, Windows, and Linux
; https://github.com/azure-storage/deco

정성태

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
11131정성태1/13/2017637.NET : 478. C# - IL 코드 분석을 위한 팁 [2]
11130정성태1/11/2017261.NET : 477. x86 실행 환경에서 SECURITY_ATTRIBUTES 구조체를 CreateEvent에 전달할 때 예외 발생파일 다운로드1
11129정성태1/11/2017331.NET : 476. async/await에 대한 "There Is No Thread" 글의 부가 설명파일 다운로드1
11128정성태1/9/2017339.NET : 475. C# - Interlocked.CompareExchange 사용 예제 [3]파일 다운로드1
11127정성태1/8/2017180(개발자를 위한) Visual Studio의 "with MSDN" 라이선스 설명
11126정성태1/7/2017131Edge 웹 브라우저의 즐겨찾기(Favorites)를 편집/백업/복원하는 방법파일 다운로드1
11125정성태1/7/2017156개발 환경 구성: 308. IIS - appcmd.exe를 이용해 특정 페이지에 클라이언트 측 인증서를 제출하도록 설정하는 방법
11124정성태1/4/2017483개발 환경 구성: 307. 3년짜리 유효 기간을 제공하는 StartSSL [2]
11123정성태1/4/2017510.NET : 474. .NET Core의 dotnet.exe CLI 명령어 확장 방법 [1]
11122정성태1/3/2017431.NET : 473. TransactionScope에 사용자 정의 트랜잭션을 참여시키는 방법 [2]파일 다운로드1
11121정성태1/1/2017494개발 환경 구성: 306. "ASP.NET Core Web Application (.NET Core)"와 "ASP.NET Core Web Application (.NET Framework)" 차이점
11120정성태12/25/2016462개발 환경 구성: 305. ASP.NET Core Web Application을 IIS에서 호스팅하는 방법
11119정성태12/23/2016513개발 환경 구성: 304. Visual Studio Code에서 Python 개발 환경 구성
11118정성태12/22/2016331오류 유형 : 371. Python 64비트 설치 시 0x80070659 오류 발생
11117정성태12/21/2016577웹 : 22. nopCommerce 예제 사이트 구성 방법
11116정성태12/21/2016418Debug Features : 82. NopCommerce의 Autofac 부하(CPU, Memory) [2]
11115정성태12/21/2016402Windows: 73. 윈도우 서버 2016에서 플래시가 동작하지 않는 경우
11114정성태12/19/2016592Windows: 72. 역 슬래시(backslash) 문자가 왜 통화 표기 문자(한글인 경우 "\")로 보일까요? [2]
11113정성태12/19/2016481오류 유형 : 370. ICOMAdminCatalog::GetCollection에서 CO_E_ISOLEVELMISMATCH(0x8004E02F) 오류 발생파일 다운로드1
11112정성태11/23/2016461오류 유형 : 369. MySQL 서비스가 올라오지 않는 경우 - Error 1067
11111정성태11/23/2016627.NET : 472. C++로 만든 DLL 을 C#에서 사용하기
11110정성태11/17/2016643.NET : 471. Commit 메모리가 낮은 상황에서도 메모리 부족(Out-of-memory) 예외 발생
11109정성태11/17/2016670.NET : 470. ASP.NET에서 System.Web.HttpApplication 인스턴스는 다중으로 생성됩니다.
11108정성태11/13/2016449.NET : 469. WPF - Line 요소를 Canvas에 위치시켰을 때 흐림(blur) 현상파일 다운로드1
11107정성태11/10/2016607오류 유형 : 368. Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers.파일 다운로드1
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...