test.html
파일을 생성하고, 다음의 코드를 test.html
파일로 복사합니다.test.html
에 있는 서명 서비스 주소를 수정합니다.test.html
을 Web 서버에 놓고, 브라우저를 통해 페이지를 방문하고, 파일 업로드 기능을 테스트합니다.<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8" /><title>Ajax Put 업로드(서버 측 서명 계산)</title><style>h1,h2 {font-weight: normal;}#msg {margin-top: 10px;}</style></head><body><h1>Ajax Put 업로드(서버 측 서명 계산)</h1><input id="fileSelector" type="file" /><input id="submitBtn" type="submit" /><div id="msg"></div><script>(function () {// 더 많은 문자로 인코딩된 url encode 격식const camSafeUrlEncode = function (str) {return encodeURIComponent(str).replace(/!/g, '%21').replace(/'/g, '%27').replace(/\\(/g, '%28').replace(/\\)/g, '%29').replace(/\\*/g, '%2A');};// 서명 계산const getAuthorization = function (opt, callback) {// 자신의 서버 측 주소로 교체하여 put 업로드 서명을 얻습니다. demo: https://github.com/tencentyun/cos-demo/blob/main/server/upload-sign/nodejs/app.jsconst url = `http://127.0.0.1:3000/put-sign?ext=${opt.ext}`;const xhr = new XMLHttpRequest();xhr.open('GET', url, true);xhr.onload = function (e) {let credentials;try {const result = JSON.parse(e.target.responseText);credentials = result;} catch (e) {callback('서명 가져오기 오류');}if (credentials) {// 프린터하여 credentials가 정확한지 여부를 확인// console.log(credentials);callback(null, {securityToken: credentials.sessionToken,authorization: credentials.authorization,cosKey: credentials.cosKey,cosHost: credentials.cosHost,});} else {console.error(xhr.responseText);callback('서명 가져오기 오류');}};xhr.onerror = function (e) {callback('서명 가져오기 오류');};xhr.send();};// 파일 업로드const uploadFile = function (file, callback) {const fileName = file.name;// 파일 접미사 이름 가져오기let ext = '';const lastDotIndex = fileName.lastIndexOf('.');if (lastDotIndex > -1) {// 여기서는 파일 접미사를 가져옵니다. 서버 측에 의해 최종 업로드 경로가 생성됩니다.ext = fileName.substring(lastDotIndex + 1);}getAuthorization({ ext }, function (err, info) {if (err) {alert(err);return;}const auth = info.authorization;const securityToken = info.securityToken;const Key = info.cosKey;const protocol =location.protocol === 'https:' ? 'https:' : 'http:';const prefix = protocol + '//' + info.cosHost;const url =prefix + '/' + camSafeUrlEncode(Key).replace(/%2F/g, '/');const xhr = new XMLHttpRequest();xhr.open('PUT', url, true);xhr.setRequestHeader('Authorization', auth);securityToken &&xhr.setRequestHeader('x-cos-security-token', securityToken);xhr.upload.onprogress = function (e) {console.log('업로드 진행률 ' +Math.round((e.loaded / e.total) * 10000) / 100 +'%');};xhr.onload = function () {if (/^2\\d\\d$/.test('' + xhr.status)) {const ETag = xhr.getResponseHeader('etag');callback(null, { url: url, ETag: ETag });} else {callback('파일 ' + Key + ' 업로드 실패, 상태 코드: ' + xhr.status);}};xhr.onerror = function () {callback('파일 ' + Key + ' 업로드 실패, CORS 크로스 도메인 규칙이 구성되지 않았는지 여부를 검사하세요');};xhr.send(file);});};// 모니터링 리스트 제출document.getElementById('submitBtn').onclick = function (e) {const file = document.getElementById('fileSelector').files[0];if (!file) {document.getElementById('msg').innerText = '업로드 파일을 선택하지 않음';return;}file &&uploadFile(file, function (err, data) {console.log(err || data);document.getElementById('msg').innerText = err? err: '업로드 성공, ETag=' + data.ETag;});};})();</script></body></html>
test.html
파일을 생성하고,다음의 코드를 test.html
파일로 복사합니다.test.html
의 서명 서비스 주소를 수정합니다.test.html
을 Web 서버에 올리고, 브라우저로 페이지에 액세스하여 파일 업로드 기능을 테스트합니다.<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8" /><title>Ajax Post 업로드(서버 측 서명 계산)</title><style>h1,h2 {font-weight: normal;}#msg {margin-top: 10px;}</style></head><body><h1>PostObject 업로드(서버 측 서명 계산)</h1><input id="fileSelector" type="file" /><input id="submitBtn" type="submit" /><div id="msg"></div><script>(function () {let prefix = '';let Key = '';// 더 많은 문자로 인코딩된 url encode 격식const camSafeUrlEncode = function (str) {return encodeURIComponent(str).replace(/!/g, '%21').replace(/'/g, '%27').replace(/\\(/g, '%28').replace(/\\)/g, '%29').replace(/\\*/g, '%2A');};// 권한 정책 가져오기const getAuthorization = function (opt, callback) {// 자신의 서버 측 주소로 교체하여 post 업로드 서명을 얻습니다. demo: https://github.com/tencentyun/cos-demo/blob/main/server/upload-sign/nodejs/app.jsconst url = `http://127.0.0.1:3000/post-policy?ext=${opt.ext}`;const xhr = new XMLHttpRequest();xhr.open('GET', url, true);xhr.onload = function (e) {let credentials;try {const result = JSON.parse(e.target.responseText);credentials = result;} catch (e) {callback('서명 가져오기 오류');}if (credentials) {// 프린터하여 credentials가 정확한지 여부를 확인// console.log(credentials);callback(null, {securityToken: credentials.sessionToken,cosKey: credentials.cosKey,cosHost: credentials.cosHost,policy: credentials.policy,qAk: credentials.qAk,qKeyTime: credentials.qKeyTime,qSignAlgorithm: credentials.qSignAlgorithm,qSignature: credentials.qSignature,});} else {console.error(xhr.responseText);callback('서명 가져오기 오류');}};xhr.send();};// 업로드파일const uploadFile = function (file, callback) {const fileName = file.name;// 파일 접미사 이름 가져오기let ext = '';const lastDotIndex = fileName.lastIndexOf('.');if (lastDotIndex > -1) {// 여기서는 파일 접미사를 가져옵니다. 서버 측에 의해 최종 업로드 경로가 생성됩니다.ext = fileName.substring(lastDotIndex + 1);}getAuthorization({ ext }, function (err, credentials) {if (err) {alert(err);return;}const protocol =location.protocol === 'https:' ? 'https:' : 'http:';prefix = protocol + '//' + credentials.cosHost;Key = credentials.cosKey;const fd = new FormData();// 현재 목록에는 하나의 빈 empty.html를 추가합니다. 인터페이스 업로드가 완료되면 다시 이동하여 돌아오기 위해서입니다.fd.append('key', Key);// policy 서명 보호 격식 사용하기credentials.securityToken &&fd.append('x-cos-security-token', credentials.securityToken);fd.append('q-sign-algorithm', credentials.qSignAlgorithm);fd.append('q-ak', credentials.qAk);fd.append('q-key-time', credentials.qKeyTime);fd.append('q-signature', credentials.qSignature);fd.append('policy', credentials.policy);// 파일 내용, file 필드를 리스트 마지막에 놓아 파일 내용이 너무 길어 서명 판단과 인증에 영향을 미치는 것을 피면합니다.fd.append('file', file);// xhrconst url = prefix;const xhr = new XMLHttpRequest();xhr.open('POST', url, true);xhr.upload.onprogress = function (e) {console.log('업로드 진행률 ' +Math.round((e.loaded / e.total) * 10000) / 100 +'%');};xhr.onload = function () {if (Math.floor(xhr.status / 100) === 2) {const ETag = xhr.getResponseHeader('etag');callback(null, {url:prefix + '/' + camSafeUrlEncode(Key).replace(/%2F/g, '/'),ETag: ETag,});} else {callback('파일 ' + Key + ' 업로드 실패, 상태 코드: ' + xhr.status);}};xhr.onerror = function () {callback('파일 ' + Key + ' 업로드 실패, CORS 크로스 도메인 규칙이 구성되지 않았는지 여부를 검사하세요');};xhr.send(fd);});};// 모니터링 리스트 제출document.getElementById('submitBtn').onclick = function (e) {const file = document.getElementById('fileSelector').files[0];if (!file) {document.getElementById('msg').innerText = '업로드 파일을 선택하지 않음';return;}file &&uploadFile(file, function (err, data) {console.log(err || data);document.getElementById('msg').innerText = err? err: '업로드 성공, ETag=' + data.ETag + 'url=' + data.url;});};})();</script></body></html>
test.html
파일을 생성하고, 다음의 코드를 test.html
파일로 복사합니다.test.html
의 서명 서비스 주소를 수정합니다.test.html
과 동일한 디렉터리에 비어 있는 empty.html
을 생성하여 업로드 성공 시 리디렉션하는 데 사용합니다.test.html
과 empty.html
을 Web 서버에 올리고, 브라우저로 페이지에 액세스하여 파일 업로드 기능을 테스트합니다.<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8" /><title>Form 리스트 간단 업로드(IE8 호환 가능)(서버 측 서명 계산)</title><style>h1,h2 {font-weight: normal;}#msg {margin-top: 10px;}</style></head><body><h1>Form 리스트 간단 업로드(IE8 호환 가능)(서버 측 서명 계산)</h1><div>최소 IE6 업로드 호환 가능, onprogress</div>는 지원하지 않음<formid="form"target="submitTarget"action=""method="post"enctype="multipart/form-data"accept="*/*"><input id="name" name="name" type="hidden" value="" /><input name="success_action_status" type="hidden" value="200" /><inputid="success_action_redirect"name="success_action_redirect"type="hidden"value=""/><input id="key" name="key" type="hidden" value="" /><input id="policy" name="policy" type="hidden" value="" /><inputid="q-sign-algorithm"name="q-sign-algorithm"type="hidden"value=""/><input id="q-ak" name="q-ak" type="hidden" value="" /><input id="q-key-time" name="q-key-time" type="hidden" value="" /><input id="q-signature" name="q-signature" type="hidden" value="" /><input name="Content-Type" type="hidden" value="" /><inputid="x-cos-security-token"name="x-cos-security-token"type="hidden"value=""/><!-- file 필드를 리스트 마지막에 놓음으로써 파일 내용이 너무 길어 서명 판단과 인증에 영향을 미치는 것을 방지합니다.--><input id="fileSelector" name="file" type="file" /><input id="submitBtn" type="button" value="제출" /></form><iframeid="submitTarget"name="submitTarget"style="display: none"frameborder="0"></iframe><div id="msg"></div><script>(function () {const form = document.getElementById('form');let prefix = '';// 더 많은 문자로 인코딩된 url encode 격식const camSafeUrlEncode = function (str) {return encodeURIComponent(str).replace(/!/g, '%21').replace(/'/g, '%27').replace(/\\(/g, '%28').replace(/\\)/g, '%29').replace(/\\*/g, '%2A');};// 서명 계산const getAuthorization = function (opt, callback) {// 자신의 서버 측 주소로 교체하여 post 업로드 서명을 얻습니다. demo: https://github.com/tencentyun/cos-demo/blob/main/server/upload-sign/nodejs/app.jsconst url = `http://127.0.0.1:3000/post-policy?ext=${opt.ext || ''}`;const xhr = new XMLHttpRequest();xhr.open('GET', url, true);xhr.onload = function (e) {let credentials;try {const result = JSON.parse(e.target.responseText);credentials = result;} catch (e) {callback('서명 가져오기 오류');}if (credentials) {// 프린터하여 credentials가 정확한지 여부를 확인// console.log(credentials);callback(null, {securityToken: credentials.sessionToken,cosKey: credentials.cosKey,cosHost: credentials.cosHost,policy: credentials.policy,qAk: credentials.qAk,qKeyTime: credentials.qKeyTime,qSignAlgorithm: credentials.qSignAlgorithm,qSignature: credentials.qSignature,});} else {console.error(xhr.responseText);callback('서명 가져오기 오류');}};xhr.send();};// 모니터링 업로드 완료let Key;const submitTarget = document.getElementById('submitTarget');const showMessage = function (err, data) {console.log(err || data);document.getElementById('msg').innerText = err? err: '업로드 성공, ETag=' + data.ETag;};submitTarget.onload = function () {let search;try {search = submitTarget.contentWindow.location.search.substr(1);} catch (e) {showMessage('파일 ' + Key + ' 업로드 실패');}if (search) {const items = search.split('&');let i = 0;let arr = [];const data = {};for (i = 0; i < items.length; i++) {arr = items[i].split('=');data[arr[0]] = decodeURIComponent(arr[1] || '');}showMessage(null, {url: prefix + camSafeUrlEncode(Key).replace(/%2F/g, '/'),ETag: data.etag,});} else {}};// 업로드 개시document.getElementById('submitBtn').onclick = function (e) {const filePath = document.getElementById('fileSelector').value;if (!filePath) {document.getElementById('msg').innerText = '업로드 파일을 선택하지 않음';return;}// 파일 접미사 이름 가져오기let ext = '';const lastDotIndex = filePath.lastIndexOf('.');if (lastDotIndex > -1) {// 여기서는 파일 접미사를 가져옵니다. 서버 측에 의해 최종 업로드 경로가 생성됩니다.ext = filePath.substring(lastDotIndex + 1);}getAuthorization({ ext }, function (err, AuthData) {if (err) {alert(err);return;}const protocol =location.protocol === 'https:' ? 'https:' : 'http:';prefix = protocol + '//' + AuthData.cosHost;form.action = prefix;Key = AuthData.cosKey;// 현재 목록에는 하나의 빈 empty.html를 추가합니다. 인터페이스 업로드가 완료되면 다시 이동하여 돌아오기 위해서입니다.document.getElementById('success_action_redirect').value =location.href.substr(0, location.href.lastIndexOf('/') + 1) +'empty.html';document.getElementById('key').value = AuthData.cosKey;document.getElementById('policy').value = AuthData.policy;document.getElementById('q-sign-algorithm').value =AuthData.qSignAlgorithm;document.getElementById('q-ak').value = AuthData.qAk;document.getElementById('q-key-time').value = AuthData.qKeyTime;document.getElementById('q-signature').value = AuthData.qSignature;document.getElementById('x-cos-security-token').value =AuthData.securityToken || '';form.submit();});};})();</script></body></html>
// 기타 코드 생략document.getElementById('submitBtn').onclick = function (e) {const file = document.getElementById('fileSelector').files[0];if (!file) {document.getElementById('msg').innerText = '업로드 파일을 선택하지 않음';return;}// 파일 접미사 가져오기const fileName = file.name;const lastDotIndex = fileName.lastIndexOf('.');const ext = lastDotIndex > -1 ? fileName.substring(lastDotIndex + 1) : '';// 제한하고 싶은 격식으로 교체하세요. 예를 들어 jpg png 유형 파일만 업로드합니다.const allowExt = ['jpg', 'png'];if (!allowExt.includes(ext)) {alert('jpg, png 파일 업로드만 지원');return;}file &&uploadFile(file, function (err, data) {console.log(err || data);document.getElementById('msg').innerText = err? err: '업로드 성공, ETag=' + data.ETag + 'url=' + data.url;});};
// 기타 코드 생략document.getElementById('submitBtn').onclick = function (e) {const file = document.getElementById('fileSelector').files[0];if (!file) {document.getElementById('msg').innerText = '업로드 파일을 선택하지 않음';return;}const fileSize = file.size;// 제한하고 싶은 개체 크기로 교체하세요. 최대 하나의 개체가 5GB까지 가능합니다. 예를 들어 업로드 파일이 5MB를 초과할 수 없습니다.if (fileSize > 5 * 1024 * 1024) {alert('선택한 파일이 5MB를 초과하였습니다. 다시 선택하세요.');return;}file &&uploadFile(file, function (err, data) {console.log(err || data);document.getElementById('msg').innerText = err? err: '업로드 성공, ETag=' + data.ETag + 'url=' + data.url;});};
문제 해결에 도움이 되었나요?