ScreenController.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. using HTEXScreen.Service;
  2. using Microsoft.AspNetCore.Authorization;
  3. using Microsoft.AspNetCore.Mvc;
  4. using PuppeteerSharp;
  5. using PuppeteerSharp.Media;
  6. using System.Configuration;
  7. using System.Drawing;
  8. using System.IO;
  9. using System.Runtime.InteropServices;
  10. using System.Text;
  11. using System.Text.Json;
  12. using System.Text.RegularExpressions;
  13. using System.Web;
  14. using TEAMModelOS.SDK.DI;
  15. namespace HTEX.Screen.Controllers
  16. {
  17. [ApiController]
  18. [Route("screen")]
  19. public class ScreenController : ControllerBase
  20. {
  21. private readonly AzureStorageFactory _azureStorage;
  22. private readonly HttpClient _httpClient;
  23. private readonly IConfiguration _configuration;
  24. // private readonly HttpContext _httpContext;
  25. public ScreenController(HttpClient httpClient, AzureStorageFactory azureStorage, IConfiguration configuration)
  26. {
  27. _httpClient = httpClient;
  28. _azureStorage = azureStorage;
  29. _configuration = configuration;
  30. // _httpContext=httpContext;
  31. }
  32. /// <summary>
  33. /// 上传到blob
  34. /// </summary>
  35. /// <param name="request"></param>
  36. /// <returns></returns>
  37. [HttpPost("from-miniapp-delete")]
  38. [Authorize(Roles = "AClassONE")]
  39. [RequestSizeLimit(102_400_000_00)] //最大10000m左右
  40. public async Task<IActionResult> FromMiniAPPDelete(JsonElement json)
  41. {
  42. List<string>? urls = json.GetProperty("urls").Deserialize<List<string>>();
  43. string? cnt = json.GetProperty("cnt").GetString();
  44. string? test = json.GetProperty("test").GetString();
  45. if (urls!=null)
  46. {
  47. foreach (var url in urls)
  48. {
  49. string dev = "Default";
  50. if (test.Equals("Test", StringComparison.OrdinalIgnoreCase))
  51. {
  52. dev="Test";
  53. }
  54. string uri = url.Split("?")[0];
  55. var blobinfo = BlobUrlString(uri);
  56. if (blobinfo.ContainerName.Equals(cnt))
  57. {
  58. var uld = HttpUtility.UrlDecode(blobinfo.blob);
  59. bool ds = await _azureStorage.GetBlobContainerClient(cnt, dev).DeleteBlobIfExistsAsync(uld);
  60. }
  61. }
  62. }
  63. return Ok();
  64. }
  65. private static (string ContainerName, string blob) BlobUrlString(string sasUrl)
  66. {
  67. sasUrl = sasUrl.Substring(8);
  68. string[] sasUrls = sasUrl.Split("/");
  69. string ContainerName;
  70. ContainerName = sasUrls[1].Clone().ToString();
  71. string item = sasUrls[0] + "/" + sasUrls[1] + "/";
  72. string blob = sasUrl.Replace(item, "");
  73. return (ContainerName, blob);
  74. }
  75. /// <summary>
  76. /// 上传到blob
  77. /// </summary>
  78. /// <param name="request"></param>
  79. /// <returns></returns>
  80. [HttpPost("from-miniapp-v2")]
  81. [Authorize(Roles = "AClassONE")]
  82. [RequestSizeLimit(102_400_000_00)] //最大10000m左右
  83. public async Task<IActionResult> FromMiniAPPV2()
  84. {
  85. try
  86. {
  87. List<dynamic> urls = new List<dynamic>();
  88. var request = HttpContext.Request;
  89. var files = request.Form.Files;
  90. var cnt = request.Form["cnt"].ToString();
  91. var test = request.Form["test"].ToString();
  92. var path = request.Form["path"].ToString();
  93. foreach (var file in files)
  94. {
  95. string dev = "Default";
  96. if (test.Equals("Test", StringComparison.OrdinalIgnoreCase))
  97. {
  98. dev="Test";
  99. }
  100. if (path.EndsWith("/"))
  101. {
  102. path=path.Substring(0, path.Length -1);
  103. }
  104. (string name, string url) = await _azureStorage.GetBlobContainerClient(cnt, dev).UploadFileByContainerBName(file.OpenReadStream(), path, $"{file.FileName}", true);
  105. var auth = _azureStorage.GetBlobContainerClient(cnt, dev).GetBlobSasUriRead(_configuration, cnt, name, dev);
  106. var ext = file.FileName.Split(".");
  107. urls.Add(new { url = $"{url}?{auth.sas}", name = file.FileName, size = file.Length, extension = ext[ext.Length-1], type = file.ContentType.Split("/")[0], blob = name, cnt = cnt });
  108. }
  109. return Ok(new { code = 200, urls, });
  110. }
  111. catch (Exception vErr)
  112. {
  113. return Ok(new { code = 500, msg = vErr.Message });
  114. }
  115. }
  116. /// <summary>
  117. /// 上传到blob
  118. /// </summary>
  119. /// <param name="request"></param>
  120. /// <returns></returns>
  121. [HttpPost("from-miniapp")]
  122. [Authorize(Roles = "AClassONE")]
  123. [RequestSizeLimit(102_400_000_00)] //最大10000m左右
  124. public async Task<IActionResult> FromMiniAPP([FromForm] IFormFile[] files, [FromForm] string path, [FromForm] string cnt, [FromForm] string test)
  125. {
  126. List<string> urls = new List<string>();
  127. foreach (var file in files)
  128. {
  129. string dev = "Default";
  130. if (test.Equals("Test", StringComparison.OrdinalIgnoreCase))
  131. {
  132. dev="Test";
  133. }
  134. if (path.EndsWith("/"))
  135. {
  136. path=path.Substring(0, path.Length -1);
  137. }
  138. (string name, string url) = await _azureStorage.GetBlobContainerClient(cnt, dev).UploadFileByContainerBName(file.OpenReadStream(), path, $"{file.FileName}", true);
  139. var auth = _azureStorage.GetBlobContainerClient(cnt, dev).GetBlobSasUriRead(_configuration, cnt, name, dev);
  140. urls.Add($"{url}?{auth.sas}");
  141. }
  142. return Ok(new { urls });
  143. }
  144. [HttpGet("download")]
  145. public async Task<IActionResult> Download([FromQuery] ScreenshotDto screenshot)
  146. {
  147. try
  148. {
  149. HttpResponseMessage response = await _httpClient.GetAsync(screenshot.url);
  150. if (!string.IsNullOrWhiteSpace(screenshot?.url))
  151. {
  152. string? url = screenshot?.url;
  153. string[] path = url.Split("/");
  154. string fileName = path[path.Length - 1];
  155. fileName=HttpUtility.UrlDecode(fileName);
  156. Stream stream = response.Content.ReadAsStream();
  157. if (!Directory.Exists("Download"))
  158. {
  159. Directory.CreateDirectory("Download");
  160. }
  161. FileStream fs = new FileStream($"Download/{fileName}", FileMode.Create);
  162. byte[] bytes = new byte[stream.Length];
  163. stream.Read(bytes, 0, bytes.Length);
  164. stream.Seek(0, SeekOrigin.Begin);
  165. BinaryWriter bw = new BinaryWriter(fs);
  166. bw.Write(bytes);
  167. bw.Close();
  168. fs.Close();
  169. return Ok(screenshot);
  170. }
  171. else
  172. {
  173. return Ok("链接为空");
  174. }
  175. }
  176. catch (Exception ex)
  177. {
  178. return Ok($"{ex.Message}{ex.StackTrace}");
  179. }
  180. }
  181. /// <summary>
  182. /// C#使用Puppeteer http://t.zoukankan.com/zhaotianff-p-13528507.html
  183. /// 文档https://learnku.com/docs/puppeteer/3.1.0/class-request/8559
  184. /// https://www.w3cschool.cn/puppeteer/puppeteer-gp1737se.html
  185. /// </summary>
  186. /// <param name="screenshot"></param>
  187. /// <returns></returns>
  188. [HttpGet("screenshot-png")]
  189. public async Task<IActionResult> ScreenshotPng([FromQuery] ScreenshotDto screenshot)
  190. {
  191. // 进入容器的命令 docker exec -it f9e27d175498 /bin/bash
  192. //依赖包 https://blog.csdn.net/weixin_45447477/article/details/115188938
  193. //sudo apt-get install libgdk-pixbuf2.0-0 libgdk-pixbuf-xlib-2.0-0 libdbusmenu-gtk3-4 libdbusmenu-glib4 libindicator3-7 ca-certificates fonts-liberation libappindicator3-1 libasound2 libatk-bridge2.0-0 libatk1.0-0 libc6 libcairo2 libcups2 libdbus-1-3 libexpat1 libfontconfig1 libgbm1 libgcc1 libglib2.0-0 libgtk-3-0 libnspr4 libnss3 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 libxcomposite1 libxcursor1 libxdamage1 libxext6 libxfixes3 libxi6 libxrandr2 libxrender1 libxss1 libxtst6 lsb-release wget xdg-utils -y
  194. //解决ubuntu18上使用puppeteer https://blog.csdn.net/qq_42414062/article/details/114539378
  195. //https://www.hardkoded.com/blog/running-puppeteer-sharp-azure-functions 使用。
  196. //string url = "https://teammodelos.blob.core.chinacloudapi.cn/0-public/pie-borderRadius.html";
  197. try
  198. {
  199. var bfOptions = new BrowserFetcherOptions();
  200. if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
  201. {
  202. string dir = "/app";
  203. if (!Directory.Exists(dir))
  204. {
  205. Directory.CreateDirectory(dir);
  206. }
  207. bfOptions.Path = dir;
  208. }
  209. var bf = new BrowserFetcher(bfOptions);
  210. var revisionInfo = bf.DownloadAsync(BrowserFetcher.DefaultChromiumRevision).Result;
  211. string BrowserExecutablePath = revisionInfo.ExecutablePath;
  212. var browser = await Puppeteer.LaunchAsync(new LaunchOptions
  213. {
  214. ExecutablePath = BrowserExecutablePath,
  215. Headless = true,
  216. Args = new string[] { "--no-sandbox", "--disable-setuid-sandbox" }
  217. });
  218. var page = await browser.NewPageAsync();
  219. bool fullPage = true;
  220. await page.SetViewportAsync(new ViewPortOptions
  221. {
  222. Width = screenshot.width,
  223. Height = screenshot.height
  224. }); ;
  225. fullPage = false;
  226. await page.GoToAsync(System.Web.HttpUtility.UrlDecode(screenshot.url));
  227. await Task.Delay(screenshot.delay);
  228. string base64 = await page.ScreenshotBase64Async(new ScreenshotOptions { FullPage = fullPage, BurstMode = true });
  229. //关闭浏览器
  230. await browser.CloseAsync();
  231. await browser.DisposeAsync();
  232. return Ok(new { url = base64, type = "data:image/png;base64," });
  233. }
  234. catch (Exception ex)
  235. {
  236. return BadRequest($"{ex.Message}\n{ex.StackTrace}");
  237. }
  238. }
  239. [HttpPost("screenshot-pdf")]
  240. public async Task<IActionResult> ScreenshotPdf(ScreenshotDto screenshot)
  241. {
  242. //W3C School教程 https://www.w3cschool.cn/puppeteer/puppeteer-rip537tj.html
  243. // 进入容器的命令 docker exec -it f9e27d175498 /bin/bash
  244. //依赖包 https://blog.csdn.net/weixin_45447477/article/details/115188938
  245. //sudo apt-get install libgdk-pixbuf2.0-0 libgdk-pixbuf-xlib-2.0-0 libdbusmenu-gtk3-4 libdbusmenu-glib4 libindicator3-7 ca-certificates fonts-liberation libappindicator3-1 libasound2 libatk-bridge2.0-0 libatk1.0-0 libc6 libcairo2 libcups2 libdbus-1-3 libexpat1 libfontconfig1 libgbm1 libgcc1 libglib2.0-0 libgtk-3-0 libnspr4 libnss3 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 libxcomposite1 libxcursor1 libxdamage1 libxext6 libxfixes3 libxi6 libxrandr2 libxrender1 libxss1 libxtst6 lsb-release wget xdg-utils -y
  246. //解决ubuntu18上使用puppeteer https://blog.csdn.net/qq_42414062/article/details/114539378
  247. //https://www.hardkoded.com/blog/running-puppeteer-sharp-azure-functions 使用。
  248. //string url = "https://teammodelos.blob.core.chinacloudapi.cn/0-public/pie-borderRadius.html";
  249. Browser browser = null;
  250. try
  251. {
  252. var bfOptions = new BrowserFetcherOptions();
  253. if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
  254. {
  255. string dir = "/app";
  256. if (!Directory.Exists(dir))
  257. {
  258. Directory.CreateDirectory(dir);
  259. }
  260. bfOptions.Path = dir;
  261. }
  262. var bf = new BrowserFetcher(bfOptions);
  263. var revisionInfo = bf.DownloadAsync(BrowserFetcher.DefaultChromiumRevision).Result;
  264. string BrowserExecutablePath = revisionInfo.ExecutablePath;
  265. browser = await Puppeteer.LaunchAsync(new LaunchOptions
  266. {
  267. ExecutablePath = BrowserExecutablePath,
  268. Headless = true,
  269. Args = new string[] { "--no-sandbox", "--disable-setuid-sandbox" }
  270. });
  271. double unitPX = 37.7813;
  272. ViewPortOptions viewPortOptions = new ViewPortOptions
  273. {
  274. // Width = (int)Math.Ceiling(unitPX * 21),
  275. // Height = (int)Math.Ceiling(unitPX * 29.7 * 1)
  276. };
  277. //ViewPortOptions viewPortOptions = new ViewPortOptions
  278. //{
  279. // Width = screenshot.width,
  280. // Height = screenshot.height
  281. //};
  282. PdfOptions pdfOptions = new PdfOptions { DisplayHeaderFooter = true, FooterTemplate = "", PreferCSSPageSize = true, Format = PaperFormat.A4 };
  283. //ScreenshotOptions screenshotOptions= new ScreenshotOptions { FullPage = fullPage, BurstMode = true };
  284. List<(string name, string url)> urls = new List<(string name, string url)>();
  285. if (screenshot.urls.Count <= screenshot.pagesize)
  286. {
  287. urls.AddRange(await PageToPdfStream(screenshot.urls, screenshot.fileNameKey, screenshot.cnt, screenshot.root, screenshot.env, browser, viewPortOptions, pdfOptions));
  288. }
  289. else
  290. {
  291. List<Task<List<(string name, string url)>>> tasks = new List<Task<List<(string name, string url)>>>();
  292. int pages = (screenshot.urls.Count + screenshot.pagesize) / screenshot.pagesize;
  293. for (int i = 0; i < pages; i++)
  294. {
  295. var lists = screenshot.urls.Skip((i) * screenshot.pagesize).Take(screenshot.pagesize).ToList();
  296. tasks.Add(PageToPdfStream(lists, screenshot.fileNameKey, screenshot.cnt, screenshot.root, screenshot.env, browser, viewPortOptions, pdfOptions));
  297. }
  298. var tsk = await Task.WhenAll(tasks);
  299. foreach (var ts in tsk)
  300. {
  301. urls.AddRange(ts);
  302. }
  303. }
  304. //browser.NewPageAsync();
  305. //foreach (var url in screenshot.urls) {
  306. // var page = await browser.NewPageAsync();
  307. // await page.SetViewportAsync(viewPortOptions);
  308. // string file = $"E://pdfs//{Guid.NewGuid().ToString()}.pdf";
  309. // var respons = await page.GoToAsync(System.Web.HttpUtility.UrlDecode(url), WaitUntilNavigation.Networkidle2);
  310. // if (respons.Ok)
  311. // {
  312. // await page.PdfAsync(file, pdfOptions);
  313. // // string base64 = await page.ScreenshotBase64Async(screenshotOptions);
  314. // }
  315. //}
  316. //关闭浏览器
  317. await browser.CloseAsync();
  318. await browser.DisposeAsync();
  319. return Ok(new { urls = urls.Select(z => z.url) });
  320. }
  321. catch (Exception ex)
  322. {
  323. StreamWriter file = new StreamWriter("erorr_log.txt", append: true);
  324. await file.WriteLineAsync($"{this.GetType().Name} {JsonSerializer.Serialize(screenshot)}-----{ex.Message}----{ex.StackTrace}");
  325. file.Close();
  326. return BadRequest($"{ex.Message}\n{ex.StackTrace}");
  327. }
  328. finally
  329. {
  330. if (browser != null && !browser.IsClosed)
  331. {
  332. await browser.CloseAsync();
  333. await browser.DisposeAsync();
  334. }
  335. }
  336. }
  337. private async Task<List<(string name, string url)>> PageToPdfStream(List<string> urls, string fileNameKey, string cnt, string root, string env, Browser browser, ViewPortOptions viewPortOptions, PdfOptions pdfOptions)
  338. {
  339. string name = env.Equals("release") ? "Default" : "Test";
  340. List<Task<Page>> pages = new List<Task<Page>>();
  341. urls.ForEach(x => {
  342. pages.Add(browser.NewPageAsync());
  343. });
  344. var page_tasks = await Task.WhenAll(pages);
  345. List<Task<Response>> responses = new List<Task<Response>>();
  346. page_tasks.ToList().ForEach(x => {
  347. x.SetViewportAsync(viewPortOptions);
  348. });
  349. for (int i = 0; i < urls.Count; i++)
  350. {
  351. responses.Add(page_tasks[i].GoToAsync(urls[i], 30000, new WaitUntilNavigation[] { WaitUntilNavigation.Networkidle2 }));
  352. }
  353. var responses_tasks = await Task.WhenAll(responses);
  354. //List<Task<Stream>> streams = new List<Task<Stream>>();
  355. List<Task> tasks = new List<Task>();
  356. List<Task<(string name, string url)>> uploads = new List<Task<(string name, string url)>>();
  357. foreach (var page_task in page_tasks)
  358. {
  359. string url = page_task.Url;
  360. string[] paths = HttpUtility.UrlDecode(url).Split("/");
  361. var reg = $"(?<=\\b{fileNameKey}=)[^&]*";
  362. Regex regex = new Regex(reg);
  363. string decode = HttpUtility.UrlDecode(url);
  364. Match match = Regex.Match(decode, reg);
  365. string id = "";
  366. while (match.Success)
  367. {
  368. id = id + $"{match.Value}";
  369. match = match.NextMatch();
  370. }
  371. //需要解析参数。paths[paths.Length-1]
  372. Stream stream = await page_task.PdfStreamAsync(pdfOptions);
  373. if (string.IsNullOrWhiteSpace(cnt))
  374. {
  375. uploads.Add(_azureStorage.GetBlobContainerClient("teammodelos", name).UploadFileByContainerBName(stream, root, $"{id}.pdf", true));
  376. }
  377. else
  378. {
  379. uploads.Add(_azureStorage.GetBlobContainerClient(cnt, name).UploadFileByContainerBName(stream, root, $"{id}.pdf", true));
  380. }
  381. }
  382. (string name, string url)[] uploadUrls = await Task.WhenAll(uploads);
  383. page_tasks.ToList().ForEach(x => {
  384. tasks.Add(x.DisposeAsync().AsTask());
  385. });
  386. await Task.WhenAll(tasks);
  387. return uploadUrls.ToList();
  388. }
  389. private async Task PageToPdf(List<string> urls, Browser browser, ViewPortOptions viewPortOptions, PdfOptions pdfOptions)
  390. {
  391. List<Task<Page>> pages = new List<Task<Page>>();
  392. urls.ForEach(x => {
  393. pages.Add(browser.NewPageAsync());
  394. });
  395. var page_tasks = await Task.WhenAll(pages);
  396. List<Task<Response>> responses = new List<Task<Response>>();
  397. page_tasks.ToList().ForEach(x => {
  398. x.SetViewportAsync(viewPortOptions);
  399. });
  400. for (int i = 0; i < urls.Count; i++)
  401. {
  402. responses.Add(page_tasks[i].GoToAsync(System.Web.HttpUtility.UrlDecode(urls[i]), WaitUntilNavigation.Networkidle2));
  403. }
  404. var responses_tasks = await Task.WhenAll(responses);
  405. List<Task> tasks = new List<Task>();
  406. page_tasks.ToList().ForEach(x => {
  407. string file = $"{Guid.NewGuid()}.pdf";
  408. tasks.Add(x.PdfAsync(file, pdfOptions));
  409. });
  410. await Task.WhenAll(tasks);
  411. tasks.Clear();
  412. page_tasks.ToList().ForEach(x => {
  413. tasks.Add(x.DisposeAsync().AsTask());
  414. });
  415. await Task.WhenAll(tasks);
  416. }
  417. }
  418. }