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)
13843정성태12/13/20244391오류 유형: 938. Docker container 내에서 빌드 시 error MSB3021: Unable to copy file "..." to "...". Access to the path '...' is denied.
13842정성태12/12/20244534디버깅 기술: 205. Windbg - KPCR, KPRCB
13841정성태12/11/20244866오류 유형: 937. error MSB4044: The "ValidateValidArchitecture" task was not given a value for the required parameter "RemoteTarget"
13840정성태12/11/20244439오류 유형: 936. msbuild - Your project file doesn't list 'win' as a "RuntimeIdentifier"
13839정성태12/11/20244877오류 유형: 936. msbuild - error CS1617: Invalid option '12.0' for /langversion. Use '/langversion:?' to list supported values.
13838정성태12/4/20244604오류 유형: 935. Windbg - Breakpoint 0's offset expression evaluation failed.
13837정성태12/3/20245073디버깅 기술: 204. Windbg - 윈도우 핸들 테이블 (3) - Windows 10 이상인 경우
13836정성태12/3/20244625디버깅 기술: 203. Windbg - x64 가상 주소를 물리 주소로 변환 (페이지 크기가 2MB인 경우)
13835정성태12/2/20245071오류 유형: 934. Azure - rm: cannot remove '...': Directory not empty
13834정성태11/29/20245302Windows: 275. C# - CUI 애플리케이션과 Console 윈도우 (Windows 10 미만의 Classic Console 모드인 경우) [1]파일 다운로드1
13833정성태11/29/20244979개발 환경 구성: 737. Azure Web App에서 Scale-out으로 늘어난 리눅스 인스턴스에 SSH 접속하는 방법
13832정성태11/27/20244913Windows: 274. Windows 7부터 도입한 conhost.exe
13831정성태11/27/20244382Linux: 111. eBPF - BPF_MAP_TYPE_PERF_EVENT_ARRAY, BPF_MAP_TYPE_RINGBUF에 대한 다양한 용어들
13830정성태11/25/20245200개발 환경 구성: 736. 파이썬 웹 앱을 Azure App Service에 배포하기
13829정성태11/25/20245169스크립트: 67. 파이썬 - Windows 버전에서 함께 설치되는 py.exe
13828정성태11/25/20244450개발 환경 구성: 735. Azure - 압축 파일을 이용한 web app 배포 시 디렉터리 구분이 안 되는 문제파일 다운로드1
13827정성태11/25/20245105Windows: 273. Windows 환경의 파일 압축 방법 (tar, Compress-Archive)
13826정성태11/21/20245337닷넷: 2313. C# - (비밀번호 등의) Console로부터 입력받을 때 문자열 출력 숨기기(echo 끄기)파일 다운로드1
13825정성태11/21/20245672Linux: 110. eBPF / bpf2go - BPF_RINGBUF_OUTPUT / BPF_MAP_TYPE_RINGBUF 사용법
13824정성태11/20/20244751Linux: 109. eBPF / bpf2go - BPF_PERF_OUTPUT / BPF_MAP_TYPE_PERF_EVENT_ARRAY 사용법
13823정성태11/20/20245304개발 환경 구성: 734. Ubuntu에 docker, kubernetes (k3s) 설치
13822정성태11/20/20245171개발 환경 구성: 733. Windbg - VirtualBox VM의 커널 디버거 연결 시 COM 포트가 없는 경우
13821정성태11/18/20245094Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식
13820정성태11/18/20245252VS.NET IDE: 195. Visual C++ - C# 프로젝트처럼 CopyToOutputDirectory 항목을 추가하는 방법
13819정성태11/15/20244490Linux: 107. eBPF - libbpf CO-RE의 CONFIG_DEBUG_INFO_BTF 빌드 여부에 대한 의존성
13818정성태11/15/20245298Windows: 272. Windows 11 24H2 - sudo 추가
1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...