[Backend] 上傳3D模型接口
This commit is contained in:
parent
930b06de18
commit
b3fe76e37e
@ -18,6 +18,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Autodesk.Forge" Version="1.9.7" />
|
||||||
<PackageReference Include="iTextSharp" Version="5.5.13.2" />
|
<PackageReference Include="iTextSharp" Version="5.5.13.2" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="3.1.21" />
|
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="3.1.21" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Mvc.Razor.RuntimeCompilation" Version="3.1.20" />
|
<PackageReference Include="Microsoft.AspNetCore.Mvc.Razor.RuntimeCompilation" Version="3.1.20" />
|
||||||
|
53
Backend/Controllers/ModelDerivativeController.cs
Normal file
53
Backend/Controllers/ModelDerivativeController.cs
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
using Autodesk.Forge;
|
||||||
|
using Autodesk.Forge.Model;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace forgeSample.Controllers
|
||||||
|
{
|
||||||
|
[ApiController]
|
||||||
|
public class ModelDerivativeController : ControllerBase
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Start the translation job for a give bucketKey/objectName
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="objModel"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
[HttpPost]
|
||||||
|
[Route("api/forge/modelderivative/jobs")]
|
||||||
|
public async Task<dynamic> TranslateObject([FromBody] TranslateObjectModel objModel)
|
||||||
|
{
|
||||||
|
dynamic oauth = await OAuthController.GetInternalAsync();
|
||||||
|
|
||||||
|
// prepare the payload
|
||||||
|
List<JobPayloadItem> outputs = new List<JobPayloadItem>()
|
||||||
|
{
|
||||||
|
new JobPayloadItem(
|
||||||
|
JobPayloadItem.TypeEnum.Svf,
|
||||||
|
new List<JobPayloadItem.ViewsEnum>()
|
||||||
|
{
|
||||||
|
JobPayloadItem.ViewsEnum._2d,
|
||||||
|
JobPayloadItem.ViewsEnum._3d
|
||||||
|
})
|
||||||
|
};
|
||||||
|
JobPayload job;
|
||||||
|
job = new JobPayload(new JobPayloadInput(objModel.objectName), new JobPayloadOutput(outputs));
|
||||||
|
|
||||||
|
// start the translation
|
||||||
|
DerivativesApi derivative = new DerivativesApi();
|
||||||
|
derivative.Configuration.AccessToken = oauth.access_token;
|
||||||
|
dynamic jobPosted = await derivative.TranslateAsync(job);
|
||||||
|
return jobPosted;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Model for TranslateObject method
|
||||||
|
/// </summary>
|
||||||
|
public class TranslateObjectModel
|
||||||
|
{
|
||||||
|
public string bucketKey { get; set; }
|
||||||
|
public string objectName { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
70
Backend/Controllers/OAuthController.cs
Normal file
70
Backend/Controllers/OAuthController.cs
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
using Autodesk.Forge;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace forgeSample.Controllers
|
||||||
|
{
|
||||||
|
[ApiController]
|
||||||
|
public class OAuthController : ControllerBase
|
||||||
|
{
|
||||||
|
// As both internal & public tokens are used for all visitors
|
||||||
|
// we don't need to request a new token on every request, so let's
|
||||||
|
// cache them using static variables. Note we still need to refresh
|
||||||
|
// them after the expires_in time (in seconds)
|
||||||
|
private static dynamic InternalToken { get; set; }
|
||||||
|
private static dynamic PublicToken { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get access token with public (viewables:read) scope
|
||||||
|
/// </summary>
|
||||||
|
[HttpGet]
|
||||||
|
[Route("api/forge/oauth/token")]
|
||||||
|
public async Task<dynamic> GetPublicAsync()
|
||||||
|
{
|
||||||
|
if (PublicToken == null || PublicToken.ExpiresAt < DateTime.UtcNow)
|
||||||
|
{
|
||||||
|
PublicToken = await Get2LeggedTokenAsync(new Scope[] { Scope.ViewablesRead });
|
||||||
|
PublicToken.ExpiresAt = DateTime.UtcNow.AddSeconds(PublicToken.expires_in);
|
||||||
|
}
|
||||||
|
return PublicToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get access token with internal (write) scope
|
||||||
|
/// </summary>
|
||||||
|
public static async Task<dynamic> GetInternalAsync()
|
||||||
|
{
|
||||||
|
if (InternalToken == null || InternalToken.ExpiresAt < DateTime.UtcNow)
|
||||||
|
{
|
||||||
|
InternalToken = await Get2LeggedTokenAsync(new Scope[] { Scope.BucketCreate, Scope.BucketRead, Scope.BucketDelete, Scope.DataRead, Scope.DataWrite, Scope.DataCreate, Scope.CodeAll });
|
||||||
|
InternalToken.ExpiresAt = DateTime.UtcNow.AddSeconds(InternalToken.expires_in);
|
||||||
|
}
|
||||||
|
|
||||||
|
return InternalToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get the access token from Autodesk
|
||||||
|
/// </summary>
|
||||||
|
private static async Task<dynamic> Get2LeggedTokenAsync(Scope[] scopes)
|
||||||
|
{
|
||||||
|
TwoLeggedApi oauth = new TwoLeggedApi();
|
||||||
|
string grantType = "client_credentials";
|
||||||
|
dynamic bearer = await oauth.AuthenticateAsync(
|
||||||
|
GetAppSetting("FORGE_CLIENT_ID"),
|
||||||
|
GetAppSetting("FORGE_CLIENT_SECRET"),
|
||||||
|
grantType,
|
||||||
|
scopes);
|
||||||
|
return bearer;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Reads appsettings from web.config
|
||||||
|
/// </summary>
|
||||||
|
public static string GetAppSetting(string settingKey)
|
||||||
|
{
|
||||||
|
return Environment.GetEnvironmentVariable(settingKey).Trim();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
150
Backend/Controllers/OSSController.cs
Normal file
150
Backend/Controllers/OSSController.cs
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
using Autodesk.Forge;
|
||||||
|
using Autodesk.Forge.Model;
|
||||||
|
using Microsoft.AspNetCore.Hosting;
|
||||||
|
using Microsoft.AspNetCore.Http;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace forgeSample.Controllers
|
||||||
|
{
|
||||||
|
[ApiController]
|
||||||
|
public class OSSController : ControllerBase
|
||||||
|
{
|
||||||
|
private IWebHostEnvironment _env;
|
||||||
|
public OSSController(IWebHostEnvironment env) { _env = env; }
|
||||||
|
public string ClientId { get { return OAuthController.GetAppSetting("FORGE_CLIENT_ID").ToLower(); } }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Return list of buckets (id=#) or list of objects (id=bucketKey)
|
||||||
|
/// </summary>
|
||||||
|
[HttpGet]
|
||||||
|
[Route("api/forge/oss/buckets")]
|
||||||
|
public async Task<IList<TreeNode>> GetOSSAsync(string id)
|
||||||
|
{
|
||||||
|
IList<TreeNode> nodes = new List<TreeNode>();
|
||||||
|
dynamic oauth = await OAuthController.GetInternalAsync();
|
||||||
|
|
||||||
|
if (id == "#") // root
|
||||||
|
{
|
||||||
|
// in this case, let's return all buckets
|
||||||
|
BucketsApi appBckets = new BucketsApi();
|
||||||
|
appBckets.Configuration.AccessToken = oauth.access_token;
|
||||||
|
|
||||||
|
// to simplify, let's return only the first 100 buckets
|
||||||
|
dynamic buckets = await appBckets.GetBucketsAsync("US", 100);
|
||||||
|
foreach (KeyValuePair<string, dynamic> bucket in new DynamicDictionaryItems(buckets.items))
|
||||||
|
{
|
||||||
|
nodes.Add(new TreeNode(bucket.Value.bucketKey, bucket.Value.bucketKey.Replace(ClientId + "-", string.Empty), "bucket", true));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// as we have the id (bucketKey), let's return all
|
||||||
|
ObjectsApi objects = new ObjectsApi();
|
||||||
|
objects.Configuration.AccessToken = oauth.access_token;
|
||||||
|
var objectsList = await objects.GetObjectsAsync(id, 100);
|
||||||
|
foreach (KeyValuePair<string, dynamic> objInfo in new DynamicDictionaryItems(objectsList.items))
|
||||||
|
{
|
||||||
|
nodes.Add(new TreeNode(Base64Encode((string)objInfo.Value.objectId),
|
||||||
|
objInfo.Value.objectKey, "object", false));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nodes;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Model data for jsTree used on GetOSSAsync
|
||||||
|
/// </summary>
|
||||||
|
public class TreeNode
|
||||||
|
{
|
||||||
|
public TreeNode(string id, string text, string type, bool children)
|
||||||
|
{
|
||||||
|
this.id = id;
|
||||||
|
this.text = text;
|
||||||
|
this.type = type;
|
||||||
|
this.children = children;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string id { get; set; }
|
||||||
|
public string text { get; set; }
|
||||||
|
public string type { get; set; }
|
||||||
|
public bool children { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Create a new bucket
|
||||||
|
/// </summary>
|
||||||
|
[HttpPost]
|
||||||
|
[Route("api/forge/oss/buckets")]
|
||||||
|
public async Task<dynamic> CreateBucket([FromBody] CreateBucketModel bucket)
|
||||||
|
{
|
||||||
|
BucketsApi buckets = new BucketsApi();
|
||||||
|
dynamic token = await OAuthController.GetInternalAsync();
|
||||||
|
buckets.Configuration.AccessToken = token.access_token;
|
||||||
|
PostBucketsPayload bucketPayload = new PostBucketsPayload(string.Format("{0}-{1}", ClientId, bucket.bucketKey.ToLower()), null,
|
||||||
|
PostBucketsPayload.PolicyKeyEnum.Transient);
|
||||||
|
return await buckets.CreateBucketAsync(bucketPayload, "US");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Input model for CreateBucket method
|
||||||
|
/// </summary>
|
||||||
|
public class CreateBucketModel
|
||||||
|
{
|
||||||
|
public string bucketKey { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Receive a file from the client and upload to the bucket
|
||||||
|
/// </summary>
|
||||||
|
/// <returns></returns>
|
||||||
|
[HttpPost]
|
||||||
|
[Route("api/forge/oss/objects")]
|
||||||
|
public async Task<dynamic> UploadObject([FromForm] UploadFile input)
|
||||||
|
{
|
||||||
|
// save the file on the server
|
||||||
|
var fileSavePath = Path.Combine(_env.WebRootPath, Path.GetFileName(input.fileToUpload.FileName));
|
||||||
|
using (var stream = new FileStream(fileSavePath, FileMode.Create))
|
||||||
|
await input.fileToUpload.CopyToAsync(stream);
|
||||||
|
|
||||||
|
|
||||||
|
// get the bucket...
|
||||||
|
dynamic oauth = await OAuthController.GetInternalAsync();
|
||||||
|
ObjectsApi objects = new ObjectsApi();
|
||||||
|
objects.Configuration.AccessToken = oauth.access_token;
|
||||||
|
|
||||||
|
// upload the file/object, which will create a new object
|
||||||
|
dynamic uploadedObj;
|
||||||
|
using (StreamReader streamReader = new StreamReader(fileSavePath))
|
||||||
|
{
|
||||||
|
uploadedObj = await objects.UploadObjectAsync(input.bucketKey,
|
||||||
|
Path.GetFileName(input.fileToUpload.FileName), (int)streamReader.BaseStream.Length, streamReader.BaseStream,
|
||||||
|
"application/octet-stream");
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanup
|
||||||
|
System.IO.File.Delete(fileSavePath);
|
||||||
|
var a = Base64Encode((string)uploadedObj.objectId);
|
||||||
|
//var test = new TreeNode(Base64Encode((string)uploadedObj.objectId), objInfo.Value.objectKey, "object", false)
|
||||||
|
|
||||||
|
return uploadedObj;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class UploadFile
|
||||||
|
{
|
||||||
|
public string bucketKey { get; set; }
|
||||||
|
public IFormFile fileToUpload { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Base64 enconde a string
|
||||||
|
/// </summary>
|
||||||
|
public static string Base64Encode(string plainText)
|
||||||
|
{
|
||||||
|
var plainTextBytes = System.Text.Encoding.UTF8.GetBytes(plainText);
|
||||||
|
return System.Convert.ToBase64String(plainTextBytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -13,7 +13,10 @@
|
|||||||
"launchBrowser": true,
|
"launchBrowser": true,
|
||||||
"environmentVariables": {
|
"environmentVariables": {
|
||||||
"ASPNETCORE_ENVIRONMENT": "Development",
|
"ASPNETCORE_ENVIRONMENT": "Development",
|
||||||
"ASPNETCORE_HOSTINGSTARTUPASSEMBLIES": "Microsoft.AspNetCore.Mvc.Razor.RuntimeCompilation"
|
"ASPNETCORE_HOSTINGSTARTUPASSEMBLIES": "Microsoft.AspNetCore.Mvc.Razor.RuntimeCompilation",
|
||||||
|
"FORGE_CALLBACK_URL": "http://localhost:3000/api/forge/callback/oauth",
|
||||||
|
"FORGE_CLIENT_ID": "TA3hqsFfzQbNOUXKpldKUKSew4SJ21w5",
|
||||||
|
"FORGE_CLIENT_SECRET": "D002f92d839144f8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Backend": {
|
"Backend": {
|
||||||
|
@ -704,6 +704,25 @@
|
|||||||
//#region 變更樓層平面圖
|
//#region 變更樓層平面圖
|
||||||
function changeImage(input) {
|
function changeImage(input) {
|
||||||
$(`#map_file_preview_modal`).attr("data-src", window.URL.createObjectURL(input.files[0]));
|
$(`#map_file_preview_modal`).attr("data-src", window.URL.createObjectURL(input.files[0]));
|
||||||
|
|
||||||
|
console.log("---- abc test ---");
|
||||||
|
var file = input.files[0];
|
||||||
|
|
||||||
|
var formData = new FormData();
|
||||||
|
formData.append('fileToUpload', file);
|
||||||
|
formData.append('bucketKey', "ta3hqsffzqbnouxkpldkuksew4sj21w5-bims_models");
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: '/api/forge/oss/objects',
|
||||||
|
data: formData,
|
||||||
|
processData: false,
|
||||||
|
contentType: false,
|
||||||
|
type: 'POST',
|
||||||
|
success: function (data) {
|
||||||
|
$('#appBuckets').jstree(true).refresh_node(node);
|
||||||
|
_this.value = '';
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
//#endregion
|
//#endregion
|
||||||
</script>
|
</script>
|
||||||
|
@ -58,7 +58,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="form-group col-12">
|
<div class="form-group col-12">
|
||||||
<label class="form-label" for="build_file_3d_modal">3D檔(限制SVG格式)</label>
|
<label class="form-label" for="build_file_3d_modal">3D檔(限制SVG格式)</label>
|
||||||
<input type="file" id="build_file_3d_modal" class="form-control" name="build_file_3d_modal" onchange="changeImage(this)" accept="image/svg+xml">
|
<input type="file" id="build_file_3d_modal" class="form-control" name="build_file_3d_modal" onchange="changeImage(this)">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
Loading…
Reference in New Issue
Block a user