Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

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

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

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

  • Database
  • 첨부 파일

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

sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
; https://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 웹 사이트의 경우에는 위와 같이 업/다운로드 처리를 하는 코드 작업이 주요했고 이외에는 거의 변경된 것이 없습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/17/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.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 에 보내는 방법이 있었군요! 좋은 정보 알아 갑니다.
[guest]
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
정성태
2018-09-05 01시30분
Azure Storage에서 정적 웹 페이지 호스팅하는 기능 제공

Static website hosting for Azure Storage now in public preview
; https://azure.microsoft.com/en-us/blog/azure-storage-static-web-hosting-public-preview/
; https://learn.microsoft.com/en-us/azure/storage/blobs/storage-blob-static-website
정성태

1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13818정성태11/15/20245308Windows: 272. Windows 11 24H2 - sudo 추가
13817정성태11/14/20244945Linux: 106. eBPF / bpf2go - (BPF_MAP_TYPE_HASH) Map을 이용한 전역 변수 구현
13816정성태11/14/20245400닷넷: 2312. C#, C++ - Windows / Linux 환경의 Thread Name 설정파일 다운로드1
13815정성태11/13/20244822Linux: 105. eBPF - bpf2go에서 전역 변수 설정 방법
13814정성태11/13/20245294닷넷: 2311. C# - Windows / Linux 환경에서 Native Thread ID 가져오기파일 다운로드1
13813정성태11/12/20245053닷넷: 2310. .NET의 Rune 타입과 emoji 표현파일 다운로드1
13812정성태11/11/20245274오류 유형: 933. Active Directory - The forest functional level is not supported.
13811정성태11/11/20244862Linux: 104. Linux - COLUMNS 환경변수가 언제나 80으로 설정되는 환경
13810정성태11/10/20245391Linux: 103. eBPF (bpf2go) - Tracepoint를 이용한 트레이스 (BPF_PROG_TYPE_TRACEPOINT)
13809정성태11/10/20245267Windows: 271. 윈도우 서버 2025 마이그레이션
13808정성태11/9/20245272오류 유형: 932. Linux - 커널 업그레이드 후 "error: bad shim signature" 오류 발생
13807정성태11/9/20244996Linux: 102. Linux - 커널 이미지 파일 서명 (Ubuntu 환경)
13806정성태11/8/20244917Windows: 270. 어댑터 상세 정보(Network Connection Details) 창의 내용이 비어 있는 경우
13805정성태11/8/20244749오류 유형: 931. Active Directory의 adprep 또는 복제가 안 되는 경우
13804정성태11/7/20245379Linux: 101. eBPF 함수의 인자를 다루는 방법
13803정성태11/7/20245337닷넷: 2309. C# - .NET Core에서 바뀐 DateTime.Ticks의 정밀도
13802정성태11/6/20245718Windows: 269. GetSystemTimeAsFileTime과 GetSystemTimePreciseAsFileTime의 차이점파일 다운로드1
13801정성태11/5/20245493Linux: 100. eBPF의 2가지 방식 - libbcc와 libbpf(CO-RE)
13800정성태11/3/20246341닷넷: 2308. C# - ICU 라이브러리를 활용한 문자열의 대소문자 변환 [2]파일 다운로드1
13799정성태11/2/20244922개발 환경 구성: 732. 모바일 웹 브라우저에서 유니코드 문자가 표시되지 않는 경우
13798정성태11/2/20245526개발 환경 구성: 731. 유니코드 - 출력 예시 및 폰트 찾기
13797정성태11/1/20245515C/C++: 185. C++ - 문자열의 대소문자를 변환하는 transform + std::tolower/toupper 방식의 문제점파일 다운로드1
13796정성태10/31/20245396C/C++: 184. C++ - ICU dll을 이용하는 예제 코드 (Windows)파일 다운로드1
13795정성태10/31/20245179Windows: 268. Windows - 리눅스 환경처럼 공백으로 끝나는 프롬프트 만들기
13794정성태10/30/20245272닷넷: 2307. C# - 윈도우에서 한글(및 유니코드)을 포함한 콘솔 프로그램을 컴파일 및 실행하는 방법
13793정성태10/28/20245149C/C++: 183. C++ - 윈도우에서 한글(및 유니코드)을 포함한 콘솔 프로그램을 컴파일 및 실행하는 방법
1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...