add term
| @@ -40,6 +40,5 @@ | ||||
| 		{{ range .AlternativeOutputFormats -}} | ||||
| 		{{ printf `<link rel="%s" type="%s" href="%s" title="%s" />` .Rel .MediaType.Type .Permalink $.Site.Title | safeHTML }} | ||||
| 		{{ end -}} | ||||
|  | ||||
| 		 | ||||
| 	</head> | ||||
|   | ||||
| @@ -12,6 +12,7 @@ | ||||
| 			--> | ||||
| 			<div class="logo">  | ||||
| 				{{ partial "svg.html" . }} | ||||
| 				{{ partial "term.html" . }} | ||||
| 			</div> | ||||
| 		</header> | ||||
|  | ||||
|   | ||||
| @@ -7,8 +7,3 @@ | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| <div id="aiterm" style="display:none;"> | ||||
| 	<iframe src="https://term.syui.ai" allowfullscreen frameborder="0" style="width:100%;height: 400px;" id="terminal-input"></iframe> | ||||
| </div> | ||||
|  | ||||
|   | ||||
							
								
								
									
										16
									
								
								layouts/partials/term.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | ||||
| <div id="aiterm" style="display:none;"> | ||||
| 	<link rel="stylesheet" href="/term/pkg/jquery.terminal/css/jquery.terminal.css" /> | ||||
| 	<link rel="stylesheet" href="/term/css/terminal.css" /> | ||||
| 	<script src="/term/pkg/jquery.ajax/jquery.min.js"></script> | ||||
| 	<script src="/term/pkg/axios/dist/axios.min.js"></script> | ||||
| 	<script src="/term/pkg/jquery.terminal/js/jquery.terminal.min.js"></script> | ||||
| 	<script src="/term/pkg/jquery.terminal/js/jquery.mousewheel-min.js"></script> | ||||
| 	<script src="/term/js/terminal.quick.js"></script> | ||||
| 	<script src="/term/js/terminal.js"></script> | ||||
| 	<app></app> | ||||
| 	<script src="/term/js/bundle.js"></script> | ||||
| 	<script src="/term/js/terminal-open.js"></script> | ||||
| 	<!-- | ||||
| 		<iframe src="https://term.syui.ai" allowfullscreen frameborder="0" style="width:100%;height: 400px;" id="terminal-input"></iframe> | ||||
| 	--> | ||||
| </div> | ||||
							
								
								
									
										9
									
								
								static/pkg/hotkeys-js/dist/terminal.js
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,17 +1,16 @@ | ||||
| hotkeys('ctrl+enter', function() { | ||||
| 	var ele = document.getElementById('aiterm'); | ||||
| 	var win = document.getElementById('window'); | ||||
| 	var svg = document.getElementById('aisvg'); | ||||
| 	var term = document.getElementById('terminal'); | ||||
| 	term.click(); | ||||
| 	//var par = document.getElementById('particles-js'); | ||||
| 	//par.style.display = 'block';  | ||||
| 	if (ele.style.display == 'none') { | ||||
| 		ele.style.display = 'block';  | ||||
| 		svg.style.display = 'none';  | ||||
| 		//par.style.display = 'none';  | ||||
| 		ele.focus(); | ||||
| 	} else { | ||||
| 		ele.style.display = 'none'; | ||||
| 		svg.style.display = 'block';  | ||||
| 		//par.style.display = 'block';  | ||||
| 		svg.focus(); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
|   | ||||
							
								
								
									
										507
									
								
								static/term/css/terminal.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,507 @@ | ||||
| #window { | ||||
| 	margin-left: auto; | ||||
| 	text-align: left; | ||||
| 	margin-right: auto; | ||||
| 	background: #343434; | ||||
| 	border-radius: 7px; | ||||
| 	overflow: hidden; | ||||
| 	width: 100%; | ||||
| 	height: 400px; | ||||
| } | ||||
|  | ||||
| #topbar { | ||||
| 	width: 100%; | ||||
| 	height: 21px; | ||||
| 	font-size: 16px; | ||||
| 	font-family: "Myriad Pro", sans-serif; | ||||
| 	text-shadow: 0px 1px 0px rgba(255, 255, 255, 0.25); | ||||
| 	-webkit-box-shadow: 0 1px 0 rgba(0, 0, 0, 0.5); | ||||
| 	-moz-box-shadow: 0 1px 0 rgba(0, 0, 0, 0.5); | ||||
| } | ||||
|  | ||||
| #topbar ul li { | ||||
| 	float: left; | ||||
| 	padding: 0 10px; | ||||
| 	height: 21px; | ||||
| 	line-height: 24px; | ||||
| } | ||||
| #topbar ul li:first-child { | ||||
| 	font-size: 20px; | ||||
| 	line-height: 26px; | ||||
| 	margin-left: 5px; | ||||
| } | ||||
| #topbar ul li:nth-child(2) { | ||||
| 	font-family: "Myriad-Semi", sans-serif; | ||||
| } | ||||
| #topbar ul li:active { | ||||
| 	background-image: -webkit-gradient(linear, left bottom, left top, color-stop(1, #4a82ff), color-stop(0, #0052fc)); | ||||
| 	color: #fff; | ||||
| 	text-shadow: none; | ||||
| } | ||||
| #toolbar { | ||||
| 	width: 100%; | ||||
| 	height: 25px; | ||||
| 	background: #ccc; | ||||
| } | ||||
| #toolbar .top { | ||||
| 	float: left; | ||||
| 	width: 100%; | ||||
| 	height: 23px; | ||||
| } | ||||
| #toolbar .bottom { | ||||
| 	float: left; | ||||
| 	width: 100%; | ||||
| 	height: 30px; | ||||
| } | ||||
| #toolbar #lights { | ||||
| 	float: left; | ||||
| 	position: relative; | ||||
| 	top: 6px; | ||||
| 	left: 7px; | ||||
| } | ||||
| .light { | ||||
| 	float: left; | ||||
| 	width: 14px; | ||||
| 	height: 14px; | ||||
| 	border-radius: 14px; | ||||
| 	-webkit-box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.5), 0px 0px 3px #000 inset; | ||||
| 	-moz-box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.5), 0px 0px 3px #000 inset; | ||||
| 	box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.5), 0px 0px 3px #000 inset; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| #lights:hover .glyph { | ||||
| 	opacity: 1; | ||||
| 	cursor: default; | ||||
| } | ||||
| .light .shine { | ||||
| 	width: 4px; | ||||
| 	height: 3px; | ||||
| 	border-radius: 10px; | ||||
| 	background: -moz-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%); | ||||
| 	background-image: -webkit-gradient(radial, center center, 0px, center center, 100%, color-stop(0%, rgba(255, 255, 255, 1)), color-stop(100%, rgba(255, 255, 255, 0))); | ||||
| 	background: -webkit-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%); | ||||
| 	background: -o-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%); | ||||
| 	background: -ms-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%); | ||||
| 	background: radial-gradient(center, ellipse cover, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%); | ||||
| } | ||||
| .light .glow { | ||||
| 	width: 14px; | ||||
| 	height: 8px; | ||||
| 	background-image: -webkit-gradient(radial, center bottom, 0, center center, 5, from(rgba(255, 255, 255, 0.75)), to(rgba(255, 255, 255, 0))); | ||||
| 	background: 0px 0px -moz-radial-gradient(bottom, cover, rgba(255, 255, 255, 0.70) 0%, rgba(255, 255, 255, 0) 80%); | ||||
| } | ||||
| .red { | ||||
| 	background: #f41b16; | ||||
| 	background: -moz-linear-gradient(top, #f41b16 0%, #fc7471 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #f41b16), color-stop(100%, #fc7471)); | ||||
| 	background: -webkit-linear-gradient(top, #f41b16 0%, #fc7471 100%); | ||||
| 	background: -o-linear-gradient(top, #f41b16 0%, #fc7471 100%); | ||||
| 	background: -ms-linear-gradient(top, #f41b16 0%, #fc7471 100%); | ||||
| 	background: linear-gradient(top, #f41b16 0%, #fc7471 100%); | ||||
| } | ||||
| .red:active { | ||||
| 	background: #972f2e; | ||||
| 	background: -moz-linear-gradient(top, #972f2e 0%, #fc7471 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #972f2e), color-stop(100%, #fc7471)); | ||||
| 	background: -webkit-linear-gradient(top, #972f2e 0%, #fc7471 100%); | ||||
| 	background: -o-linear-gradient(top, #972f2e 0%, #fc7471 100%); | ||||
| 	background: -ms-linear-gradient(top, #972f2e 0%, #fc7471 100%); | ||||
| 	background: linear-gradient(top, #972f2e 0%, #fc7471 100%); | ||||
| } | ||||
| .red .shine { | ||||
| 	position: relative; | ||||
| 	top: -21px; | ||||
| 	left: 5px; | ||||
| } | ||||
| .red .glow { | ||||
| 	position: relative; | ||||
| 	top: -20px; | ||||
| } | ||||
| .red .glyph { | ||||
| 	position: relative; | ||||
| 	top: -5px; | ||||
| 	left: 3px; | ||||
| 	font-size: 14px; | ||||
| 	font-weight: 400; | ||||
| 	color: #9b3a36; | ||||
| 	z-index: 50; | ||||
| 	opacity: 0; | ||||
| } | ||||
| .yellow { | ||||
| 	background: #f4a316; | ||||
| 	background: -moz-linear-gradient(left, #f4a316 0%, #fcc371 100%); | ||||
| 	background: -webkit-gradient(linear, left top, right top, color-stop(0%, #f4a316), color-stop(100%, #fcc371)); | ||||
| 	background: -webkit-linear-gradient(left, #f4a316 0%, #fcc371 100%); | ||||
| 	background: -o-linear-gradient(left, #f4a316 0%, #fcc371 100%); | ||||
| 	background: -ms-linear-gradient(left, #f4a316 0%, #fcc371 100%); | ||||
| 	background: linear-gradient(left, #f4a316 0%, #fcc371 100%); | ||||
| 	margin: 0px 7px; | ||||
| } | ||||
| .yellow:active { | ||||
| 	background: #ae4f1e; | ||||
| 	background: -moz-linear-gradient(top, #ae4f1e 0%, #fcc371 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #ae4f1e), color-stop(100%, #fcc371)); | ||||
| 	background: -webkit-linear-gradient(top, #ae4f1e 0%, #fcc371 100%); | ||||
| 	background: -o-linear-gradient(top, #ae4f1e 0%, #fcc371 100%); | ||||
| 	background: -ms-linear-gradient(top, #ae4f1e 0%, #fcc371 100%); | ||||
| 	background: linear-gradient(top, #ae4f1e 0%, #fcc371 100%); | ||||
| } | ||||
| .yellow .shine { | ||||
| 	position: relative; | ||||
| 	top: -21px; | ||||
| 	left: 5px; | ||||
| } | ||||
| .yellow .glow { | ||||
| 	position: relative; | ||||
| 	top: -20px; | ||||
| } | ||||
| .yellow .glyph { | ||||
| 	position: relative; | ||||
| 	top: -5px; | ||||
| 	left: 3px; | ||||
| 	font-size: 24px; | ||||
| 	color: #854322; | ||||
| 	z-index: 50; | ||||
| 	opacity: 0; | ||||
| 	-webkit-transform: scaleY(1.5) scaleX(1.3); | ||||
| } | ||||
| .green { | ||||
| 	background: #4cae2e; | ||||
| 	background: -moz-linear-gradient(top, #4cae2e 0%, #dafc71 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #4cae2e), color-stop(100%, #dafc71)); | ||||
| 	background: -webkit-linear-gradient(top, #4cae2e 0%, #dafc71 100%); | ||||
| 	background: -o-linear-gradient(top, #4cae2e 0%, #dafc71 100%); | ||||
| 	background: -ms-linear-gradient(top, #4cae2e 0%, #dafc71 100%); | ||||
| 	background: linear-gradient(top, #4cae2e 0%, #dafc71 100%); | ||||
| } | ||||
| .green:active { | ||||
| 	background: #48752b; | ||||
| 	background: -moz-linear-gradient(top, #48752b 0%, #dafc71 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #48752b), color-stop(100%, #dafc71)); | ||||
| 	background: -webkit-linear-gradient(top, #48752b 0%, #dafc71 100%); | ||||
| 	background: -o-linear-gradient(top, #48752b 0%, #dafc71 100%); | ||||
| 	background: -ms-linear-gradient(top, #48752b 0%, #dafc71 100%); | ||||
| 	background: linear-gradient(top, #48752b 0%, #dafc71 100%); | ||||
| } | ||||
| .green .shine { | ||||
| 	position: relative; | ||||
| 	top: -20px; | ||||
| 	left: 5px; | ||||
| } | ||||
| .green .glow { | ||||
| 	position: relative; | ||||
| 	top: -20px; | ||||
| } | ||||
| .green .glyph { | ||||
| 	position: relative; | ||||
| 	top: -5px; | ||||
| 	left: 3px; | ||||
| 	font-size: 14px; | ||||
| 	font-weight: bold; | ||||
| 	color: #25571d; | ||||
| 	z-index: 50; | ||||
| 	opacity: 0; | ||||
| } | ||||
| @-moz-document url-prefix() { | ||||
| 	.red .glyph { | ||||
| 		position: relative; | ||||
| 		top: -5px; | ||||
| 	} | ||||
| 	.yellow .glyph { | ||||
| 		top: -4px; | ||||
| 		left: 3px; | ||||
| 	} | ||||
| 	.green .glyph { | ||||
| 		position: relative; | ||||
| 		top: -4px; | ||||
| 	} | ||||
| } | ||||
| #title { | ||||
| 	float: left; | ||||
| 	position: relative; | ||||
| 	top: 6px; | ||||
| 	width: 40%; | ||||
| 	left: 45%; | ||||
| 	font-family: "Myriad Pro", sans-serif; | ||||
| 	font-size: 16px; | ||||
| 	line-height: 14px; | ||||
| 	font-weight: bold; | ||||
| } | ||||
| .folder { | ||||
| 	float: left; | ||||
| 	margin-right: 5px; | ||||
| } | ||||
| .folder .tab { | ||||
| 	width: 4px; | ||||
| 	height: 2px; | ||||
| 	background: #a4c5da; | ||||
| 	border: 1px solid #728ea3; | ||||
| 	border-bottom: none; | ||||
| 	border-radius: 2px 2px 0px 0px; | ||||
| 	-webkit-box-shadow: 0px -1px 0px #99b5c7 inset; | ||||
| 	margin-left: 1px; | ||||
| 	z-index: 5000; | ||||
| 	margin-bottom: -1px; | ||||
| } | ||||
| .folder .body { | ||||
| 	width: 14px; | ||||
| 	height: 10px; | ||||
| 	border: 1px solid #6e8ba1; | ||||
| 	background: #b8cfe0; | ||||
| 	background: -moz-linear-gradient(top, #b8cfe0 0%, #86adc8 100%); | ||||
| 	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #b8cfe0), color-stop(100%, #86adc8)); | ||||
| 	background: -webkit-linear-gradient(top, #b8cfe0 0%, #86adc8 100%); | ||||
| 	background: -o-linear-gradient(top, #b8cfe0 0%, #86adc8 100%); | ||||
| 	background: -ms-linear-gradient(top, #b8cfe0 0%, #86adc8 100%); | ||||
| 	background: linear-gradient(top, #b8cfe0 0%, #86adc8 100%); | ||||
| 	z-index: -50; | ||||
| 	-webkit-box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.25) inset, 0px 1px 0px rgba(0, 0, 0, 0.2); | ||||
| 	-moz-box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.25) inset, 0px 1px 0px rgba(0, 0, 0, 0.2); | ||||
| } | ||||
| #nav { | ||||
| 	margin: 1px 8px; | ||||
| 	float: left; | ||||
| } | ||||
| #view { | ||||
| 	margin: 2px 0 0 110px; | ||||
| 	display: inline-block; | ||||
| } | ||||
| .control_box { | ||||
| 	height: 20px; | ||||
| 	border-radius: 3px; | ||||
| 	border: 1px solid #555; | ||||
| 	background-image: -webkit-gradient(linear, left bottom, left top, color-stop(1, #fefefe), color-stop(0, #b8b8b8)); | ||||
| 	box-shadow: 0px 1px 0px rgba(255, 255, 255, 0.25); | ||||
| } | ||||
| .control_box .control { | ||||
| 	height: 20px; | ||||
| 	border-right: 1px solid #2e2e2e; | ||||
| 	float: left; | ||||
| 	text-align: center; | ||||
| 	width: 27px; | ||||
| } | ||||
| .control:last-child { | ||||
| 	border-right: 0px solid !important; | ||||
| } | ||||
| .control:active { | ||||
| 	background: #b0afb0; | ||||
| 	-webkit-box-shadow: 0px 0px 4px #000 inset; | ||||
| } | ||||
| .active { | ||||
| 	background: #707070 !important; | ||||
| 	-webkit-box-shadow: 0px 3px 4px rgba(0, 0, 0, 0.6) inset !important; | ||||
| } | ||||
| #body { | ||||
| 	font-family: Andale Mono, monospace; | ||||
| 	line-height: 1em; | ||||
| 	font-size: 13px; | ||||
| 	float: left; | ||||
| 	width: 100%; | ||||
| 	background: #002b36; | ||||
| 	padding: 10px; | ||||
| 	line-height: 1.5em; | ||||
| } | ||||
| #body p { | ||||
| 	color: #63de00!important; | ||||
| } | ||||
| @keyframes blink { | ||||
| 	0% { | ||||
| 		background: rgba(99, 222, 0, 100); | ||||
| 	} | ||||
| 	100% { | ||||
| 		background: rgba(99, 222, 0, 0); | ||||
| 	} | ||||
| } | ||||
| @-webkit-keyframes blink { | ||||
| 	0% { | ||||
| 		background: rgba(99, 222, 0, 100); | ||||
| 	} | ||||
| 	100% { | ||||
| 		background: rgba(99, 222, 0, 0); | ||||
| 	} | ||||
| } | ||||
| @-moz-keyframes blink { | ||||
| 	0% { | ||||
| 		background: rgba(99, 222, 0, 100); | ||||
| 	} | ||||
| 	100% { | ||||
| 		background: rgba(99, 222, 0, 0); | ||||
| 	} | ||||
| } | ||||
| .cursor { | ||||
| 	width: 10px; | ||||
| 	margin-left: 0px; | ||||
| 	color: #fff; | ||||
| } | ||||
| #body p::-webkit-selection { | ||||
| 	background: #0b209e; | ||||
| } | ||||
| #body p::selection { | ||||
| 	background: #0b209e; | ||||
| } | ||||
| #body p::-moz-selection { | ||||
| 	background: #0b209e; | ||||
| } | ||||
| #body p { | ||||
| 	margin-top: 5px; | ||||
| 	margin-bottom: 5px; | ||||
| 	font-size: 13px; | ||||
| } | ||||
| #content { | ||||
| 	float: left; | ||||
| 	margin-top: 1px; | ||||
| } | ||||
| #foot { | ||||
| 	height: 23px; | ||||
| 	width: 100%; | ||||
| 	float: left; | ||||
| 	background-image: -webkit-gradient(linear, left bottom, left top, color-stop(1, #cbcbcb), color-stop(0, #a7a7a7)); | ||||
| 	border-top: 1px solid #515151; | ||||
| 	border-radius: 0 0 5px 5px; | ||||
| } | ||||
| #foot .handle { | ||||
| 	width: 11px; | ||||
| 	height: 11px; | ||||
| 	float: right; | ||||
| 	margin: 6px; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| .handle .grip { | ||||
| 	-webkit-transform: rotate(45deg) scaley(3); | ||||
| 	margin: 2px 0 0 2px; | ||||
| 	color: #646464; | ||||
| 	text-shadow: 1px 1px 0 #c6c6c6; | ||||
| 	font-size: 14px; | ||||
| } | ||||
| .icon .frame { | ||||
| 	width: 82px; | ||||
| 	height: 82px; | ||||
| 	border-radius: 5px; | ||||
| 	border: 2px solid transparent; | ||||
| } | ||||
| .icon .name { | ||||
| 	color: #000; | ||||
| 	padding-top: 3px; | ||||
| 	border-radius: 15px; | ||||
| 	width: 55px; | ||||
| 	margin: 5px 0 0 15px; | ||||
| } | ||||
| .icon .folder { | ||||
| 	margin: 15px 0 0 6px; | ||||
| } | ||||
| #icon-github { | ||||
| 	text-align: -999px; | ||||
| 	font-size: 1px; | ||||
| 	display: block; | ||||
| 	width: 156px; | ||||
| 	height: 133px; | ||||
| 	background-image: url(../img/sprite.png); | ||||
| 	background-position: 0 133px; | ||||
| } | ||||
| #icon-github:hover { | ||||
| 	background-position: 0 0px; | ||||
| } | ||||
| #icon-rubygems { | ||||
| 	text-align: -999px; | ||||
| 	font-size: 1px; | ||||
| 	display: block; | ||||
| 	width: 156px; | ||||
| 	height: 133px; | ||||
| 	background-image: url(../img/sprite.png); | ||||
| 	background-position: 158px 133px; | ||||
| } | ||||
| #icon-rubygems:hover { | ||||
| 	background-position: 158px 0px; | ||||
| } | ||||
| .source-urls { | ||||
| 	margin-top: 40px; | ||||
| 	margin-bottom: 40px; | ||||
| } | ||||
| .no-margin-bot { | ||||
| 	color: #337AB7; | ||||
| } | ||||
| span { | ||||
| 	font-size: 15px; | ||||
| } | ||||
| #particles-js { | ||||
| 	position: absolute; | ||||
| 	width: 98%; | ||||
| } | ||||
| #particles-js-no { | ||||
| 	position: absolute; | ||||
| 	width: 98%; | ||||
| } | ||||
|  | ||||
| #icon-github:hover, | ||||
| i.icon.ion-close-circled:hover, | ||||
| i.icon.ion-minus-circled:hover, | ||||
| i.icon.ion-plus-circled:hover { | ||||
| 	color: rgba(0, 0, 0, .5); | ||||
| } | ||||
| i.icon.ion-close-circled { | ||||
| 	color: rgba(212, 42, 38, 0.83); | ||||
| 	text-shadow: 0 0 1px rgba(187, 187, 187, 0.56); | ||||
| 	padding-top: 3px; | ||||
| } | ||||
| i.icon.ion-minus-circled { | ||||
| 	color: rgb(160, 165, 34); | ||||
| 	text-shadow: 0 0 1px rgba(187, 187, 187, 0.56); | ||||
| 	padding-top: 3px; | ||||
| } | ||||
| i.icon.ion-plus-circled { | ||||
| 	color: rgb(82, 183, 51); | ||||
| 	text-shadow: 0 0 1px rgba(187, 187, 187, 0.56); | ||||
| 	padding-top: 3px; | ||||
| } | ||||
| .icon { | ||||
| 	margin: 7px; | ||||
| 	float: left; | ||||
| 	font-size: 20px; | ||||
| } | ||||
| #title-left { | ||||
| 	position: relative; | ||||
| 	top: -8px; | ||||
| 	font-family: "Myriad Pro", sans-serif; | ||||
| 	font-size: 14px; | ||||
| 	left: 49%; | ||||
| } | ||||
| #terminal-origin { | ||||
| 	font-family: monospace; | ||||
| } | ||||
| @media all and (-webkit-min-device-pixel-ratio: 0) and (min-resolution: .001dpcm) { | ||||
| 	#title-left { | ||||
| 		top: -13px; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| home-layout { | ||||
| 	background-color: #f1f1f1; | ||||
| } | ||||
|  | ||||
| .terminal, .cmd { | ||||
| 	background-color: #343434; | ||||
| } | ||||
|  | ||||
| @media screen and (max-width:1000px){i.icon.ion-minus-circled{display:none} | ||||
| 	#window{width:100%;} | ||||
| } | ||||
| @media screen and (max-width:800px){body{padding-top:0px;} | ||||
| 	#title-left{left:50%;} | ||||
| 	#window{width:100%;} | ||||
| 	.container{padding:0px;} | ||||
| 	/*article{padding:10px 20px;}*/ | ||||
| 	i.icon.ion-minus-circled{display:none} | ||||
| } | ||||
| @media screen and (max-width:800px){#title-left{left:48%;} | ||||
| } | ||||
| @media screen and (max-width:500px){#title-left{left:40%;} | ||||
| } | ||||
| @media screen and (max-width:400px){body{padding-top:0px;width:auto;} | ||||
| 	#title-left{left:50%;} | ||||
| 	/*article{padding:10px 10px;}*/ | ||||
| 	#particles-js{width:auto;} | ||||
| 	#particles-js-no{width:auto;} | ||||
| 	i.icon.ion-minus-circled{display:none} | ||||
| 	i.icon.ion-plus-circled{display:none} | ||||
| } | ||||
|  | ||||
							
								
								
									
										1
									
								
								static/term/css/terminal.mobile.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/ai.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 89 KiB | 
							
								
								
									
										1882
									
								
								static/term/icon/ai.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 50 KiB | 
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/apple-touch-icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 89 KiB | 
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/avatar.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 557 KiB | 
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/player.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 54 KiB | 
							
								
								
									
										79
									
								
								static/term/icon/player.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,79 @@ | ||||
| <?xml version="1.0" standalone="no"?> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN" | ||||
|  "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd"> | ||||
| <svg version="1.0" xmlns="http://www.w3.org/2000/svg" | ||||
|  width="645.000000pt" height="645.000000pt" viewBox="0 0 645.000000 645.000000" | ||||
|  preserveAspectRatio="xMidYMid meet"> | ||||
|  | ||||
| <g transform="translate(0.000000,645.000000) scale(0.100000,-0.100000)" | ||||
| fill="#000000" stroke="none"> | ||||
| <path d="M2926 6399 c-400 -40 -754 -142 -1111 -319 -370 -183 -627 -373 -927 | ||||
| -688 -334 -350 -608 -843 -740 -1332 -21 -78 -26 -96 -51 -220 -79 -384 -78 | ||||
| -865 2 -1255 194 -948 827 -1772 1698 -2209 291 -145 607 -246 943 -301 122 | ||||
| -20 533 -38 670 -30 657 40 1280 281 1805 699 128 101 378 352 482 481 306 | ||||
| 382 518 817 625 1280 57 249 78 434 78 715 0 315 -24 511 -100 806 -111 432 | ||||
| -311 832 -598 1194 -98 123 -370 395 -492 491 -437 345 -923 564 -1457 655 | ||||
| -81 14 -184 27 -228 30 -152 10 -510 12 -599 3z m624 -130 c408 -46 840 -185 | ||||
| 1170 -376 281 -163 463 -303 688 -531 158 -161 350 -411 456 -595 107 -187 | ||||
| 239 -493 286 -662 95 -343 122 -541 122 -880 -1 -286 -19 -452 -82 -725 -74 | ||||
| -321 -246 -710 -441 -995 -496 -726 -1261 -1198 -2134 -1316 -204 -27 -592 | ||||
| -27 -790 0 -569 80 -1070 293 -1512 643 -174 138 -366 331 -498 502 -16 22 | ||||
| -39 51 -51 65 -71 88 -233 356 -293 486 -82 174 -99 218 -152 378 -107 326 | ||||
| -153 616 -153 967 0 324 28 524 124 875 82 304 286 701 507 989 86 112 249 | ||||
| 291 342 374 47 42 88 79 91 82 20 22 172 139 250 192 380 262 816 437 1260 | ||||
| 507 36 5 85 13 110 17 67 11 379 21 500 17 58 -2 148 -8 200 -14z"/> | ||||
| <path d="M2685 5968 c-335 -80 -520 -146 -748 -266 -203 -107 -416 -252 -561 | ||||
| -381 -115 -101 -291 -289 -363 -387 -91 -123 -121 -168 -186 -276 -119 -200 | ||||
| -193 -363 -272 -603 -96 -291 -144 -724 -116 -1050 14 -158 24 -208 53 -247 | ||||
| 17 -24 26 -28 70 -28 29 0 65 4 82 9 38 12 156 42 281 73 98 23 91 14 -50 -60 | ||||
| -134 -70 -236 -130 -302 -177 -20 -14 -23 -25 -23 -83 0 -75 8 -109 52 -232 | ||||
| 278 -770 926 -1405 1708 -1672 353 -120 734 -174 1067 -149 166 12 205 17 373 | ||||
| 48 492 90 1002 349 1364 693 277 262 443 485 605 811 133 267 180 426 157 529 | ||||
| -12 52 -16 57 -77 98 -35 23 -118 72 -184 108 -66 36 -124 70 -130 76 -13 13 | ||||
| 49 6 135 -16 36 -9 83 -21 105 -27 22 -6 54 -14 70 -20 17 -5 52 -9 78 -10 50 | ||||
| -2 78 18 99 71 35 90 47 471 23 700 -22 205 -37 284 -104 535 -5 17 -22 68 | ||||
| -39 115 -95 268 -228 518 -390 735 -279 372 -577 626 -967 825 -234 120 -460 | ||||
| 200 -706 250 -102 21 -109 21 -163 6 -64 -19 -63 -17 -132 -206 -20 -52 -49 | ||||
| -131 -66 -175 -16 -44 -39 -104 -49 -132 -10 -29 -28 -77 -38 -105 -11 -29 | ||||
| -41 -113 -68 -187 -26 -74 -53 -136 -59 -138 -6 -2 -22 31 -37 74 -14 43 -40 | ||||
| 116 -57 163 -18 47 -36 96 -40 110 -5 14 -24 68 -43 120 -155 422 -157 427 | ||||
| -217 468 -35 23 -69 26 -135 10z m-630 -2381 c147 -128 374 -251 533 -290 l92 | ||||
| -22 0 -111 c0 -257 -69 -594 -184 -895 -73 -194 -68 -186 -137 -194 -59 -7 | ||||
| -60 -7 -102 32 -29 27 -56 42 -83 46 -34 6 -52 20 -129 103 -50 53 -102 110 | ||||
| -117 127 -16 16 -28 31 -28 34 0 3 14 21 31 41 82 94 103 232 54 367 -14 39 | ||||
| -39 97 -55 130 -47 98 -180 266 -247 313 -33 23 -43 42 -22 42 6 0 23 -8 37 | ||||
| -19 142 -100 442 -189 497 -148 20 15 20 16 -6 44 -93 103 -304 490 -285 522 | ||||
| 3 5 23 -7 43 -26 21 -19 69 -62 108 -96z m1006 -800 c236 -425 273 -488 343 | ||||
| -572 117 -141 217 -197 376 -211 57 -5 100 -15 131 -30 38 -19 63 -24 128 -24 | ||||
| 45 0 81 -4 81 -9 0 -19 -90 -130 -110 -136 -29 -9 -112 3 -190 28 -61 20 -136 | ||||
| 24 -264 14 -25 -2 -31 3 -37 28 -5 16 -16 36 -26 42 -25 19 -102 16 -169 -8 | ||||
| -32 -11 -66 -22 -75 -25 -12 -4 -25 10 -47 49 -70 124 -117 152 -206 122 -26 | ||||
| -9 -50 -14 -53 -11 -3 3 2 38 11 78 30 129 40 323 29 558 -5 118 -7 218 -5 | ||||
| 223 7 15 21 -5 83 -116z m561 -275 c154 -140 331 -284 465 -380 68 -48 123 | ||||
| -92 123 -98 0 -5 -13 -23 -30 -39 l-29 -28 -78 22 c-85 25 -157 61 -187 95 | ||||
| -136 153 -359 476 -328 476 6 0 34 -22 64 -48z m818 -585 c182 -77 348 -75 | ||||
| 493 6 l38 21 23 -25 c33 -33 118 -185 157 -278 17 -42 39 -115 47 -164 l16 | ||||
| -88 -85 -85 c-327 -332 -761 -584 -1214 -704 -371 -98 -793 -116 -1160 -50 | ||||
| -33 6 -75 14 -93 17 -65 12 -53 29 58 84 261 128 315 151 495 211 227 74 392 | ||||
| 116 584 149 57 10 83 20 108 42 18 16 33 33 33 38 0 31 -298 -18 -520 -86 | ||||
| -239 -72 -367 -119 -500 -181 -55 -26 -103 -44 -107 -40 -14 14 95 397 154 | ||||
| 541 76 186 79 190 130 223 80 51 283 149 365 176 154 50 250 53 367 11 54 -19 | ||||
| 221 -20 308 -1 67 14 128 45 122 61 -2 6 -32 9 -66 8 -37 0 -63 3 -63 9 0 6 | ||||
| 14 26 30 45 17 18 44 61 60 93 l30 60 78 -39 c42 -22 93 -46 112 -54z m-1357 | ||||
| 38 c14 -14 36 -49 50 -78 40 -86 52 -89 177 -53 30 9 72 16 93 16 73 0 42 -30 | ||||
| -109 -105 -78 -38 -171 -90 -207 -114 -62 -40 -70 -50 -124 -150 -31 -58 -64 | ||||
| -119 -73 -136 -9 -16 -27 -68 -39 -115 -25 -98 -43 -130 -72 -130 -32 0 -140 | ||||
| -59 -177 -97 -47 -49 -84 -135 -80 -187 5 -63 26 -51 80 44 47 83 97 138 150 | ||||
| 166 15 8 29 10 32 5 3 -4 -16 -77 -42 -162 l-47 -153 -83 -48 c-46 -26 -94 | ||||
| -48 -107 -48 -27 0 -164 45 -177 58 -5 5 15 45 50 98 74 112 188 316 291 522 | ||||
| 136 272 156 321 216 534 l28 98 60 30 c33 16 66 30 72 30 7 0 24 -11 38 -25z | ||||
| m-791 -197 c2 -6 -17 -75 -44 -152 -41 -123 -66 -214 -99 -371 -8 -40 -52 | ||||
| -319 -64 -407 -5 -38 -10 -48 -25 -48 -20 0 -145 68 -197 106 -39 29 -49 78 | ||||
| -36 176 l8 66 109 101 c60 56 115 101 122 101 17 0 48 46 39 59 -3 6 -20 14 | ||||
| -38 19 -27 7 -33 13 -35 42 -2 18 -1 48 3 66 5 30 10 33 50 39 58 9 80 35 78 | ||||
| 90 -2 47 22 78 82 109 36 19 42 20 47 4z"/> | ||||
| <path d="M2350 2864 c-83 -62 -149 -98 -197 -108 -42 -9 -60 -31 -43 -51 7 -8 | ||||
| 23 -15 36 -15 25 0 33 -27 15 -45 -6 -6 -13 -29 -17 -52 -12 -75 24 -47 79 63 | ||||
| 19 38 38 59 73 79 89 52 176 148 159 175 -11 18 -29 10 -105 -46z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 5.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/term.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 74 KiB | 
							
								
								
									
										30
									
								
								static/term/icon/term.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,30 @@ | ||||
| <?xml version="1.0" standalone="no"?> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN" | ||||
|  "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd"> | ||||
| <svg version="1.0" xmlns="http://www.w3.org/2000/svg" | ||||
|  width="1417.000000pt" height="1417.000000pt" viewBox="0 0 1417.000000 1417.000000" | ||||
|  preserveAspectRatio="xMidYMid meet"> | ||||
|  | ||||
| <g transform="translate(0.000000,1417.000000) scale(0.100000,-0.100000)" | ||||
| fill="#000000" stroke="none"> | ||||
| <path d="M0 7085 l0 -7085 7085 0 7085 0 0 7085 0 7085 -7085 0 -7085 0 0 | ||||
| -7085z m2783 4626 c102 -58 632 -354 675 -376 18 -10 40 -8 115 8 78 17 129 | ||||
| 21 327 21 209 1 247 -2 345 -22 470 -100 868 -365 1128 -754 76 -115 153 -267 | ||||
| 197 -394 l32 -91 180 -113 c213 -133 528 -325 580 -354 43 -24 47 -40 16 -59 | ||||
| -13 -8 -75 -46 -138 -84 -63 -39 -233 -143 -376 -232 l-261 -161 -33 -91 | ||||
| c-145 -406 -431 -747 -799 -953 -202 -114 -430 -189 -651 -216 -173 -21 -479 | ||||
| -5 -599 31 -37 11 -41 9 -257 -111 -120 -67 -266 -148 -324 -180 -58 -32 -141 | ||||
| -78 -185 -103 -44 -25 -83 -43 -86 -39 -4 4 -16 219 -29 477 -12 259 -25 475 | ||||
| -28 480 -3 6 -27 35 -53 65 -26 30 -77 100 -114 155 -347 520 -390 1212 -111 | ||||
| 1783 50 102 176 292 242 365 24 27 44 57 44 66 -1 47 42 918 45 929 3 6 7 12 | ||||
| 10 12 3 0 52 -26 108 -59z m6466 -3618 c31 -84 68 -239 82 -340 17 -128 7 | ||||
| -403 -20 -526 -23 -106 -75 -273 -93 -294 -8 -11 -292 -13 -1484 -13 l-1473 0 | ||||
| -12 21 c-23 43 -68 192 -91 297 -19 92 -23 136 -22 312 0 174 3 220 22 308 25 | ||||
| 115 69 261 84 280 8 9 320 12 1498 12 l1488 0 21 -57z"/> | ||||
| <path d="M3799 10649 c-156 -16 -335 -83 -479 -177 -72 -48 -215 -191 -267 | ||||
| -267 -123 -182 -183 -381 -183 -606 0 -403 219 -761 576 -942 69 -35 214 -82 | ||||
| 299 -98 87 -16 269 -16 355 0 344 64 637 285 776 586 74 160 97 265 98 450 0 | ||||
| 191 -30 326 -109 485 -79 158 -202 296 -352 397 -146 97 -298 153 -475 172 | ||||
| -107 12 -128 12 -239 0z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								static/term/icon/terminal.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 107 KiB | 
							
								
								
									
										32
									
								
								static/term/icon/terminal.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,32 @@ | ||||
| <?xml version="1.0" standalone="no"?> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN" | ||||
|  "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd"> | ||||
| <svg version="1.0" xmlns="http://www.w3.org/2000/svg" | ||||
|  width="1417.000000pt" height="1417.000000pt" viewBox="0 0 1417.000000 1417.000000" | ||||
|  preserveAspectRatio="xMidYMid meet"> | ||||
|  | ||||
| <g transform="translate(0.000000,1417.000000) scale(0.100000,-0.100000)" | ||||
| fill="#000000" stroke="none"> | ||||
| <path d="M0 7085 l0 -7085 7085 0 7085 0 0 7085 0 7085 -7085 0 -7085 0 0 | ||||
| -7085z m2759 3496 c25 -16 134 -77 241 -137 234 -129 545 -302 732 -407 85 | ||||
| -48 147 -77 166 -77 16 0 72 9 123 19 164 34 288 44 509 44 233 0 375 -14 565 | ||||
| -58 474 -107 899 -337 1249 -675 308 -298 520 -635 673 -1074 l28 -79 95 -60 | ||||
| c52 -33 133 -83 180 -112 47 -29 180 -111 295 -182 116 -71 242 -148 280 -171 | ||||
| 245 -147 340 -209 340 -222 0 -8 -13 -23 -30 -33 -16 -10 -77 -48 -135 -83 | ||||
| -117 -71 -127 -77 -395 -241 -171 -104 -502 -309 -597 -369 -30 -19 -40 -35 | ||||
| -58 -91 -129 -390 -331 -727 -608 -1015 -358 -371 -803 -625 -1295 -739 -206 | ||||
| -48 -338 -62 -582 -63 -229 -1 -355 10 -525 46 -52 11 -104 20 -115 20 -17 0 | ||||
| -194 -96 -750 -407 -27 -15 -125 -70 -218 -121 -92 -51 -177 -100 -189 -109 | ||||
| -13 -8 -32 -15 -44 -15 -20 0 -21 6 -28 128 -12 245 -26 520 -41 852 -20 444 | ||||
| -18 430 -66 488 -353 428 -535 822 -626 1352 -26 154 -26 650 0 800 57 320 | ||||
| 133 555 261 807 95 187 210 359 365 545 48 58 46 43 67 493 8 187 19 419 24 | ||||
| 515 5 96 12 239 16 318 8 163 10 165 93 113z m9685 -5373 c15 -24 64 -165 89 | ||||
| -258 93 -344 103 -763 27 -1115 -39 -177 -117 -410 -144 -427 -15 -10 -4378 | ||||
| -10 -4393 0 -39 25 -138 367 -175 602 -17 113 -17 536 0 650 32 207 124 543 | ||||
| 154 562 7 4 1006 8 2220 8 l2208 0 14 -22z"/> | ||||
| <path d="M4361 8949 c-96 -9 -252 -44 -351 -80 -520 -188 -897 -628 -1006 | ||||
| -1174 -35 -176 -34 -434 1 -610 151 -747 782 -1265 1540 -1265 130 0 337 25 | ||||
| 405 49 14 5 57 19 95 31 452 140 839 526 991 988 61 186 68 239 68 502 0 265 | ||||
| -8 325 -70 510 -88 261 -279 539 -479 699 -349 279 -731 391 -1194 350z"/> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.9 KiB | 
							
								
								
									
										5241
									
								
								static/term/js/bundle.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								static/term/js/router.min.js.map
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										2
									
								
								static/term/js/terminal-open.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | ||||
| let e = document.querySelector('home-layout'); | ||||
| e.style.display = "block"; | ||||
							
								
								
									
										275
									
								
								static/term/js/terminal.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,275 @@ | ||||
| $(function() { | ||||
| 	var prompt = "[[b;#87cefa;]ai][[b;#FFFF00;]@aios] ~$ "; | ||||
| 	let url = new URL(window.location.href); | ||||
| 	let user = url.searchParams.get('user'); | ||||
|  let id = url.searchParams.get('id'); | ||||
|  if (id == null) { id = 2 }; | ||||
|  let did = url.searchParams.get('did'); | ||||
|  if (did == null) { did = "did:plc:4hqjfn7m6n5hno3doamuhgef" }; | ||||
| 	if (user) { prompt = "[[b;#87cefa;]" + user + "][[b;#FFFF00;]@aios] ~$ " }; | ||||
| 	var tab = "[[b;#87cefa;]<tab>]"; | ||||
| 	var command_all = ["ai","ls","ai.json"]; | ||||
| 	var handle = "yui.syui.ai"; | ||||
|  | ||||
| 	var file_all = "ai.json"; | ||||
| 	var ai_help = "\n\ | ||||
| \nCommands:\ | ||||
| \n        ai\ | ||||
| \n\ | ||||
| \nFlags:\ | ||||
| \n\t\t-h, --help\ | ||||
| \n\ | ||||
| \nOptions:\ | ||||
| \n\ta\t\t\t\t: logo\ | ||||
| \n\tv\t\t\t\t: pds version\ | ||||
| \n\tp\t\t\t\t: ai profile\ | ||||
| \n\tdid\t<handle>  : did\ | ||||
| \n\tplc\t<handle>  : plc\ | ||||
| \n\trecord <handle>  : timeline\ | ||||
| \n\ | ||||
| \nex:\ | ||||
| \n\t$ ai v\t: {\"version\":\"0.0.1\"}\ | ||||
| \n\t\ | ||||
| \nhttps://git.syui.ai/ai/bot\ | ||||
| \n\t\ | ||||
| 	"; | ||||
|  | ||||
| 	var ascii_logo = '\n\ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣿⣧⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣰⣿⣿⣿⣧⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣰⣿⣿⣿⣿⣿⣷⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣼⣿⣿⣿⣿⣿⣿⣿⣿⡄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣆⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣴⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣶⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣦⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣴⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣶⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣦⡀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣴⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠟⠋⠉⠀⠀⠀⠀⠀⠈⠉⠛⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⡄⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⣼⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡄⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⢀⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣇⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⢸⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⢹⣿⣿⣿⣿⣿⣿⣿⣿⣷⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢰⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣾⣿⣿⣿⣿⣿⣿⣿⣿⡿⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⣰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⡀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⢰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣧⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⢠⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣄⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣀⣴⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣧⠀⠀⠀\⠀ \n⠀⠀⠀⠀⢠⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣶⣦⣤⣤⣤⣤⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣇⠀⠀\⠀ \n⠀⠀⠀⢀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣆⠀\⠀ \n⠀⠀⠀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡆\⠀ \n⠀⠀⠘⠛⠛⠛⠛⠉⠉⠉⠉⠁⠀⠀⠀⠀⠈⠛⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⠋⠀⠀⠀⠀⠀⠉⠉⠉⠉⠙⠛⠛⠛⠛\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠛⠿⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠿⠟⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠉⠉⠉⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\⠀ \n\n'; | ||||
|  | ||||
| 	var ascii_ai = "\n\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⠉⣁⠉⠻⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⠟⠛⠋⣉⣉⠡⠤⠤⠤⠤⠤⠤⠤⠤⠬⠤⣁⣉⡉⠛⠻⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⢉⣤⡖⠚⣉⣡⣤⣦⣶⣶⣶⣶⣶⣶⣶⣶⣶⣶⣶⣴⣤⣤⠉⠑⣦⣌⠙⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⣾⣿⣦⣄⣅⣌⣄⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣤⣶⣾⣿⣿⣁⣘⡙⠻⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⢃⣤⣨⣿⡿⡿⠟⠟⠛⠻⠚⠛⠛⠛⠛⠛⠛⠋⡋⡋⠛⠛⠉⠉⠼⠿⠿⠿⠿⠿⠷⢶⢤⣈⠙⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⢋⡡⠞⠛⣉⣉⣤⣤⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣶⣶⣶⣶⣶⣶⣶⣦⣦⣦⣦⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣟⢁⣀⣤⣴⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠿⠟⠛⠋⡉⣉⣉⣁⣨⣀⣅⣉⣉⢉⠛⠛⠿⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠟⠋⣁⡤⣴⣲⢯⢯⡯⣗⡯⣞⡾⣺⣺⢵⢯⢯⣟⢶⢦⣤⣀⡉⠛⠿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⠉⣠⣔⡯⣗⡯⣗⡯⡯⣟⣞⣗⡯⣗⡯⣗⡯⡯⣟⣽⣺⢽⢽⣺⣳⢯⣟⡦⣌⡙⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⠁⣠⣻⣺⣺⢽⣳⢯⣗⡯⡯⣗⡯⡾⣝⣗⡯⣗⡯⡯⣗⣗⡯⣟⣽⣺⣺⢽⣺⢽⣳⣻⢦⣈⠛⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠋⢠⣺⣳⢽⣺⣺⢽⣺⢽⣺⢽⣫⣗⡯⡯⣗⣗⡯⣗⡯⡯⣗⣗⡯⣗⣗⡯⡾⣽⣺⢽⡽⣾⣽⣺⢵⡀⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠁⣔⣟⣞⡾⣽⣺⣺⢽⣺⢽⣺⢽⣺⣺⢽⣫⣗⣗⡯⣗⡯⡯⣗⣗⡯⣗⣗⡯⡯⣗⡯⣟⣞⣗⡿⣾⣽⢽⢦⠈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⢀⡮⣗⡯⡾⣝⡷⣽⣺⢽⣺⢽⣺⢽⣺⣺⢽⣺⣺⢵⢯⣗⡯⡯⣗⣗⡯⣗⣗⡯⡯⣗⡯⣗⣟⡾⣽⣻⣿⣯⣟⢷⡀⢻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⢀⡾⣝⣗⡯⡯⣗⡯⣗⡯⣟⡾⣽⣺⢽⣺⣺⢽⣺⣺⢽⢽⣺⢽⣫⣗⣗⡯⣗⣗⡯⡯⣗⡯⣗⡷⣻⣗⡷⣿⣟⣿⣽⣳⠈⣻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⣼⢽⣳⢗⡯⡯⣗⡯⣗⡯⣗⡯⣗⡯⠛⣞⡾⣽⣺⣺⢽⢽⣺⢽⣺⣺⢵⢯⣗⣗⡏⠙⣗⡯⣗⣯⢯⣷⢿⣽⣿⣿⡾⡽⣇⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠇⢸⣳⢽⣺⢽⢽⣫⣗⡯⣗⡯⣗⣯⡓⢡⡂⢵⢯⣗⡯⡾⣽⢽⣺⢽⣺⣺⢽⢽⣺⡞⢀⢇⠸⣽⣳⢽⣯⢿⣿⣿⣽⣷⣿⢯⢿⠀⣻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⣞⡾⣽⣺⢽⢽⣺⣺⢽⣳⢯⣗⠇⢠⣿⠆⢸⢽⣺⢽⢽⣳⣻⣺⢽⣺⣺⢽⢽⡺⠁⡬⣿⡀⢳⡽⡽⣾⡯⢿⣯⣿⣽⣿⣻⢽⡃⢸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡏⠠⣗⡯⣗⡯⣟⣽⣺⣺⢽⣺⢽⡺⢀⣿⣿⢅⠈⣟⡾⣽⢽⣺⢵⢯⣻⣺⣺⢽⢽⠃⡰⣽⣿⡇⢸⡽⡽⣿⢝⣽⣿⣿⣺⣿⡯⣯⡇⢸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⣳⢯⣗⡯⣗⣗⡯⡾⣽⣺⢽⠂⣸⣿⣾⡕⡀⢗⡯⣗⣟⡾⡽⣽⣺⣳⢽⢽⠃⢰⣱⣿⣷⡇⢸⡯⣟⠟⠀⣿⡿⣞⣾⣻⣽⡳⡇⢸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⣺⢽⣺⢽⣳⢗⡯⡯⣗⡯⡏⢀⣿⢿⣾⣗⠆⠸⣽⡳⣗⡯⡯⣗⣗⡯⣯⠏⢠⢣⣿⣿⣾⡇⢸⢯⠏⣰⠀⣿⣻⣳⣳⣿⣺⢽⠅⣸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠸⣺⢽⣺⢽⣺⢽⢽⣫⢷⠉⡇⠰⣿⣿⣻⣽⣎⠀⢗⡯⡗⢙⣽⡳⣗⡯⡏⢀⣗⣿⣿⣷⣿⠂⣺⠋⣠⣿⠀⣿⣺⢵⣫⣿⣺⢽⠀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠈⣗⣟⡾⣽⣺⢽⢽⣺⠍⠀⡇⠘⣉⣤⣬⣈⣑⣅⠘⣽⠂⠘⣮⢯⣗⠏⢠⡟⣉⣡⠤⢤⠄⢠⠃⢴⡟⣯⠀⣗⡯⠟⡾⣗⡯⡇⢰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡀⢸⣳⢯⣗⡯⣟⣽⡺⢀⠃⠀⠸⠋⢡⢡⡉⠻⣷⡄⠸⢀⡆⢹⣳⠃⣴⣿⣿⠋⣠⠐⡔⠀⠂⢳⡄⢰⣟⠀⠉⣠⣦⠘⣷⡻⠀⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⢡⡀⢯⣗⣗⡯⣗⣷⡃⠀⣼⠀⠀⠓⠀⠓⢽⠄⢹⣿⡄⢐⣧⠘⢁⣾⣿⣯⡯⢐⢌⠀⠁⣜⠆⢸⣿⡀⢟⠀⢨⣄⠙⡇⢹⠃⣼⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣌⠣⡈⢾⢵⢯⣗⢷⡃⢰⣿⡧⠈⣟⡀⠀⡸⡅⢸⡿⣿⣦⣹⣶⣻⣿⣷⣿⣇⠨⣫⢀⢀⢎⠃⣼⡿⣇⡾⠀⡢⠘⣨⠇⠈⣰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣧⡐⠈⢫⣟⢾⢽⡂⢹⣿⣿⡄⠱⢕⢵⠝⢀⣾⣿⣿⣻⣿⣻⣿⣻⣾⣟⣿⣆⠑⡕⠗⣁⣴⣿⡿⣿⠃⠄⣾⣵⡟⠀⢰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣄⠫⣟⡽⡃⢸⣿⣾⣿⡷⣾⣴⣻⡿⣯⣷⣿⣿⣻⣿⣽⣿⣟⣿⣻⣿⣟⣷⡿⣿⣽⣾⣿⡏⢠⠴⠟⠁⠄⣰⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⡈⢯⢧⠈⢿⣿⣾⣿⣿⣾⣿⢿⣿⣟⣿⣾⣿⣼⣿⣾⢿⣻⣿⣽⣿⣻⣿⣿⣻⣯⡿⠀⡄⡔⡔⠅⣴⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡄⠙⢆⠈⠻⣾⣿⣾⢿⣾⣿⣿⣽⠻⢷⣿⣻⣾⠿⠟⢿⣻⣽⣾⣿⢿⣾⡿⠋⡁⡜⠌⠈⣠⠪⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⡂⠀⣤⡈⠻⣾⣿⢿⣷⡿⣯⣷⣤⣤⣤⣤⣶⣾⣿⣿⢿⣻⠽⠛⢉⢠⠸⠈⡠⣰⠀⣿⡀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⠀⢨⣗⡯⣖⡄⢈⠉⠓⠻⠿⢿⣟⣿⣻⣿⠽⠟⠞⠋⢉⢠⠀⡰⠑⢁⢰⠀⡯⣪⠀⣳⣇⠸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡗⠨⠀⣻⢮⡯⢺⠀⢜⠈⡊⢐⢶⢔⡤⡤⣤⢤⢖⣖⢷⠅⢸⠀⠐⢁⡴⡇⡘⢠⡫⡎⡇⢸⣺⡄⢻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⡯⣗⠇⡪⠀⡃⠠⡃⠨⣿⣷⣯⣿⣾⣽⣷⣯⣿⠂⠀⡠⠀⣻⣺⡇⠀⢸⡂⢝⢦⠐⣗⣧⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⣜⠀⡯⡿⠀⠆⢐⠁⢜⠌⢨⣿⣾⣿⣽⣯⣿⣷⣿⢿⡁⢌⠐⠀⣳⣳⡃⠀⡪⡇⢸⢕⠄⢳⢽⡆⠸⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠁⡰⢱⠀⡯⡏⠨⠂⢀⡈⢈⡄⢸⣷⣿⣷⢿⣯⣷⣿⣾⣿⣆⠙⠅⠀⣞⣾⠀⢀⢯⡢⠘⣎⢇⠘⡽⣽⡀⢻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠏⠀⠀⣰⡹⠀⢯⠇⢘⠀⠍⣴⣿⣤⣿⡿⣷⣿⢿⣻⣽⣷⣿⡿⣿⣷⡂⢀⣗⣿⠀⠰⠣⢯⡀⢗⡝⡄⠹⣳⢧⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⢁⠔⢁⠞⣈⣴⠀⣫⠃⡐⢀⢞⣿⡿⣾⣟⣿⣿⣽⣿⡿⣟⣿⣾⡿⣿⣽⡂⢐⣗⡗⢀⣶⣦⣦⣤⣄⣉⠘⠀⠹⡽⣆⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠋⡠⠃⡔⢁⣼⣿⡿⠀⢸⠀⡂⢸⢸⣿⡿⣟⣿⢿⣞⡿⣾⣿⢿⣟⣷⢿⣿⣷⠁⢰⢯⡇⢰⣿⣯⣿⣟⣿⡿⣿⣷⣤⣈⠛⢆⠘⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠟⢠⠊⡠⢊⣰⣿⣿⠟⣰⠀⢸⠀⠀⡎⣾⠿⠟⠛⣉⢉⠚⢮⢿⣾⢿⣝⠮⢋⣠⣈⠂⣸⢽⠀⢼⣟⣯⣷⣿⣟⣿⣿⣽⣿⣿⡷⡌⢣⡈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠁⡔⢁⡔⢡⣾⣿⣿⣧⡐⢿⠀⣞⠀⠀⣏⢦⡶⣞⣿⣽⣳⣳⣽⣿⢿⣻⣾⣴⣳⣷⣿⠀⣺⡝⠀⣽⣿⢿⣻⣽⣿⣽⣾⡿⣯⣿⣿⣿⡀⢳⡀⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| \n⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠋⢀⠎⢠⠎⣠⣿⣿⣿⣿⣿⣿⣦⠀⡯⠀⠐⠧⠻⠛⠛⢋⢋⠋⠙⠛⠿⣿⢿⣷⡿⣿⣽⡿⠀⣷⠃⠀⣿⡿⣿⡿⣿⣽⣯⣷⣿⣿⡿⣿⣻⡇⢀⠹⣆⠘⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿\ | ||||
| "; | ||||
|  | ||||
| 	let list = 'https://bsky.social/xrpc/com.atproto.repo.listRecords?repo='; | ||||
| 	function bsky_record() { | ||||
| 		var u; | ||||
| 		u = axios.get(list + did + '&collection=app.bsky.feed.post&limit=1') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data.records[0].value,null,"\t"); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	function bsky_ver() { | ||||
| 		var u; | ||||
| 		u = axios.get('https://bsky.social/xrpc/_health') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data,null,"\t"); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	let plc_server = "https://plc.directory/" | ||||
| 	function bsky_plc() { | ||||
| 		var u; | ||||
| 		u = axios.get(plc_server + did + '/log') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data,null,"\t"); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	let desc = 'https://bsky.social/xrpc/com.atproto.repo.describeRepo?repo='; | ||||
| 	function bsky_desc() { | ||||
| 		var u; | ||||
| 		u = axios.get(desc + did + '&collection=app.bsky.actor.profile') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data,null,"\t"); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	function bsky_did() { | ||||
| 		var u; | ||||
| 		u = axios.get(desc + handle + '&collection=app.bsky.actor.profile') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data.did,null,"\t").replaceAll('"', ''); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	function test_json() { | ||||
| 		var u; | ||||
| 		u = axios.get('/term/json/ai.json') | ||||
| 			.then(function (response) { | ||||
| 				return JSON.stringify(response.data,null,"\t"); | ||||
| 			}) | ||||
| 		return u; | ||||
| 	} | ||||
|  | ||||
| 	function print_slowly(term, paragraph, callback) { | ||||
| 		var foo, i, lines; | ||||
| 		lines = paragraph.split("\n"); | ||||
| 		term.pause(); | ||||
| 		i = 0; | ||||
| 		foo = function(lines) { | ||||
| 			return setTimeout((function() { | ||||
| 				if (i < lines.length - 1) { | ||||
| 					term.echo(lines[i]); | ||||
| 					i++; | ||||
| 					return foo(lines); | ||||
| 				} else { | ||||
| 					term.resume(); | ||||
| 					return callback(); | ||||
| 				} | ||||
| 			}), 100); | ||||
| 		}; | ||||
| 		return foo(lines); | ||||
| 	} | ||||
|  | ||||
| 	function require_command(command_regex, terminal_history) { | ||||
| 		var executed = true; | ||||
| 		$.each(terminal_history, function(index, value) { | ||||
| 			if (command_regex.test(value)) { | ||||
| 				executed = true | ||||
| 			} | ||||
| 		}); | ||||
| 		return executed; | ||||
| 	} | ||||
|  | ||||
| 	function interpreter(input, term) { | ||||
| 		var command, inputs; | ||||
| 		inputs = input.split(/ +/) | ||||
| 		command = inputs[0]; | ||||
| 		option = inputs[1]; | ||||
| 		if (did == null) { did = "did:plc:4hqjfn7m6n5hno3doamuhgef" }; | ||||
| 		if (handle == null) { handle = "did:plc:4hqjfn7m6n5hno3doamuhgef" }; | ||||
| 		if (command === 'ai' && inputs[1] === undefined) { | ||||
| 			term.echo(ai_help); | ||||
| 		} else if (command === 'ls') { | ||||
| 			term.echo(file_all); | ||||
| 		} else if (command === 'cat') { | ||||
| 			if (option === 'ai.json') { | ||||
| 				term.echo(test_json()); | ||||
| 			} else { | ||||
| 				term.echo(file_all); | ||||
| 			} | ||||
| 		} else if (command === 'ai' && option === 'a') { | ||||
| 			print_slowly(term, ascii_logo); | ||||
| 		} else if (command === 'ai' && option === 'p') { | ||||
| 			term.echo(ascii_ai); | ||||
| 			term.echo(test_json()); | ||||
| 		} else if (command === 'ai' && option === 'plc') { | ||||
| 			if (inputs[2] != undefined) { did = inputs[2]; } | ||||
| 			url = desc + did + '&collection=app.bsky.actor.profile'; | ||||
| 			$.ajaxSetup({async: false}); | ||||
| 			$.getJSON(url, function(data) { | ||||
| 				did = JSON.stringify(data.did,null,"\t").replaceAll('"', '') | ||||
| 				url = plc_server + did + '/log'; | ||||
| 				$.ajaxSetup({async: false}); | ||||
| 				$.getJSON(url, function(data) {		 | ||||
| 					term.echo(JSON.stringify(data,null,"\t")); | ||||
| 				});$.ajaxSetup({async: true}); | ||||
| 			});$.ajaxSetup({async: true}); | ||||
| 		} else if (command === 'ai' && option === 'record') { | ||||
| 			if (inputs[2] != undefined) { did = inputs[2]; } | ||||
| 			url = list + did + '&collection=app.bsky.feed.post&limit=1'; | ||||
| 			$.ajaxSetup({async: false}); | ||||
| 			$.getJSON(url, function(data) { | ||||
| 				term.echo(JSON.stringify(data,null,"\t")); | ||||
| 			});$.ajaxSetup({async: true});		 | ||||
| 		} else if (command === 'ai' && option === 'did') { | ||||
| 			if (inputs[2] != undefined) { handle = inputs[2]; } | ||||
| 			url = desc + handle + '&collection=app.bsky.actor.profile'; | ||||
| 			$.ajaxSetup({async: false}); | ||||
| 			$.getJSON(url, function(data) { | ||||
| 				term.echo(JSON.stringify(data.did,null,"\t").replaceAll('"', '')); | ||||
| 			});$.ajaxSetup({async: true});		 | ||||
| 		} else if (command === 'ai' && option === 'handle') { | ||||
| 			if (inputs[2] != undefined) { did = inputs[2]; } | ||||
| 			url = desc + did + '&collection=app.bsky.actor.profile'; | ||||
| 			$.ajaxSetup({async: false}); | ||||
| 			$.getJSON(url, function(data) { | ||||
| 				term.echo(JSON.stringify(data,null,"\t")); | ||||
| 			});$.ajaxSetup({async: true});		 | ||||
| 		} else if (command === 'ai' && option === 'v') { | ||||
| 			url = 'https://bsky.social/xrpc/_health'; | ||||
| 			$.ajaxSetup({async: false}); | ||||
| 			$.getJSON(url, function(data) { | ||||
| 				term.echo(JSON.stringify(data,null,"\t")); | ||||
| 			});$.ajaxSetup({async: true});		 | ||||
| 		} else { | ||||
| 			term.echo(ai_help); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	function bash(inputs, term) { | ||||
| 		var argument, echo, insert; | ||||
| 		echo = term.echo; | ||||
| 		insert = term.insert; | ||||
| 		if (!inputs[1]) { | ||||
| 			//return console.log("none"); | ||||
| 		} else { | ||||
| 			argument = inputs[1]; | ||||
| 			if (/^\.\./.test(argument)) { | ||||
| 				return echo("-bash: cd: " + argument + ": Permission denied"); | ||||
| 			} else { | ||||
| 				return echo("-bash: cd: " + argument + ": No such file or directory"); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	$('#terminal').terminal(interpreter, { | ||||
| 		prompt: prompt, | ||||
| 		name: 'test', | ||||
| 		greetings: "", | ||||
| 		exit: false, | ||||
| 		height: 450, | ||||
| 		onInit: function(term) { | ||||
| 			//term.insert("ai"); | ||||
| 			//print_slowly(term, ascii_ai); | ||||
| 		}, | ||||
| 		completion: function(term, string, callback) { | ||||
| 			var t = $(term[0]).text(); | ||||
| 			if (t.match(/none/)) { | ||||
| 				term.clear(); | ||||
| 			} else { | ||||
| 				callback(command_all); | ||||
| 				term.history().clear(); | ||||
| 			} | ||||
| 		}, | ||||
| 		tabcompletion: true | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										11
									
								
								static/term/js/terminal.quick.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| $(function(){ | ||||
|     $("#mainTitle").click(function(){ | ||||
|         $("home-layout").toggle(); | ||||
|     }); | ||||
|     $("#mainTitleA").click(function(){ | ||||
|         $("home-layout").toggle(); | ||||
|     }); | ||||
|     $("#mainTitleB").click(function(){ | ||||
|         $("home-layout").toggle(); | ||||
|     }); | ||||
| }); | ||||
							
								
								
									
										6
									
								
								static/term/json/ai.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| { | ||||
|   "handle": "yui.syui.ai", | ||||
|   "displayName": "ai", | ||||
|   "day": "01/23", | ||||
|   "size": "123cm" | ||||
| } | ||||
							
								
								
									
										131
									
								
								static/term/pkg/axios/bin/GithubAPI.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,131 @@ | ||||
| import util from "util"; | ||||
| import cp from "child_process"; | ||||
| import {parseVersion} from "./helpers/parser.js"; | ||||
| import githubAxios from "./githubAxios.js"; | ||||
| import memoize from 'memoizee'; | ||||
|  | ||||
| const exec = util.promisify(cp.exec); | ||||
|  | ||||
| export default class GithubAPI { | ||||
|   constructor(owner, repo) { | ||||
|     if (!owner) { | ||||
|       throw new Error('repo owner must be specified'); | ||||
|     } | ||||
|  | ||||
|     if (!repo) { | ||||
|       throw new Error('repo must be specified'); | ||||
|     } | ||||
|  | ||||
|     this.repo = repo; | ||||
|     this.owner = owner; | ||||
|     this.axios = githubAxios.create({ | ||||
|       baseURL: `https://api.github.com/repos/${this.owner}/${this.repo}/`, | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   async createComment(issue, body) { | ||||
|     return (await this.axios.post(`/issues/${issue}/comments`, {body})).data; | ||||
|   } | ||||
|  | ||||
|   async getComments(issue, {desc = false, per_page= 100, page = 1} = {}) { | ||||
|     return (await this.axios.get(`/issues/${issue}/comments`, {params: {direction: desc ? 'desc' : 'asc', per_page, page}})).data; | ||||
|   } | ||||
|  | ||||
|   async getComment(id) { | ||||
|     return (await this.axios.get(`/issues/comments/${id}`)).data; | ||||
|   } | ||||
|  | ||||
|   async updateComment(id, body) { | ||||
|     return (await this.axios.patch(`/issues/comments/${id}`, {body})).data; | ||||
|   } | ||||
|  | ||||
|   async appendLabels(issue, labels) { | ||||
|     return (await this.axios.post(`/issues/${issue}/labels`, {labels})).data; | ||||
|   } | ||||
|  | ||||
|   async getUser(user) { | ||||
|     return (await githubAxios.get(`/users/${user}`)).data; | ||||
|   } | ||||
|  | ||||
|   async isCollaborator(user) { | ||||
|     try { | ||||
|       return (await this.axios.get(`/collaborators/${user}`)).status === 204; | ||||
|     } catch (e) { | ||||
|  | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   async deleteLabel(issue, label) { | ||||
|     return (await this.axios.delete(`/issues/${issue}/labels/${label}`)).data; | ||||
|   } | ||||
|  | ||||
|   async getIssue(issue) { | ||||
|     return (await this.axios.get(`/issues/${issue}`)).data; | ||||
|   } | ||||
|  | ||||
|   async getPR(issue) { | ||||
|     return (await this.axios.get(`/pulls/${issue}`)).data; | ||||
|   } | ||||
|  | ||||
|   async getIssues({state= 'open', labels, sort = 'created', desc = false, per_page = 100, page = 1}) { | ||||
|     return (await this.axios.get(`/issues`, {params: {state, labels, sort, direction: desc ? 'desc' : 'asc', per_page, page}})).data; | ||||
|   } | ||||
|  | ||||
|   async updateIssue(issue, data) { | ||||
|     return (await this.axios.patch(`/issues/${issue}`, data)).data; | ||||
|   } | ||||
|  | ||||
|   async closeIssue(issue) { | ||||
|     return this.updateIssue(issue, { | ||||
|       state: "closed" | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   async getReleases({per_page = 30, page= 1} = {}) { | ||||
|     return (await this.axios.get(`/releases`, {params: {per_page, page}})).data; | ||||
|   } | ||||
|  | ||||
|   async getRelease(release = 'latest') { | ||||
|     return (await this.axios.get(parseVersion(release) ? `/releases/tags/${release}` : `/releases/${release}`)).data; | ||||
|   } | ||||
|  | ||||
|   async getTags({per_page = 30, page= 1} = {}) { | ||||
|     return (await this.axios.get(`/tags`, {params: {per_page, page}})).data; | ||||
|   } | ||||
|  | ||||
|   async reopenIssue(issue) { | ||||
|     return this.updateIssue(issue, { | ||||
|       state: "open" | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   static async getTagRef(tag) { | ||||
|     try { | ||||
|       return (await exec(`git show-ref --tags "refs/tags/${tag}"`)).stdout.split(' ')[0]; | ||||
|     } catch (e) { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async getLatestTag() { | ||||
|     try{ | ||||
|       const {stdout} = await exec(`git for-each-ref refs/tags --sort=-taggerdate --format='%(refname)' --count=1`); | ||||
|  | ||||
|       return stdout.split('/').pop(); | ||||
|     } catch (e) {} | ||||
|   } | ||||
|  | ||||
|   static normalizeTag(tag){ | ||||
|     return tag ? 'v' + tag.replace(/^v/, '') : ''; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const {prototype} = GithubAPI; | ||||
|  | ||||
| ['getUser', 'isCollaborator'].forEach(methodName => { | ||||
|   prototype[methodName] = memoize(prototype[methodName], { promise: true }) | ||||
| }); | ||||
|  | ||||
| ['get', 'post', 'put', 'delete', 'isAxiosError'].forEach((method) => prototype[method] = function(...args){ | ||||
|   return this.axios[method](...args); | ||||
| }); | ||||
|  | ||||
							
								
								
									
										128
									
								
								static/term/pkg/axios/bin/RepoBot.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,128 @@ | ||||
| import GithubAPI from "./GithubAPI.js"; | ||||
| import api from './api.js'; | ||||
| import Handlebars from "handlebars"; | ||||
| import fs from "fs/promises"; | ||||
| import {colorize} from "./helpers/colorize.js"; | ||||
| import {getReleaseInfo} from "./contributors.js"; | ||||
| import path from "path"; | ||||
| import {fileURLToPath} from "url"; | ||||
|  | ||||
| const __dirname = path.dirname(fileURLToPath(import.meta.url)); | ||||
|  | ||||
| const NOTIFY_PR_TEMPLATE = path.resolve(__dirname, '../templates/pr_published.hbs'); | ||||
|  | ||||
| const normalizeTag = (tag) => tag ? 'v' + tag.replace(/^v/, '') : ''; | ||||
|  | ||||
| const GITHUB_BOT_LOGIN = 'github-actions[bot]'; | ||||
|  | ||||
| const skipCollaboratorPRs = true; | ||||
|  | ||||
| class RepoBot { | ||||
|   constructor(options) { | ||||
|     const { | ||||
|       owner, repo, | ||||
|       templates | ||||
|     } = options || {}; | ||||
|  | ||||
|     this.templates = Object.assign({ | ||||
|       published: NOTIFY_PR_TEMPLATE | ||||
|     }, templates); | ||||
|  | ||||
|     this.github = api || new GithubAPI(owner, repo); | ||||
|  | ||||
|     this.owner = this.github.owner; | ||||
|     this.repo = this.github.repo; | ||||
|   } | ||||
|  | ||||
|   async addComment(targetId, message) { | ||||
|     return this.github.createComment(targetId, message); | ||||
|   } | ||||
|  | ||||
|   async notifyPRPublished(id, tag) { | ||||
|     let pr; | ||||
|  | ||||
|     try { | ||||
|       pr = await this.github.getPR(id); | ||||
|     } catch (err) { | ||||
|       if(err.response?.status === 404) { | ||||
|         throw new Error(`PR #${id} not found (404)`); | ||||
|       } | ||||
|  | ||||
|       throw err; | ||||
|     } | ||||
|  | ||||
|     tag = normalizeTag(tag); | ||||
|  | ||||
|     const {merged, labels, user: {login, type}} = pr; | ||||
|  | ||||
|     const isBot = type === 'Bot'; | ||||
|  | ||||
|     if (!merged) { | ||||
|       return false | ||||
|     } | ||||
|  | ||||
|     await this.github.appendLabels(id, [tag]); | ||||
|  | ||||
|     if (isBot || labels.find(({name}) => name === 'automated pr') || (skipCollaboratorPRs && await this.github.isCollaborator(login))) { | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     const comments = await this.github.getComments(id, {desc: true}); | ||||
|  | ||||
|     const comment = comments.find( | ||||
|       ({body, user}) => user.login === GITHUB_BOT_LOGIN && body.indexOf('published in') >= 0 | ||||
|     ) | ||||
|  | ||||
|     if (comment) { | ||||
|       console.log(colorize()`Release comment [${comment.html_url}] already exists in #${pr.id}`); | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     const author = await this.github.getUser(login); | ||||
|  | ||||
|     author.isBot = isBot; | ||||
|  | ||||
|     const message = await this.constructor.renderTemplate(this.templates.published, { | ||||
|       id, | ||||
|       author, | ||||
|       release: { | ||||
|         tag, | ||||
|         url: `https://github.com/${this.owner}/${this.repo}/releases/tag/${tag}` | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     return await this.addComment(id, message); | ||||
|   } | ||||
|  | ||||
|   async notifyPublishedPRs(tag) { | ||||
|     tag = normalizeTag(tag); | ||||
|  | ||||
|     const release = await getReleaseInfo(tag); | ||||
|  | ||||
|     if (!release) { | ||||
|       throw Error(colorize()`Can't get release info for ${tag}`); | ||||
|     } | ||||
|  | ||||
|     const {merges} = release; | ||||
|  | ||||
|     console.log(colorize()`Found ${merges.length} PRs in ${tag}:`); | ||||
|  | ||||
|     let i = 0; | ||||
|  | ||||
|     for (const pr of merges) { | ||||
|       try { | ||||
|         console.log(colorize()`${i++}) Notify PR #${pr.id}`) | ||||
|         const result = await this.notifyPRPublished(pr.id, tag); | ||||
|         console.log('✔️', result ? 'Label, comment' : 'Label'); | ||||
|       } catch (err) { | ||||
|         console.warn(colorize('green', 'red')`❌ Failed notify PR ${pr.id}: ${err.message}`); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async renderTemplate(template, data) { | ||||
|     return Handlebars.compile(String(await fs.readFile(template)))(data); | ||||
|   } | ||||
| } | ||||
|  | ||||
| export default RepoBot; | ||||
							
								
								
									
										28
									
								
								static/term/pkg/axios/bin/actions/notify_published.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,28 @@ | ||||
| import minimist from "minimist"; | ||||
| import RepoBot from '../RepoBot.js'; | ||||
| import fs from 'fs/promises'; | ||||
|  | ||||
| const argv = minimist(process.argv.slice(2)); | ||||
| console.log(argv); | ||||
|  | ||||
| let {tag} = argv; | ||||
|  | ||||
| (async() => { | ||||
|   if (!tag || tag === true) { | ||||
|     const {version} = JSON.parse((await fs.readFile('./package.json')).toString()); | ||||
|  | ||||
|     tag = 'v' + version; | ||||
|   } else if (typeof tag !== 'string') { | ||||
|  | ||||
|     throw new Error('tag must be a string'); | ||||
|   } | ||||
|  | ||||
|   const bot = new RepoBot(); | ||||
|  | ||||
|   try { | ||||
|     await bot.notifyPublishedPRs(tag); | ||||
|   } catch (err) { | ||||
|     console.warn('Error:', err.message); | ||||
|   } | ||||
| })(); | ||||
|  | ||||
							
								
								
									
										3
									
								
								static/term/pkg/axios/bin/api.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| import GithubAPI from "./GithubAPI.js"; | ||||
|  | ||||
| export default new GithubAPI('axios', 'axios'); | ||||
							
								
								
									
										29
									
								
								static/term/pkg/axios/bin/check-build-version.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | ||||
| import fs from 'fs'; | ||||
| import assert from 'assert'; | ||||
| import axios from '../index.js'; | ||||
| import axiosBuild from '../dist/node/axios.cjs'; | ||||
|  | ||||
| const {version} = JSON.parse(fs.readFileSync('./package.json')); | ||||
|  | ||||
| console.log('Checking versions...\n----------------------------') | ||||
|  | ||||
| console.log(`Package version: v${version}`); | ||||
| console.log(`Axios version: v${axios.VERSION}`); | ||||
| console.log(`Axios build version: v${axiosBuild.VERSION}`); | ||||
| console.log(`----------------------------`); | ||||
|  | ||||
| assert.strictEqual( | ||||
|   version, | ||||
|   axios.VERSION, | ||||
|   `Version mismatch between package and Axios ${version} != ${axios.VERSION}` | ||||
| ); | ||||
|  | ||||
| assert.strictEqual( | ||||
|   version, | ||||
|   axiosBuild.VERSION, | ||||
|   `Version mismatch between package and build ${version} != ${axiosBuild.VERSION}` | ||||
| ); | ||||
|  | ||||
| console.log('✔️ PASSED\n'); | ||||
|  | ||||
|  | ||||
							
								
								
									
										241
									
								
								static/term/pkg/axios/bin/contributors.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,241 @@ | ||||
| import axios from "./githubAxios.js"; | ||||
| import util from "util"; | ||||
| import cp from "child_process"; | ||||
| import Handlebars from "handlebars"; | ||||
| import fs from "fs/promises"; | ||||
| import {colorize} from "./helpers/colorize.js"; | ||||
|  | ||||
| const exec = util.promisify(cp.exec); | ||||
|  | ||||
| const ONE_MB = 1024 * 1024; | ||||
|  | ||||
| const removeExtraLineBreaks = (str) => str.replace(/(?:\r\n|\r|\n){3,}/gm, '\r\n\r\n'); | ||||
|  | ||||
| const cleanTemplate = template => template | ||||
|   .replace(/\n +/g, '\n') | ||||
|   .replace(/^ +/, '') | ||||
|   .replace(/\n\n\n+/g, '\n\n') | ||||
|   .replace(/\n\n$/, '\n'); | ||||
|  | ||||
| const getUserFromCommit = ((commitCache) => async (sha) => { | ||||
|   try { | ||||
|     if(commitCache[sha] !== undefined) { | ||||
|       return commitCache[sha]; | ||||
|     } | ||||
|  | ||||
|     console.log(colorize()`fetch github commit info (${sha})`); | ||||
|  | ||||
|     const {data} = await axios.get(`https://api.github.com/repos/axios/axios/commits/${sha}`); | ||||
|  | ||||
|     return commitCache[sha] = { | ||||
|       ...data.commit.author, | ||||
|       ...data.author, | ||||
|       avatar_url_sm: data.author.avatar_url ? data.author.avatar_url + '&s=18' : '', | ||||
|     }; | ||||
|   } catch (err) { | ||||
|     return commitCache[sha] = null; | ||||
|   } | ||||
| })({}); | ||||
|  | ||||
| const getIssueById = ((cache) => async (id) => { | ||||
|   if(cache[id] !== undefined) { | ||||
|     return cache[id]; | ||||
|   } | ||||
|  | ||||
|   try { | ||||
|     const {data} = await axios.get(`https://api.github.com/repos/axios/axios/issues/${id}`); | ||||
|  | ||||
|     return cache[id] = data; | ||||
|   } catch (err) { | ||||
|     return null; | ||||
|   } | ||||
| })({}); | ||||
|  | ||||
| const getUserInfo = ((userCache) => async (userEntry) => { | ||||
|   const {email, commits} = userEntry; | ||||
|  | ||||
|   if (userCache[email] !== undefined) { | ||||
|     return userCache[email]; | ||||
|   } | ||||
|  | ||||
|   console.log(colorize()`fetch github user info [${userEntry.name}]`); | ||||
|  | ||||
|   return userCache[email] = { | ||||
|     ...userEntry, | ||||
|     ...await getUserFromCommit(commits[0].hash) | ||||
|   } | ||||
| })({}); | ||||
|  | ||||
| const deduplicate = (authors) => { | ||||
|   const loginsMap = {}; | ||||
|   const combined= {}; | ||||
|  | ||||
|   const assign = (a, b) => { | ||||
|     const {insertions, deletions, points, ...rest} = b; | ||||
|  | ||||
|     Object.assign(a, rest); | ||||
|  | ||||
|     a.insertions += insertions; | ||||
|     a.deletions += insertions; | ||||
|     a.insertions += insertions; | ||||
|   } | ||||
|  | ||||
|   for(const [email, user] of Object.entries(authors)) { | ||||
|     const {login} = user; | ||||
|     let entry; | ||||
|  | ||||
|     if(login && (entry = loginsMap[login])) { | ||||
|        assign(entry, user); | ||||
|     } else { | ||||
|       login && (loginsMap[login] = user); | ||||
|       combined[email] = user; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return combined; | ||||
| } | ||||
|  | ||||
| const getReleaseInfo = ((releaseCache) => async (tag) => { | ||||
|   if(releaseCache[tag] !== undefined) { | ||||
|     return releaseCache[tag]; | ||||
|   } | ||||
|  | ||||
|   const isUnreleasedTag = !tag; | ||||
|  | ||||
|   const version = 'v' + tag.replace(/^v/, ''); | ||||
|  | ||||
|   const command = isUnreleasedTag ? | ||||
|     `npx auto-changelog --unreleased-only --stdout --commit-limit false --template json` : | ||||
|     `npx auto-changelog ${ | ||||
|       version ? '--starting-version ' + version + ' --ending-version ' + version : '' | ||||
|     } --stdout --commit-limit false --template json`; | ||||
|  | ||||
|   console.log(command); | ||||
|  | ||||
|   const {stdout} = await exec(command, {maxBuffer: 10 * ONE_MB}); | ||||
|  | ||||
|   const release = JSON.parse(stdout)[0]; | ||||
|  | ||||
|   if(release) { | ||||
|     const authors = {}; | ||||
|  | ||||
|     const commits = [ | ||||
|       ...release.commits, | ||||
|       ...release.fixes.map(fix => fix.commit), | ||||
|       ...release.merges.map(fix => fix.commit) | ||||
|     ].filter(Boolean); | ||||
|  | ||||
|     const commitMergeMap = {}; | ||||
|  | ||||
|     for(const merge of release.merges) { | ||||
|       commitMergeMap[merge.commit.hash] = merge.id; | ||||
|     } | ||||
|  | ||||
|     for (const {hash, author, email, insertions, deletions} of commits) { | ||||
|       const entry = authors[email] = (authors[email] || { | ||||
|         name: author, | ||||
|         prs: [], | ||||
|         email, | ||||
|         commits: [], | ||||
|         insertions: 0, deletions: 0 | ||||
|       }); | ||||
|  | ||||
|       entry.commits.push({hash}); | ||||
|  | ||||
|       let pr; | ||||
|  | ||||
|       if((pr = commitMergeMap[hash])) { | ||||
|         entry.prs.push(pr); | ||||
|       } | ||||
|  | ||||
|       console.log(colorize()`Found commit [${hash}]`); | ||||
|  | ||||
|       entry.displayName = entry.name || author || entry.login; | ||||
|  | ||||
|       entry.github = entry.login ? `https://github.com/${encodeURIComponent(entry.login)}` : ''; | ||||
|  | ||||
|       entry.insertions += insertions; | ||||
|       entry.deletions += deletions; | ||||
|       entry.points = entry.insertions + entry.deletions; | ||||
|     } | ||||
|  | ||||
|     for (const [email, author] of Object.entries(authors)) { | ||||
|       const entry = authors[email] = await getUserInfo(author); | ||||
|  | ||||
|       entry.isBot = entry.type === "Bot"; | ||||
|     } | ||||
|  | ||||
|     release.authors = Object.values(deduplicate(authors)) | ||||
|       .sort((a, b) => b.points - a.points); | ||||
|  | ||||
|     release.allCommits = commits; | ||||
|   } | ||||
|  | ||||
|   releaseCache[tag] = release; | ||||
|  | ||||
|   return release; | ||||
| })({}); | ||||
|  | ||||
| const renderContributorsList = async (tag, template) => { | ||||
|   const release = await getReleaseInfo(tag); | ||||
|  | ||||
|   const compile = Handlebars.compile(String(await fs.readFile(template))) | ||||
|  | ||||
|   const content = compile(release); | ||||
|  | ||||
|   return removeExtraLineBreaks(cleanTemplate(content)); | ||||
| } | ||||
|  | ||||
| const renderPRsList = async (tag, template, {comments_threshold= 5, awesome_threshold= 5, label = 'add_to_changelog'} = {}) => { | ||||
|   const release = await getReleaseInfo(tag); | ||||
|  | ||||
|   const prs = {}; | ||||
|  | ||||
|   for(const merge of release.merges) { | ||||
|     const pr = await getIssueById(merge.id); | ||||
|  | ||||
|     if (pr && pr.labels.find(({name})=> name === label)) { | ||||
|       const {reactions, body} = pr; | ||||
|       prs[pr.number] = pr; | ||||
|       pr.isHot = pr.comments > comments_threshold; | ||||
|       const points = reactions['+1'] + | ||||
|         reactions['hooray'] + reactions['rocket'] + reactions['heart'] + reactions['laugh'] - reactions['-1']; | ||||
|  | ||||
|       pr.isAwesome = points > awesome_threshold; | ||||
|  | ||||
|       let match; | ||||
|  | ||||
|       pr.messages = []; | ||||
|  | ||||
|       if (body) { | ||||
|         const reg = /```+changelog\n*(.+?)?\n*```/gms; | ||||
|  | ||||
|         while((match = reg.exec(body))) { | ||||
|           match[1] && pr.messages.push(match[1]); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   release.prs = Object.values(prs); | ||||
|  | ||||
|   const compile = Handlebars.compile(String(await fs.readFile(template))) | ||||
|  | ||||
|   const content = compile(release); | ||||
|  | ||||
|   return removeExtraLineBreaks(cleanTemplate(content)); | ||||
| } | ||||
|  | ||||
| const getTagRef = async (tag) => { | ||||
|   try { | ||||
|     return (await exec(`git show-ref --tags "refs/tags/${tag}"`)).stdout.split(' ')[0]; | ||||
|   } catch(e) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| export { | ||||
|   renderContributorsList, | ||||
|   getReleaseInfo, | ||||
|   renderPRsList, | ||||
|   getTagRef | ||||
| } | ||||
							
								
								
									
										19
									
								
								static/term/pkg/axios/bin/githubAxios.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| import axios from '../index.js'; | ||||
| import {colorize} from "./helpers/colorize.js"; | ||||
|  | ||||
| const {GITHUB_TOKEN} = process.env; | ||||
|  | ||||
| GITHUB_TOKEN ? console.log(`[GITHUB_TOKEN OK]`) : console.warn(`[GITHUB_TOKEN is not defined]`); | ||||
|  | ||||
| const defaultTransform = axios.defaults.transformRequest; | ||||
|  | ||||
| export default axios.create({ | ||||
|   transformRequest: [defaultTransform[0], function (data) { | ||||
|     console.log(colorize()`[${this.method.toUpperCase()}] Request [${new URL(axios.getUri(this)).pathname}]`); | ||||
|     return data; | ||||
|   }], | ||||
|   baseURL: 'https://api.github.com/', | ||||
|   headers: { | ||||
|     Authorization: GITHUB_TOKEN ? `token ${GITHUB_TOKEN}` : null | ||||
|   } | ||||
| }); | ||||
							
								
								
									
										14
									
								
								static/term/pkg/axios/bin/helpers/colorize.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| import chalk from 'chalk'; | ||||
|  | ||||
| export const colorize = (...colors)=> { | ||||
|   if(!colors.length) { | ||||
|     colors = ['green', 'cyan', 'magenta', 'blue', 'yellow', 'red']; | ||||
|   } | ||||
|  | ||||
|   const colorsCount = colors.length; | ||||
|  | ||||
|   return (strings, ...values) => { | ||||
|     const {length} = values; | ||||
|     return strings.map((str, i) => i < length ? str + chalk[colors[i%colorsCount]].bold(values[i]) : str).join(''); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								static/term/pkg/axios/bin/helpers/parser.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| export const matchAll = (text, regexp, cb) => { | ||||
|   let match; | ||||
|   while((match = regexp.exec(text))) { | ||||
|     cb(match); | ||||
|   } | ||||
| } | ||||
|  | ||||
| export const parseSection = (body, name, cb) => { | ||||
|   matchAll(body, new RegExp(`^(#+)\\s+${name}?(.*?)^\\1\\s+\\w+`, 'gims'), cb); | ||||
| } | ||||
|  | ||||
| export const parseVersion = (rawVersion) => /^v?(\d+).(\d+).(\d+)/.exec(rawVersion); | ||||
							
								
								
									
										78
									
								
								static/term/pkg/axios/bin/injectContributorsList.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,78 @@ | ||||
| import fs from 'fs/promises'; | ||||
| import path from 'path'; | ||||
| import {renderContributorsList, getTagRef, renderPRsList} from './contributors.js'; | ||||
| import asyncReplace from 'string-replace-async'; | ||||
| import {fileURLToPath} from "url"; | ||||
| import {colorize} from "./helpers/colorize.js"; | ||||
|  | ||||
| const __dirname = path.dirname(fileURLToPath(import.meta.url)); | ||||
|  | ||||
| const CONTRIBUTORS_TEMPLATE = path.resolve(__dirname, '../templates/contributors.hbs'); | ||||
| const PRS_TEMPLATE = path.resolve(__dirname, '../templates/prs.hbs'); | ||||
|  | ||||
| const injectSection = async (name, contributorsRE, injector, infile = '../CHANGELOG.md') => { | ||||
|   console.log(colorize()`Checking ${name} sections in ${infile}`); | ||||
|  | ||||
|   infile = path.resolve(__dirname, infile); | ||||
|  | ||||
|   const content = String(await fs.readFile(infile)); | ||||
|   const headerRE = /^#+\s+\[([-_\d.\w]+)].+?$/mig; | ||||
|  | ||||
|   let tag; | ||||
|   let index = 0; | ||||
|   let isFirstTag = true; | ||||
|  | ||||
|   const newContent = await asyncReplace(content, headerRE, async (match, nextTag, offset) => { | ||||
|     const releaseContent = content.slice(index, offset); | ||||
|  | ||||
|     const hasSection = contributorsRE.test(releaseContent); | ||||
|  | ||||
|     const currentTag = tag; | ||||
|  | ||||
|     tag = nextTag; | ||||
|     index = offset + match.length; | ||||
|  | ||||
|     if(currentTag) { | ||||
|       if (hasSection) { | ||||
|         console.log(colorize()`[${currentTag}]: ✓ OK`); | ||||
|       } else { | ||||
|         const target = isFirstTag && (!await getTagRef(currentTag)) ? '' : currentTag; | ||||
|  | ||||
|         console.log(colorize()`[${currentTag}]: ❌ MISSED` + (!target ? ' (UNRELEASED)' : '')); | ||||
|  | ||||
|         isFirstTag = false; | ||||
|  | ||||
|         console.log(`Generating section...`); | ||||
|  | ||||
|         const section = await injector(target); | ||||
|  | ||||
|         if (!section) { | ||||
|           return match; | ||||
|         } | ||||
|  | ||||
|         console.log(colorize()`\nRENDERED SECTION [${name}] for [${currentTag}]:`); | ||||
|         console.log('-------------BEGIN--------------\n'); | ||||
|         console.log(section); | ||||
|         console.log('--------------END---------------\n'); | ||||
|  | ||||
|         return section + '\n' + match; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return match; | ||||
|   }); | ||||
|  | ||||
|   await fs.writeFile(infile, newContent); | ||||
| } | ||||
|  | ||||
| await injectSection( | ||||
|   'PRs', | ||||
|   /^\s*### PRs/mi, | ||||
|   (tag) => tag ? '' : renderPRsList(tag, PRS_TEMPLATE, {awesome_threshold: 5, comments_threshold: 7}), | ||||
| ); | ||||
|  | ||||
| await injectSection( | ||||
|   'contributors', | ||||
|   /^\s*### Contributors/mi, | ||||
|   (tag) => renderContributorsList(tag, CONTRIBUTORS_TEMPLATE) | ||||
| ); | ||||
							
								
								
									
										75
									
								
								static/term/pkg/axios/bin/pr.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,75 @@ | ||||
| import util from "util"; | ||||
| import cp from "child_process"; | ||||
| import Handlebars from "handlebars"; | ||||
| import fs from "fs/promises"; | ||||
| import prettyBytes from 'pretty-bytes'; | ||||
| import {gzipSize} from 'gzip-size'; | ||||
|  | ||||
| const exec = util.promisify(cp.exec); | ||||
|  | ||||
| const getBlobHistory = async (filepath, maxCount= 5) => { | ||||
|   const log = (await exec( | ||||
|     `git log --max-count=${maxCount} --no-walk --tags=v* --oneline --format=%H%d -- ${filepath}` | ||||
|   )).stdout; | ||||
|  | ||||
|   const commits = []; | ||||
|  | ||||
|   let match; | ||||
|  | ||||
|   const regexp = /^(\w+) \(tag: (v?[.\d]+)\)$/gm; | ||||
|  | ||||
|   while((match = regexp.exec(log))) { | ||||
|     commits.push({ | ||||
|       sha: match[1], | ||||
|       tag: match[2], | ||||
|       size: await getBlobSize(filepath, match[1]) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   return commits; | ||||
| } | ||||
|  | ||||
| const getBlobSize = async (filepath, sha ='HEAD') => { | ||||
|   const size = (await exec( | ||||
|     `git cat-file -s ${sha}:${filepath}` | ||||
|   )).stdout; | ||||
|  | ||||
|   return size ? +size : 0; | ||||
| } | ||||
|  | ||||
| const generateFileReport = async (files) => { | ||||
|   const stat = {}; | ||||
|  | ||||
|   for(const [name, file] of Object.entries(files)) { | ||||
|     const commits = await getBlobHistory(file); | ||||
|  | ||||
|     stat[file] = { | ||||
|       name, | ||||
|       size: (await fs.stat(file)).size, | ||||
|       path: file, | ||||
|       gzip: await gzipSize(String(await fs.readFile(file))), | ||||
|       commits, | ||||
|       history: commits.map(({tag, size}) => `${prettyBytes(size)} (${tag})`).join(' ← ') | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return stat; | ||||
| } | ||||
|  | ||||
| const generateBody = async ({files, template = './templates/pr.hbs'} = {}) => { | ||||
|   const data = { | ||||
|     files: await generateFileReport(files) | ||||
|   }; | ||||
|  | ||||
|   Handlebars.registerHelper('filesize', (bytes)=> prettyBytes(bytes)); | ||||
|  | ||||
|   return Handlebars.compile(String(await fs.readFile(template)))(data); | ||||
| } | ||||
|  | ||||
| console.log(await generateBody({ | ||||
|   files: { | ||||
|     'Browser build (UMD)' : './dist/axios.min.js', | ||||
|     'Browser build (ESM)' : './dist/esm/axios.min.js', | ||||
|   } | ||||
| })); | ||||
|  | ||||
							
								
								
									
										22
									
								
								static/term/pkg/axios/bin/ssl_hotfix.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| import {spawn} from 'child_process'; | ||||
|  | ||||
| const args = process.argv.slice(2); | ||||
|  | ||||
| console.log(`Running ${args.join(' ')} on ${process.version}\n`); | ||||
|  | ||||
| const match = /v(\d+)/.exec(process.version); | ||||
|  | ||||
| const isHotfixNeeded = match && match[1] > 16; | ||||
|  | ||||
| isHotfixNeeded && console.warn('Setting --openssl-legacy-provider as ssl hotfix'); | ||||
|  | ||||
| const test = spawn('cross-env', | ||||
|   isHotfixNeeded ? ['NODE_OPTIONS=--openssl-legacy-provider', ...args] : args, { | ||||
|     shell: true, | ||||
|     stdio: 'inherit' | ||||
|   } | ||||
| ); | ||||
|  | ||||
| test.on('exit', function (code) { | ||||
|   process.exit(code) | ||||
| }) | ||||
							
								
								
									
										3061
									
								
								static/term/pkg/axios/dist/axios.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/axios.js.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										2
									
								
								static/term/pkg/axios/dist/axios.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/axios.min.js.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										3234
									
								
								static/term/pkg/axios/dist/browser/axios.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/browser/axios.cjs.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										3257
									
								
								static/term/pkg/axios/dist/esm/axios.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/esm/axios.js.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										2
									
								
								static/term/pkg/axios/dist/esm/axios.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/esm/axios.min.js.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										4327
									
								
								static/term/pkg/axios/dist/node/axios.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								static/term/pkg/axios/dist/node/axios.cjs.map
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										221
									
								
								static/term/pkg/jquery-mousewheel/jquery.mousewheel.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,221 @@ | ||||
| /*! | ||||
|  * jQuery Mousewheel 3.1.13 | ||||
|  * | ||||
|  * Copyright jQuery Foundation and other contributors | ||||
|  * Released under the MIT license | ||||
|  * http://jquery.org/license | ||||
|  */ | ||||
|  | ||||
| (function (factory) { | ||||
|     if ( typeof define === 'function' && define.amd ) { | ||||
|         // AMD. Register as an anonymous module. | ||||
|         define(['jquery'], factory); | ||||
|     } else if (typeof exports === 'object') { | ||||
|         // Node/CommonJS style for Browserify | ||||
|         module.exports = factory; | ||||
|     } else { | ||||
|         // Browser globals | ||||
|         factory(jQuery); | ||||
|     } | ||||
| }(function ($) { | ||||
|  | ||||
|     var toFix  = ['wheel', 'mousewheel', 'DOMMouseScroll', 'MozMousePixelScroll'], | ||||
|         toBind = ( 'onwheel' in document || document.documentMode >= 9 ) ? | ||||
|                     ['wheel'] : ['mousewheel', 'DomMouseScroll', 'MozMousePixelScroll'], | ||||
|         slice  = Array.prototype.slice, | ||||
|         nullLowestDeltaTimeout, lowestDelta; | ||||
|  | ||||
|     if ( $.event.fixHooks ) { | ||||
|         for ( var i = toFix.length; i; ) { | ||||
|             $.event.fixHooks[ toFix[--i] ] = $.event.mouseHooks; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     var special = $.event.special.mousewheel = { | ||||
|         version: '3.1.12', | ||||
|  | ||||
|         setup: function() { | ||||
|             if ( this.addEventListener ) { | ||||
|                 for ( var i = toBind.length; i; ) { | ||||
|                     this.addEventListener( toBind[--i], handler, false ); | ||||
|                 } | ||||
|             } else { | ||||
|                 this.onmousewheel = handler; | ||||
|             } | ||||
|             // Store the line height and page height for this particular element | ||||
|             $.data(this, 'mousewheel-line-height', special.getLineHeight(this)); | ||||
|             $.data(this, 'mousewheel-page-height', special.getPageHeight(this)); | ||||
|         }, | ||||
|  | ||||
|         teardown: function() { | ||||
|             if ( this.removeEventListener ) { | ||||
|                 for ( var i = toBind.length; i; ) { | ||||
|                     this.removeEventListener( toBind[--i], handler, false ); | ||||
|                 } | ||||
|             } else { | ||||
|                 this.onmousewheel = null; | ||||
|             } | ||||
|             // Clean up the data we added to the element | ||||
|             $.removeData(this, 'mousewheel-line-height'); | ||||
|             $.removeData(this, 'mousewheel-page-height'); | ||||
|         }, | ||||
|  | ||||
|         getLineHeight: function(elem) { | ||||
|             var $elem = $(elem), | ||||
|                 $parent = $elem['offsetParent' in $.fn ? 'offsetParent' : 'parent'](); | ||||
|             if (!$parent.length) { | ||||
|                 $parent = $('body'); | ||||
|             } | ||||
|             return parseInt($parent.css('fontSize'), 10) || parseInt($elem.css('fontSize'), 10) || 16; | ||||
|         }, | ||||
|  | ||||
|         getPageHeight: function(elem) { | ||||
|             return $(elem).height(); | ||||
|         }, | ||||
|  | ||||
|         settings: { | ||||
|             adjustOldDeltas: true, // see shouldAdjustOldDeltas() below | ||||
|             normalizeOffset: true  // calls getBoundingClientRect for each event | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     $.fn.extend({ | ||||
|         mousewheel: function(fn) { | ||||
|             return fn ? this.bind('mousewheel', fn) : this.trigger('mousewheel'); | ||||
|         }, | ||||
|  | ||||
|         unmousewheel: function(fn) { | ||||
|             return this.unbind('mousewheel', fn); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|  | ||||
|     function handler(event) { | ||||
|         var orgEvent   = event || window.event, | ||||
|             args       = slice.call(arguments, 1), | ||||
|             delta      = 0, | ||||
|             deltaX     = 0, | ||||
|             deltaY     = 0, | ||||
|             absDelta   = 0, | ||||
|             offsetX    = 0, | ||||
|             offsetY    = 0; | ||||
|         event = $.event.fix(orgEvent); | ||||
|         event.type = 'mousewheel'; | ||||
|  | ||||
|         // Old school scrollwheel delta | ||||
|         if ( 'detail'      in orgEvent ) { deltaY = orgEvent.detail * -1;      } | ||||
|         if ( 'wheelDelta'  in orgEvent ) { deltaY = orgEvent.wheelDelta;       } | ||||
|         if ( 'wheelDeltaY' in orgEvent ) { deltaY = orgEvent.wheelDeltaY;      } | ||||
|         if ( 'wheelDeltaX' in orgEvent ) { deltaX = orgEvent.wheelDeltaX * -1; } | ||||
|  | ||||
|         // Firefox < 17 horizontal scrolling related to DOMMouseScroll event | ||||
|         if ( 'axis' in orgEvent && orgEvent.axis === orgEvent.HORIZONTAL_AXIS ) { | ||||
|             deltaX = deltaY * -1; | ||||
|             deltaY = 0; | ||||
|         } | ||||
|  | ||||
|         // Set delta to be deltaY or deltaX if deltaY is 0 for backwards compatabilitiy | ||||
|         delta = deltaY === 0 ? deltaX : deltaY; | ||||
|  | ||||
|         // New school wheel delta (wheel event) | ||||
|         if ( 'deltaY' in orgEvent ) { | ||||
|             deltaY = orgEvent.deltaY * -1; | ||||
|             delta  = deltaY; | ||||
|         } | ||||
|         if ( 'deltaX' in orgEvent ) { | ||||
|             deltaX = orgEvent.deltaX; | ||||
|             if ( deltaY === 0 ) { delta  = deltaX * -1; } | ||||
|         } | ||||
|  | ||||
|         // No change actually happened, no reason to go any further | ||||
|         if ( deltaY === 0 && deltaX === 0 ) { return; } | ||||
|  | ||||
|         // Need to convert lines and pages to pixels if we aren't already in pixels | ||||
|         // There are three delta modes: | ||||
|         //   * deltaMode 0 is by pixels, nothing to do | ||||
|         //   * deltaMode 1 is by lines | ||||
|         //   * deltaMode 2 is by pages | ||||
|         if ( orgEvent.deltaMode === 1 ) { | ||||
|             var lineHeight = $.data(this, 'mousewheel-line-height'); | ||||
|             delta  *= lineHeight; | ||||
|             deltaY *= lineHeight; | ||||
|             deltaX *= lineHeight; | ||||
|         } else if ( orgEvent.deltaMode === 2 ) { | ||||
|             var pageHeight = $.data(this, 'mousewheel-page-height'); | ||||
|             delta  *= pageHeight; | ||||
|             deltaY *= pageHeight; | ||||
|             deltaX *= pageHeight; | ||||
|         } | ||||
|  | ||||
|         // Store lowest absolute delta to normalize the delta values | ||||
|         absDelta = Math.max( Math.abs(deltaY), Math.abs(deltaX) ); | ||||
|  | ||||
|         if ( !lowestDelta || absDelta < lowestDelta ) { | ||||
|             lowestDelta = absDelta; | ||||
|  | ||||
|             // Adjust older deltas if necessary | ||||
|             if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { | ||||
|                 lowestDelta /= 40; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Adjust older deltas if necessary | ||||
|         if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { | ||||
|             // Divide all the things by 40! | ||||
|             delta  /= 40; | ||||
|             deltaX /= 40; | ||||
|             deltaY /= 40; | ||||
|         } | ||||
|  | ||||
|         // Get a whole, normalized value for the deltas | ||||
|         delta  = Math[ delta  >= 1 ? 'floor' : 'ceil' ](delta  / lowestDelta); | ||||
|         deltaX = Math[ deltaX >= 1 ? 'floor' : 'ceil' ](deltaX / lowestDelta); | ||||
|         deltaY = Math[ deltaY >= 1 ? 'floor' : 'ceil' ](deltaY / lowestDelta); | ||||
|  | ||||
|         // Normalise offsetX and offsetY properties | ||||
|         if ( special.settings.normalizeOffset && this.getBoundingClientRect ) { | ||||
|             var boundingRect = this.getBoundingClientRect(); | ||||
|             offsetX = event.clientX - boundingRect.left; | ||||
|             offsetY = event.clientY - boundingRect.top; | ||||
|         } | ||||
|  | ||||
|         // Add information to the event object | ||||
|         event.deltaX = deltaX; | ||||
|         event.deltaY = deltaY; | ||||
|         event.deltaFactor = lowestDelta; | ||||
|         event.offsetX = offsetX; | ||||
|         event.offsetY = offsetY; | ||||
|         // Go ahead and set deltaMode to 0 since we converted to pixels | ||||
|         // Although this is a little odd since we overwrite the deltaX/Y | ||||
|         // properties with normalized deltas. | ||||
|         event.deltaMode = 0; | ||||
|  | ||||
|         // Add event and delta to the front of the arguments | ||||
|         args.unshift(event, delta, deltaX, deltaY); | ||||
|  | ||||
|         // Clearout lowestDelta after sometime to better | ||||
|         // handle multiple device types that give different | ||||
|         // a different lowestDelta | ||||
|         // Ex: trackpad = 3 and mouse wheel = 120 | ||||
|         if (nullLowestDeltaTimeout) { clearTimeout(nullLowestDeltaTimeout); } | ||||
|         nullLowestDeltaTimeout = setTimeout(nullLowestDelta, 200); | ||||
|  | ||||
|         return ($.event.dispatch || $.event.handle).apply(this, args); | ||||
|     } | ||||
|  | ||||
|     function nullLowestDelta() { | ||||
|         lowestDelta = null; | ||||
|     } | ||||
|  | ||||
|     function shouldAdjustOldDeltas(orgEvent, absDelta) { | ||||
|         // If this is an older event and the delta is divisable by 120, | ||||
|         // then we are assuming that the browser is treating this as an | ||||
|         // older mouse wheel event and that we should divide the deltas | ||||
|         // by 40 to try and get a more usable deltaFactor. | ||||
|         // Side note, this actually impacts the reported scroll distance | ||||
|         // in older browsers and can cause scrolling to be slower than native. | ||||
|         // Turn this off by setting $.event.special.mousewheel.settings.adjustOldDeltas to false. | ||||
|         return special.settings.adjustOldDeltas && orgEvent.type === 'mousewheel' && absDelta % 120 === 0; | ||||
|     } | ||||
|  | ||||
| })); | ||||
							
								
								
									
										8
									
								
								static/term/pkg/jquery-mousewheel/jquery.mousewheel.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| /*! | ||||
|  * jQuery Mousewheel 3.1.13 | ||||
|  * | ||||
|  * Copyright 2015 jQuery Foundation and other contributors | ||||
|  * Released under the MIT license. | ||||
|  * http://jquery.org/license | ||||
|  */ | ||||
| !function(a){"function"==typeof define&&define.amd?define(["jquery"],a):"object"==typeof exports?module.exports=a:a(jQuery)}(function(a){function b(b){var g=b||window.event,h=i.call(arguments,1),j=0,l=0,m=0,n=0,o=0,p=0;if(b=a.event.fix(g),b.type="mousewheel","detail"in g&&(m=-1*g.detail),"wheelDelta"in g&&(m=g.wheelDelta),"wheelDeltaY"in g&&(m=g.wheelDeltaY),"wheelDeltaX"in g&&(l=-1*g.wheelDeltaX),"axis"in g&&g.axis===g.HORIZONTAL_AXIS&&(l=-1*m,m=0),j=0===m?l:m,"deltaY"in g&&(m=-1*g.deltaY,j=m),"deltaX"in g&&(l=g.deltaX,0===m&&(j=-1*l)),0!==m||0!==l){if(1===g.deltaMode){var q=a.data(this,"mousewheel-line-height");j*=q,m*=q,l*=q}else if(2===g.deltaMode){var r=a.data(this,"mousewheel-page-height");j*=r,m*=r,l*=r}if(n=Math.max(Math.abs(m),Math.abs(l)),(!f||f>n)&&(f=n,d(g,n)&&(f/=40)),d(g,n)&&(j/=40,l/=40,m/=40),j=Math[j>=1?"floor":"ceil"](j/f),l=Math[l>=1?"floor":"ceil"](l/f),m=Math[m>=1?"floor":"ceil"](m/f),k.settings.normalizeOffset&&this.getBoundingClientRect){var s=this.getBoundingClientRect();o=b.clientX-s.left,p=b.clientY-s.top}return b.deltaX=l,b.deltaY=m,b.deltaFactor=f,b.offsetX=o,b.offsetY=p,b.deltaMode=0,h.unshift(b,j,l,m),e&&clearTimeout(e),e=setTimeout(c,200),(a.event.dispatch||a.event.handle).apply(this,h)}}function c(){f=null}function d(a,b){return k.settings.adjustOldDeltas&&"mousewheel"===a.type&&b%120===0}var e,f,g=["wheel","mousewheel","DOMMouseScroll","MozMousePixelScroll"],h="onwheel"in document||document.documentMode>=9?["wheel"]:["mousewheel","DomMouseScroll","MozMousePixelScroll"],i=Array.prototype.slice;if(a.event.fixHooks)for(var j=g.length;j;)a.event.fixHooks[g[--j]]=a.event.mouseHooks;var k=a.event.special.mousewheel={version:"3.1.12",setup:function(){if(this.addEventListener)for(var c=h.length;c;)this.addEventListener(h[--c],b,!1);else this.onmousewheel=b;a.data(this,"mousewheel-line-height",k.getLineHeight(this)),a.data(this,"mousewheel-page-height",k.getPageHeight(this))},teardown:function(){if(this.removeEventListener)for(var c=h.length;c;)this.removeEventListener(h[--c],b,!1);else this.onmousewheel=null;a.removeData(this,"mousewheel-line-height"),a.removeData(this,"mousewheel-page-height")},getLineHeight:function(b){var c=a(b),d=c["offsetParent"in a.fn?"offsetParent":"parent"]();return d.length||(d=a("body")),parseInt(d.css("fontSize"),10)||parseInt(c.css("fontSize"),10)||16},getPageHeight:function(b){return a(b).height()},settings:{adjustOldDeltas:!0,normalizeOffset:!0}};a.fn.extend({mousewheel:function(a){return a?this.bind("mousewheel",a):this.trigger("mousewheel")},unmousewheel:function(a){return this.unbind("mousewheel",a)}})}); | ||||
							
								
								
									
										2
									
								
								static/term/pkg/jquery.ajax/jquery.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								static/term/pkg/jquery.terminal/bin/yuicompressor-2.4.8.jar
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										249
									
								
								static/term/pkg/jquery.terminal/css/jquery.terminal.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,249 @@ | ||||
| /*! | ||||
|  *       __ _____                     ________                              __ | ||||
|  *      / // _  /__ __ _____ ___ __ _/__  ___/__ ___ ______ __ __  __ ___  / / | ||||
|  *  __ / // // // // // _  // _// // / / // _  // _//     // //  \/ // _ \/ / | ||||
|  * /  / // // // // // ___// / / // / / // ___// / / / / // // /\  // // / /__ | ||||
|  * \___//____ \\___//____//_/ _\_  / /_//____//_/ /_/ /_//_//_/ /_/ \__\_\___/ | ||||
|  *           \/              /____/                              version 0.10.8 | ||||
|  * http://terminal.jcubic.pl | ||||
|  * | ||||
|  * This file is part of jQuery Terminal. | ||||
|  * | ||||
|  * Copyright (c) 2011-2016 Jakub Jankiewicz <http://jcubic.pl> | ||||
|  * Released under the MIT license | ||||
|  * | ||||
|  * Date: Tue, 17 May 2016 09:01:55 +0000 | ||||
|  */ | ||||
| .terminal .terminal-output .format, .cmd .format, | ||||
| .cmd .prompt, .cmd .prompt div, .terminal .terminal-output div div{ | ||||
|     display: inline-block; | ||||
| } | ||||
| .terminal h1, .terminal h2, .terminal h3, .terminal h4, .terminal h5, .terminal h6, .terminal pre, .cmd { | ||||
|     margin: 0; | ||||
| } | ||||
| .terminal h1, .terminal h2, .terminal h3, .terminal h4, .terminal h5, .terminal h6 { | ||||
|     line-height: 1.2em; | ||||
| } | ||||
| /* | ||||
| .cmd .mask { | ||||
|     width: 10px; | ||||
|     height: 11px; | ||||
|     background: black; | ||||
|     z-index: 100; | ||||
| } | ||||
| */ | ||||
| .cmd .clipboard { | ||||
|     position: absolute; | ||||
|     height: 16px; | ||||
|     left: -6px; | ||||
|     /* this seems to work after all on Android */ | ||||
|     /*left: -99999px; | ||||
|     clip: rect(1px,1px,1px,1px); | ||||
|     /* on desktop textarea appear when paste */ | ||||
|     /* | ||||
|     opacity: 0.01; | ||||
|     filter: alpha(opacity = 0.01); | ||||
|     filter: progid:DXImageTransform.Microsoft.Alpha(opacity=0.01); | ||||
|     */ | ||||
|     width: 5px; /* textarea need to have width or it will not work on Android */ | ||||
|     background: transparent; | ||||
|     border: none; | ||||
|     color: transparent; | ||||
|     outline: none; | ||||
|     padding: 0; | ||||
|     resize: none; | ||||
|     z-index: 0; | ||||
|     overflow: hidden; | ||||
| } | ||||
| .terminal .error { | ||||
|     color: #f00; | ||||
| } | ||||
| .terminal { | ||||
|     padding: 10px; | ||||
|     position: relative; | ||||
|     /*overflow: hidden;*/ | ||||
|     overflow: auto; | ||||
| } | ||||
| .cmd { | ||||
|     padding: 0; | ||||
|     height: 1.3em; | ||||
|     position: relative; | ||||
|     /*margin-top: 3px; */ | ||||
| } | ||||
| .terminal .inverted, .cmd .inverted, .cmd .cursor.blink { | ||||
|     background-color: #aaa; | ||||
|     color: #000; | ||||
| } | ||||
| .cmd .cursor.blink { | ||||
|     -webkit-animation: terminal-blink 1s infinite steps(1, start); | ||||
|        -moz-animation: terminal-blink 1s infinite steps(1, start); | ||||
|         -ms-animation: terminal-blink 1s infinite steps(1, start); | ||||
|             animation: terminal-blink 1s infinite steps(1, start); | ||||
| } | ||||
| @-webkit-keyframes terminal-blink { | ||||
|   0%, 100% { | ||||
|       background-color: #000; | ||||
|       color: #aaa; | ||||
|   } | ||||
|   50% { | ||||
|       background-color: #bbb; | ||||
|       color: #000; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @-ms-keyframes terminal-blink { | ||||
|   0%, 100% { | ||||
|       background-color: #000; | ||||
|       color: #aaa; | ||||
|   } | ||||
|   50% { | ||||
|       background-color: #bbb; | ||||
|       color: #000; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @-moz-keyframes terminal-blink { | ||||
|   0%, 100% { | ||||
|       background-color: #000; | ||||
|       color: #aaa; | ||||
|   } | ||||
|   50% { | ||||
|       background-color: #bbb; | ||||
|       color: #000; | ||||
|   } | ||||
| } | ||||
| @keyframes terminal-blink { | ||||
|   0%, 100% { | ||||
|       background-color: #000; | ||||
|       color: #aaa; | ||||
|   } | ||||
|   50% { | ||||
|       background-color: #bbb; /* not #aaa because it's seems there is Google Chrome bug */ | ||||
|       color: #000; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .terminal .terminal-output div div, .cmd .prompt { | ||||
|     display: block; | ||||
|     line-height: 14px; | ||||
|     height: auto; | ||||
| } | ||||
| .cmd .prompt { | ||||
|     float: left; | ||||
| } | ||||
| .terminal, .cmd { | ||||
|     font-family: SFMono-Regular,Consolas,Liberation Mono,Menlo,Courier,monospace; | ||||
|     color: #aaa; | ||||
|     background-color: #000; | ||||
|     font-size: 14px; | ||||
|     line-height: 16px; | ||||
|     padding-bottom:3px; | ||||
| } | ||||
| .terminal-output > div { | ||||
|     /*padding-top: 3px;*/ | ||||
|     min-height: 14px; | ||||
| } | ||||
| .terminal .terminal-output div span { | ||||
|     display: inline-block; | ||||
| } | ||||
| .cmd span { | ||||
|     float: left; | ||||
|     /*display: inline-block; */ | ||||
| } | ||||
| /* fix double style of selecting text in terminal */ | ||||
| .terminal-output span, .terminal-output a, .cmd div, .cmd span, .terminal td, | ||||
| .terminal pre, .terminal h1, .terminal h2, .terminal h3, .terminal h4, | ||||
| .terminal h5, .terminal h6 { | ||||
|     -webkit-touch-callout: initial; | ||||
|     -webkit-user-select: initial; | ||||
|     -khtml-user-select: initial; | ||||
|     -moz-user-select: initial; | ||||
|     -ms-user-select: initial; | ||||
|     user-select: initial; | ||||
| } | ||||
| .terminal, .terminal-output, .terminal-output div { | ||||
|     -webkit-touch-callout: none; | ||||
|     -webkit-user-select: none; | ||||
|     -khtml-user-select: none; | ||||
|     -moz-user-select: none; | ||||
|     -ms-user-select: none; | ||||
|     user-select: none; | ||||
| } | ||||
| /* firefox hack */ | ||||
| @-moz-document url-prefix() { | ||||
|     .terminal, .terminal-output, .terminal-output div { | ||||
|         -webkit-touch-callout: initial; | ||||
|         -webkit-user-select: initial; | ||||
|         -khtml-user-select: initial; | ||||
|         -moz-user-select: initial; | ||||
|         -ms-user-select: initial; | ||||
|         user-select: initial; | ||||
|     } | ||||
| } | ||||
| .terminal table { | ||||
|     border-collapse: collapse; | ||||
| } | ||||
| .terminal td { | ||||
|     border: 1px solid #aaa; | ||||
| } | ||||
| .terminal h1::-moz-selection, | ||||
| .terminal h2::-moz-selection, | ||||
| .terminal h3::-moz-selection, | ||||
| .terminal h4::-moz-selection, | ||||
| .terminal h5::-moz-selection, | ||||
| .terminal h6::-moz-selection, | ||||
| .terminal pre::-moz-selection, | ||||
| .terminal td::-moz-selection, | ||||
| .terminal .terminal-output div div::-moz-selection, | ||||
| .terminal .terminal-output div span::-moz-selection, | ||||
| .terminal .terminal-output div div a::-moz-selection, | ||||
| .cmd div::-moz-selection, | ||||
| .cmd > span::-moz-selection, | ||||
| .cmd .prompt span::-moz-selection { | ||||
|     background-color: #aaa; | ||||
|     color: #000; | ||||
| } | ||||
| /* this don't work in Chrome | ||||
| .terminal tr td::-moz-selection { | ||||
|     border-color: #000; | ||||
| } | ||||
| .terminal tr td::selection { | ||||
|     border-color: #000; | ||||
| } | ||||
| */ | ||||
| .terminal h1::selection, | ||||
| .terminal h2::selection, | ||||
| .terminal h3::selection, | ||||
| .terminal h4::selection, | ||||
| .terminal h5::selection, | ||||
| .terminal h6::selection, | ||||
| .terminal pre::selection, | ||||
| .terminal td::selection, | ||||
| .terminal .terminal-output div div::selection, | ||||
| .terminal .terminal-output div div a::selection, | ||||
| .terminal .terminal-output div span::selection, | ||||
| .cmd div::selection, | ||||
| .cmd > span::selection, | ||||
| .cmd .prompt span::selection { | ||||
|     background-color: #aaa; | ||||
|     color: #000; | ||||
| } | ||||
| .terminal .terminal-output div.error, .terminal .terminal-output div.error div { | ||||
|     color: red; | ||||
| } | ||||
| .tilda { | ||||
|     position: fixed; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     width: 100%; | ||||
|     z-index: 1100; | ||||
| } | ||||
| .clear { | ||||
|     clear: both; | ||||
| } | ||||
| .terminal a { | ||||
|     color: #0F60FF; | ||||
| } | ||||
| .terminal a:hover { | ||||
|     color: red; | ||||
| } | ||||
							
								
								
									
										1
									
								
								static/term/pkg/jquery.terminal/css/jquery.terminal.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| .terminal .terminal-output .format,.cmd .format,.cmd .prompt,.cmd .prompt div,.terminal .terminal-output div div{display:inline-block}.terminal h1,.terminal h2,.terminal h3,.terminal h4,.terminal h5,.terminal h6,.terminal pre,.cmd{margin:0}.terminal h1,.terminal h2,.terminal h3,.terminal h4,.terminal h5,.terminal h6{line-height:1.2em}.cmd .clipboard{position:absolute;height:16px;left:-6px;width:5px;background:transparent;border:0;color:transparent;outline:0;padding:0;resize:none;z-index:0;overflow:hidden}.terminal .error{color:red}.terminal{padding:10px;position:relative;overflow:auto}.cmd{padding:0;height:1.3em;position:relative}.terminal .inverted,.cmd .inverted,.cmd .cursor.blink{background-color:#aaa;color:#000}.cmd .cursor.blink{-webkit-animation:terminal-blink 1s infinite steps(1,start);-moz-animation:terminal-blink 1s infinite steps(1,start);-ms-animation:terminal-blink 1s infinite steps(1,start);animation:terminal-blink 1s infinite steps(1,start)}@-webkit-keyframes terminal-blink{0,100%{background-color:#000;color:#aaa}50%{background-color:#bbb;color:#000}}@-ms-keyframes terminal-blink{0,100%{background-color:#000;color:#aaa}50%{background-color:#bbb;color:#000}}@-moz-keyframes terminal-blink{0,100%{background-color:#000;color:#aaa}50%{background-color:#bbb;color:#000}}@keyframes terminal-blink{0,100%{background-color:#000;color:#aaa}50%{background-color:#bbb;color:#000}}.terminal .terminal-output div div,.cmd .prompt{display:block;line-height:14px;height:auto}.cmd .prompt{float:left}.terminal,.cmd{font-family:monospace;color:#aaa;background-color:#000;font-size:12px;line-height:14px}.terminal-output>div{min-height:14px}.terminal .terminal-output div span{display:inline-block}.cmd span{float:left}.terminal-output span,.terminal-output a,.cmd div,.cmd span,.terminal td,.terminal pre,.terminal h1,.terminal h2,.terminal h3,.terminal h4,.terminal h5,.terminal h6{-webkit-touch-callout:initial;-webkit-user-select:initial;-khtml-user-select:initial;-moz-user-select:initial;-ms-user-select:initial;user-select:initial}.terminal,.terminal-output,.terminal-output div{-webkit-touch-callout:none;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}@-moz-document url-prefix(){.terminal,.terminal-output,.terminal-output div{-webkit-touch-callout:initial;-webkit-user-select:initial;-khtml-user-select:initial;-moz-user-select:initial;-ms-user-select:initial;user-select:initial}}.terminal table{border-collapse:collapse}.terminal td{border:1px solid #aaa}.terminal h1::-moz-selection,.terminal h2::-moz-selection,.terminal h3::-moz-selection,.terminal h4::-moz-selection,.terminal h5::-moz-selection,.terminal h6::-moz-selection,.terminal pre::-moz-selection,.terminal td::-moz-selection,.terminal .terminal-output div div::-moz-selection,.terminal .terminal-output div span::-moz-selection,.terminal .terminal-output div div a::-moz-selection,.cmd div::-moz-selection,.cmd>span::-moz-selection,.cmd .prompt span::-moz-selection{background-color:#aaa;color:#000}.terminal h1::selection,.terminal h2::selection,.terminal h3::selection,.terminal h4::selection,.terminal h5::selection,.terminal h6::selection,.terminal pre::selection,.terminal td::selection,.terminal .terminal-output div div::selection,.terminal .terminal-output div div a::selection,.terminal .terminal-output div span::selection,.cmd div::selection,.cmd>span::selection,.cmd .prompt span::selection{background-color:#aaa;color:#000}.terminal .terminal-output div.error,.terminal .terminal-output div.error div{color:red}.tilda{position:fixed;top:0;left:0;width:100%;z-index:1100}.clear{clear:both}.terminal a{color:#0f60ff}.terminal a:hover{color:red} | ||||
							
								
								
									
										113
									
								
								static/term/pkg/jquery.terminal/js/ascii_table.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,113 @@ | ||||
| /**@license | ||||
|  *       __ _____                     ________                              __ | ||||
|  *      / // _  /__ __ _____ ___ __ _/__  ___/__ ___ ______ __ __  __ ___  / / | ||||
|  *  __ / // // // // // _  // _// // / / // _  // _//     // //  \/ // _ \/ / | ||||
|  * /  / // // // // // ___// / / // / / // ___// / / / / // // /\  // // / /__ | ||||
|  * \___//____ \\___//____//_/ _\_  / /_//____//_/ /_/ /_//_//_/ /_/ \__\_\___/ | ||||
|  *           \/              /____/ | ||||
|  * http://terminal.jcubic.pl | ||||
|  * | ||||
|  * utility that renders simple ascii table, like the one from mysql cli tool | ||||
|  * it was first created for leash shell https://leash.jcubic.pl | ||||
|  * | ||||
|  * usage: | ||||
|  * | ||||
|  *        var arr = [[1,2,3,4,5], ["lorem", "ipsum", "dolor", "sit", "amet"]]; | ||||
|  *        term.echo(ascii_table(arr)); | ||||
|  *        // or | ||||
|  *        term.echo(ascii_table(arr, true)); // this will render first row as header | ||||
|  * | ||||
|  * Copyright (c) 2018-2019 Jakub Jankiewicz <https://jcubic.pl/me> | ||||
|  * Released under the MIT license | ||||
|  * | ||||
|  */ | ||||
| /* global define, module, global, wcwidth, require */ | ||||
| (function(factory) { | ||||
|     var root = typeof window !== 'undefined' ? window : global; | ||||
|     if (typeof define === 'function' && define.amd) { | ||||
|         // AMD. Register as an anonymous module. | ||||
|         define(['wcwidth'], function(wcwidth) { | ||||
|             return (root.ascii_table = factory(wcwidth)); | ||||
|         }); | ||||
|     } else if (typeof module === 'object' && module.exports) { | ||||
|         // Node/CommonJS | ||||
|         module.exports = factory(require('wcwidth')); | ||||
|     } else { | ||||
|         root.ascii_table = factory(root.wcwidth); | ||||
|     } | ||||
| })(function(wcwidth, undefined) { | ||||
|     var strlen = (function() { | ||||
|         if (typeof wcwidth === 'undefined') { | ||||
|             return function(string) { | ||||
|                 return string.length; | ||||
|             }; | ||||
|         } else { | ||||
|             return wcwidth; | ||||
|         } | ||||
|     })(); | ||||
|     function ascii_table(array, header) { | ||||
|         if (!array.length) { | ||||
|             return ''; | ||||
|         } | ||||
|         for (var i = array.length - 1; i >= 0; i--) { | ||||
|             var row = array[i]; | ||||
|             var stacks = []; | ||||
|             for (var j = 0; j < row.length; j++) { | ||||
|                 var new_lines = row[j].toString().replace(/\r/g).split("\n"); | ||||
|                 row[j] = new_lines.shift(); | ||||
|                 stacks.push(new_lines); | ||||
|             } | ||||
|             var stack_lengths = stacks.map(function(column) { | ||||
|                 return column.length; | ||||
|             }); | ||||
|             var new_rows_count = Math.max.apply(Math, stack_lengths); | ||||
|             for (var k = new_rows_count - 1; k >= 0; k--) { | ||||
|                 array.splice(i + 1, 0, stacks.map(function(column) { | ||||
|                     return column[k] || ""; | ||||
|                 })); | ||||
|             } | ||||
|         } | ||||
|         var lengths = array[0].map(function(_, i) { | ||||
|             var col = array.map(function(row) { | ||||
|                 if (row[i] != undefined) { | ||||
|                     var len = strlen(row[i]); | ||||
|                     if (row[i].match(/\t/g)) { | ||||
|                         // tab is 4 spaces | ||||
|                         len += row[i].match(/\t/g).length*3; | ||||
|                     } | ||||
|                     return len; | ||||
|                 } else { | ||||
|                     return 0; | ||||
|                 } | ||||
|             }); | ||||
|             return Math.max.apply(Math, col); | ||||
|         }); | ||||
|         // column padding | ||||
|         array = array.map(function(row) { | ||||
|             return '| ' + row.map(function(item, i) { | ||||
|                 var size = strlen(item); | ||||
|                 if (item.match(/\t/g)) { | ||||
|                     // tab is 4 spaces | ||||
|                     size += item.match(/\t/g).length*3; | ||||
|                 } | ||||
|                 if (size < lengths[i]) { | ||||
|                     item += new Array(lengths[i] - size + 1).join(' '); | ||||
|                 } | ||||
|                 return item; | ||||
|             }).join(' | ') + ' |'; | ||||
|         }); | ||||
|         array = array.map(function(line) { | ||||
|             return line.replace(/&(?![^;]+;)/g, '&'); | ||||
|         }); | ||||
|         var sep = '+' + lengths.map(function(length) { | ||||
|             return new Array(length + 3).join('-'); | ||||
|         }).join('+') + '+'; | ||||
|         if (header) { | ||||
|             return sep + '\n' + array[0] + '\n' + sep + '\n' + | ||||
|                 array.slice(1).join('\n') + '\n' + sep; | ||||
|         } else { | ||||
|             return sep + '\n' + array.join('\n') + '\n' + sep; | ||||
|         } | ||||
|     } | ||||
|     return ascii_table; | ||||
| }); | ||||
							
								
								
									
										69
									
								
								static/term/pkg/jquery.terminal/js/dterm.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | ||||
| /*! | ||||
|  * Example plugin using JQuery Terminal Emulator | ||||
|  * Copyright (C) 2010-2016 Jakub Jankiewicz <http://jcubic.pl> | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  * | ||||
|  */ | ||||
| (function($) { | ||||
|     $.extend_if_has = function(desc, source, array) { | ||||
|         for (var i=array.length;i--;) { | ||||
|             if (typeof source[array[i]] != 'undefined') { | ||||
|                 desc[array[i]] = source[array[i]]; | ||||
|             } | ||||
|         } | ||||
|         return desc; | ||||
|     }; | ||||
|     $.fn.dterm = function(interpreter, options) { | ||||
|         var op = $.extend_if_has({}, options,  | ||||
|                                    ['greetings', 'prompt', 'onInit', | ||||
|                                     'onExit', 'clear', | ||||
|                                     'login', 'name', 'exit']); | ||||
|         op.enabled = false; | ||||
|         var terminal = this.terminal(interpreter, op).css('overflow', 'hidden'); | ||||
|         if (!options.title) { | ||||
|             options.title = 'JQuery Terminal Emulator'; | ||||
|         } | ||||
|         if (options.logoutOnClose) { | ||||
|             options.close = function(e, ui) { | ||||
|                 terminal.logout(); | ||||
|                 terminal.clear(); | ||||
|             }; | ||||
|         } else { | ||||
|             options.close = function(e, ui) { | ||||
|                 terminal.disable(); | ||||
|             }; | ||||
|         } | ||||
|         var self = this; | ||||
|         this.dialog($.extend(options, { | ||||
|             resizeStop: function(e, ui) { | ||||
|                 var content = self.find('.ui-dialog-content'); | ||||
|                 terminal.resize(content.width(), content.height()); | ||||
|             }, | ||||
|             open: function(e, ui) { | ||||
|                 terminal.focus(); | ||||
|                 terminal.resize(); | ||||
|             }, | ||||
|             show: 'fade', | ||||
|             closeOnEscape: false | ||||
|         })); | ||||
|         self.terminal = terminal; | ||||
|         return self; | ||||
|     }; | ||||
| })(jQuery); | ||||
							
								
								
									
										2
									
								
								static/term/pkg/jquery.terminal/js/jquery.mousewheel-min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | ||||
| (function(c){function g(a){var b=a||window.event,i=[].slice.call(arguments,1),e=0,h=0,f=0;a=c.event.fix(b);a.type="mousewheel";if(b.wheelDelta)e=b.wheelDelta/120;if(b.detail)e=-b.detail/3;f=e;if(b.axis!==undefined&&b.axis===b.HORIZONTAL_AXIS){f=0;h=-1*e}if(b.wheelDeltaY!==undefined)f=b.wheelDeltaY/120;if(b.wheelDeltaX!==undefined)h=-1*b.wheelDeltaX/120;i.unshift(a,e,h,f);return(c.event.dispatch||c.event.handle).apply(this,i)}var d=["DOMMouseScroll","mousewheel"];if(c.event.fixHooks)for(var j=d.length;j;)c.event.fixHooks[d[--j]]= | ||||
| c.event.mouseHooks;c.event.special.mousewheel={setup:function(){if(this.addEventListener)for(var a=d.length;a;)this.addEventListener(d[--a],g,false);else this.onmousewheel=g},teardown:function(){if(this.removeEventListener)for(var a=d.length;a;)this.removeEventListener(d[--a],g,false);else this.onmousewheel=null}};c.fn.extend({mousewheel:function(a){return a?this.bind("mousewheel",a):this.trigger("mousewheel")},unmousewheel:function(a){return this.unbind("mousewheel",a)}})})(jQuery); | ||||
							
								
								
									
										37
									
								
								static/term/pkg/jquery.terminal/js/jquery.terminal.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										317
									
								
								static/term/pkg/jquery.terminal/js/unix_formatting.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,317 @@ | ||||
| /**@license | ||||
|  *       __ _____                     ________                              __ | ||||
|  *      / // _  /__ __ _____ ___ __ _/__  ___/__ ___ ______ __ __  __ ___  / / | ||||
|  *  __ / // // // // // _  // _// // / / // _  // _//     // //  \/ // _ \/ / | ||||
|  * /  / // // // // // ___// / / // / / // ___// / / / / // // /\  // // / /__ | ||||
|  * \___//____ \\___//____//_/ _\_  / /_//____//_/ /_/ /_//_//_/ /_/ \__\_\___/ | ||||
|  *           \/              /____/ | ||||
|  * http://terminal.jcubic.pl | ||||
|  * | ||||
|  * This is example of how to create custom formatter for jQuery Terminal | ||||
|  * | ||||
|  * Copyright (c) 2014-2016 Jakub Jankiewicz <http://jcubic.pl> | ||||
|  * Released under the MIT license | ||||
|  * | ||||
|  */ | ||||
| (function($) { | ||||
|     if (!$.terminal) { | ||||
|         throw new Error('$.terminal is not defined'); | ||||
|     } | ||||
|     // --------------------------------------------------------------------- | ||||
|     // :: Replace overtyping (from man) formatting with terminal formatting | ||||
|     // --------------------------------------------------------------------- | ||||
|     $.terminal.overtyping = function(string) { | ||||
|         return string.replace(/((?:_\x08.|.\x08_)+)/g, function(full, g) { | ||||
|             var striped = full.replace(/_x08|\x08_|_\u0008|\u0008_/g, ''); | ||||
|             return '[[u;;]' + striped + ']'; | ||||
|         }).replace(/((?:.\x08.)+)/g, function(full, g) { | ||||
|             return '[[b;#fff;]' + full.replace(/(.)(?:\x08|\u0008)(.)/g, | ||||
|                                                function(full, g1, g2) { | ||||
|                                                    return g2; | ||||
|                                                }) + ']'; | ||||
|         }); | ||||
|     }; | ||||
|     // --------------------------------------------------------------------- | ||||
|     // :: Html colors taken from ANSI formatting in Linux Terminal | ||||
|     // --------------------------------------------------------------------- | ||||
|     $.terminal.ansi_colors = { | ||||
|         normal: { | ||||
|             black: '#000', | ||||
|             red: '#A00', | ||||
|             green: '#008400', | ||||
|             yellow: '#A50', | ||||
|             blue: '#00A', | ||||
|             magenta: '#A0A', | ||||
|             cyan: '#0AA', | ||||
|             white: '#AAA' | ||||
|         }, | ||||
|         faited: { | ||||
|             black: '#000', | ||||
|             red: '#640000', | ||||
|             green: '#006100', | ||||
|             yellow: '#737300', | ||||
|             blue: '#000087', | ||||
|             magenta: '#650065', | ||||
|             cyan: '#008787', | ||||
|             white: '#818181' | ||||
|         }, | ||||
|         bold: { | ||||
|             black: '#000', | ||||
|             red: '#F55', | ||||
|             green: '#44D544', | ||||
|             yellow: '#FF5', | ||||
|             blue: '#55F', | ||||
|             magenta: '#F5F', | ||||
|             cyan: '#5FF', | ||||
|             white: '#FFF' | ||||
|         }, | ||||
|         // XTerm 8-bit pallete | ||||
|         palette: [ | ||||
|             '#000000', '#AA0000', '#00AA00', '#AA5500', '#0000AA', '#AA00AA', | ||||
|             '#00AAAA', '#AAAAAA', '#555555', '#FF5555', '#55FF55', '#FFFF55', | ||||
|             '#5555FF', '#FF55FF', '#55FFFF', '#FFFFFF', '#000000', '#00005F', | ||||
|             '#000087', '#0000AF', '#0000D7', '#0000FF', '#005F00', '#005F5F', | ||||
|             '#005F87', '#005FAF', '#005FD7', '#005FFF', '#008700', '#00875F', | ||||
|             '#008787', '#0087AF', '#0087D7', '#0087FF', '#00AF00', '#00AF5F', | ||||
|             '#00AF87', '#00AFAF', '#00AFD7', '#00AFFF', '#00D700', '#00D75F', | ||||
|             '#00D787', '#00D7AF', '#00D7D7', '#00D7FF', '#00FF00', '#00FF5F', | ||||
|             '#00FF87', '#00FFAF', '#00FFD7', '#00FFFF', '#5F0000', '#5F005F', | ||||
|             '#5F0087', '#5F00AF', '#5F00D7', '#5F00FF', '#5F5F00', '#5F5F5F', | ||||
|             '#5F5F87', '#5F5FAF', '#5F5FD7', '#5F5FFF', '#5F8700', '#5F875F', | ||||
|             '#5F8787', '#5F87AF', '#5F87D7', '#5F87FF', '#5FAF00', '#5FAF5F', | ||||
|             '#5FAF87', '#5FAFAF', '#5FAFD7', '#5FAFFF', '#5FD700', '#5FD75F', | ||||
|             '#5FD787', '#5FD7AF', '#5FD7D7', '#5FD7FF', '#5FFF00', '#5FFF5F', | ||||
|             '#5FFF87', '#5FFFAF', '#5FFFD7', '#5FFFFF', '#870000', '#87005F', | ||||
|             '#870087', '#8700AF', '#8700D7', '#8700FF', '#875F00', '#875F5F', | ||||
|             '#875F87', '#875FAF', '#875FD7', '#875FFF', '#878700', '#87875F', | ||||
|             '#878787', '#8787AF', '#8787D7', '#8787FF', '#87AF00', '#87AF5F', | ||||
|             '#87AF87', '#87AFAF', '#87AFD7', '#87AFFF', '#87D700', '#87D75F', | ||||
|             '#87D787', '#87D7AF', '#87D7D7', '#87D7FF', '#87FF00', '#87FF5F', | ||||
|             '#87FF87', '#87FFAF', '#87FFD7', '#87FFFF', '#AF0000', '#AF005F', | ||||
|             '#AF0087', '#AF00AF', '#AF00D7', '#AF00FF', '#AF5F00', '#AF5F5F', | ||||
|             '#AF5F87', '#AF5FAF', '#AF5FD7', '#AF5FFF', '#AF8700', '#AF875F', | ||||
|             '#AF8787', '#AF87AF', '#AF87D7', '#AF87FF', '#AFAF00', '#AFAF5F', | ||||
|             '#AFAF87', '#AFAFAF', '#AFAFD7', '#AFAFFF', '#AFD700', '#AFD75F', | ||||
|             '#AFD787', '#AFD7AF', '#AFD7D7', '#AFD7FF', '#AFFF00', '#AFFF5F', | ||||
|             '#AFFF87', '#AFFFAF', '#AFFFD7', '#AFFFFF', '#D70000', '#D7005F', | ||||
|             '#D70087', '#D700AF', '#D700D7', '#D700FF', '#D75F00', '#D75F5F', | ||||
|             '#D75F87', '#D75FAF', '#D75FD7', '#D75FFF', '#D78700', '#D7875F', | ||||
|             '#D78787', '#D787AF', '#D787D7', '#D787FF', '#D7AF00', '#D7AF5F', | ||||
|             '#D7AF87', '#D7AFAF', '#D7AFD7', '#D7AFFF', '#D7D700', '#D7D75F', | ||||
|             '#D7D787', '#D7D7AF', '#D7D7D7', '#D7D7FF', '#D7FF00', '#D7FF5F', | ||||
|             '#D7FF87', '#D7FFAF', '#D7FFD7', '#D7FFFF', '#FF0000', '#FF005F', | ||||
|             '#FF0087', '#FF00AF', '#FF00D7', '#FF00FF', '#FF5F00', '#FF5F5F', | ||||
|             '#FF5F87', '#FF5FAF', '#FF5FD7', '#FF5FFF', '#FF8700', '#FF875F', | ||||
|             '#FF8787', '#FF87AF', '#FF87D7', '#FF87FF', '#FFAF00', '#FFAF5F', | ||||
|             '#FFAF87', '#FFAFAF', '#FFAFD7', '#FFAFFF', '#FFD700', '#FFD75F', | ||||
|             '#FFD787', '#FFD7AF', '#FFD7D7', '#FFD7FF', '#FFFF00', '#FFFF5F', | ||||
|             '#FFFF87', '#FFFFAF', '#FFFFD7', '#FFFFFF', '#080808', '#121212', | ||||
|             '#1C1C1C', '#262626', '#303030', '#3A3A3A', '#444444', '#4E4E4E', | ||||
|             '#585858', '#626262', '#6C6C6C', '#767676', '#808080', '#8A8A8A', | ||||
|             '#949494', '#9E9E9E', '#A8A8A8', '#B2B2B2', '#BCBCBC', '#C6C6C6', | ||||
|             '#D0D0D0', '#DADADA', '#E4E4E4', '#EEEEEE' | ||||
|         ] | ||||
|     }; | ||||
|     // --------------------------------------------------------------------- | ||||
|     // :: Replace ANSI formatting with terminal formatting | ||||
|     // --------------------------------------------------------------------- | ||||
|     $.terminal.from_ansi = (function() { | ||||
|         var color_list = { | ||||
|             30: 'black', | ||||
|             31: 'red', | ||||
|             32: 'green', | ||||
|             33: 'yellow', | ||||
|             34: 'blue', | ||||
|             35: 'magenta', | ||||
|             36: 'cyan', | ||||
|             37: 'white', | ||||
|  | ||||
|             39: 'inherit' // default color | ||||
|         }; | ||||
|         var background_list = { | ||||
|             40: 'black', | ||||
|             41: 'red', | ||||
|             42: 'green', | ||||
|             43: 'yellow', | ||||
|             44: 'blue', | ||||
|             45: 'magenta', | ||||
|             46: 'cyan', | ||||
|             47: 'white', | ||||
|  | ||||
|             49: 'transparent' // default background | ||||
|         }; | ||||
|         function format_ansi(code) { | ||||
|             var controls = code.split(';'); | ||||
|             var num; | ||||
|             var faited = false; | ||||
|             var reverse = false; | ||||
|             var bold = false; | ||||
|             var styles = []; | ||||
|             var output_color = ''; | ||||
|             var output_background = ''; | ||||
|             var _8bit_color = false; | ||||
|             var _8bit_background = false; | ||||
|             var process_8bit = false; | ||||
|             var palette = $.terminal.ansi_colors.palette; | ||||
|             for(var i in controls) { | ||||
|                 if (controls.hasOwnProperty(i)) { | ||||
|                     num = parseInt(controls[i], 10); | ||||
|                     if (process_8bit && (_8bit_background || _8bit_color)) { | ||||
|                         if (_8bit_color && palette[num]) { | ||||
|                             output_color = palette[num]; | ||||
|                         } | ||||
|                         if (_8bit_background && palette[num]) { | ||||
|                             output_background = palette[num]; | ||||
|                         } | ||||
|                     } else { | ||||
|                         switch(num) { | ||||
|                         case 1: | ||||
|                             styles.push('b'); | ||||
|                             bold = true; | ||||
|                             faited = false; | ||||
|                             break; | ||||
|                         case 4: | ||||
|                             styles.push('u'); | ||||
|                             break; | ||||
|                         case 3: | ||||
|                             styles.push('i'); | ||||
|                             break; | ||||
|                         case 5: | ||||
|                             process_8bit = true; | ||||
|                             break; | ||||
|                         case 38: | ||||
|                             _8bit_color = true; | ||||
|                             break; | ||||
|                         case 48: | ||||
|                             _8bit_background = true; | ||||
|                             break; | ||||
|                         case 2: | ||||
|                             faited = true; | ||||
|                             bold = false; | ||||
|                             break; | ||||
|                         case 7: | ||||
|                             reverse = true; | ||||
|                             break; | ||||
|                         default: | ||||
|                             if (controls.indexOf('5') == -1) { | ||||
|                                 if (color_list[num]) { | ||||
|                                     output_color = color_list[num]; | ||||
|                                 } | ||||
|                                 if (background_list[num]) { | ||||
|                                     output_background = background_list[num]; | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             if (reverse) { | ||||
|                 if (output_color || output_background) { | ||||
|                     var tmp = output_background; | ||||
|                     output_background = output_color; | ||||
|                     output_color = tmp; | ||||
|                 } else { | ||||
|                     output_color = 'black'; | ||||
|                     output_background = 'white'; | ||||
|                 } | ||||
|             } | ||||
|             var colors, color, background, backgrounds; | ||||
|             if (bold) { | ||||
|                 colors = backgrounds = $.terminal.ansi_colors.bold; | ||||
|             } else if (faited) { | ||||
|                 colors = backgrounds = $.terminal.ansi_colors.faited; | ||||
|             } else { | ||||
|                 colors = backgrounds = $.terminal.ansi_colors.normal; | ||||
|             } | ||||
|             if (_8bit_color) { | ||||
|                 color = output_color; | ||||
|             } else if (output_color == 'inherit') { | ||||
|                 color = output_color; | ||||
|             } else { | ||||
|                 color = colors[output_color]; | ||||
|             } | ||||
|             if (_8bit_background) { | ||||
|                 background = output_background; | ||||
|             } else if (output_background == 'transparent') { | ||||
|                 background = output_background; | ||||
|             } else { | ||||
|                 background = backgrounds[output_background]; | ||||
|             } | ||||
|             return [styles.join(''), color, background]; | ||||
|         } | ||||
|         return function(input) { | ||||
|             //merge multiple codes | ||||
|             /*input = input.replace(/((?:\x1B\[[0-9;]*[A-Za-z])*)/g, function(group) { | ||||
|                 return group.replace(/m\x1B\[/g, ';'); | ||||
|             });*/ | ||||
|             var splitted = input.split(/(\x1B\[[0-9;]*[A-Za-z])/g); | ||||
|             if (splitted.length == 1) { | ||||
|                 return input; | ||||
|             } | ||||
|             var output = []; | ||||
|             //skip closing at the begining | ||||
|             if (splitted.length > 3) { | ||||
|                 var str = splitted.slice(0,3).join(''); | ||||
|                 if (str == '[0m' || str == '[m') { | ||||
|                     splitted = splitted.slice(3); | ||||
|                 } | ||||
|             } | ||||
|             var next, prev_color, prev_background, code, match; | ||||
|             var inside = false; | ||||
|             for (var i=0; i<splitted.length; ++i) { | ||||
|                 match = splitted[i].match(/^\x1B\[([0-9;]*)([A-Za-z])$/); | ||||
|                 if (match) { | ||||
|                     switch (match[2]) { | ||||
|                     case 'm': | ||||
|                         if (match[1] !== '0') { | ||||
|                             code = format_ansi(match[1]); | ||||
|                         } | ||||
|                         if (inside) { | ||||
|                             output.push(']'); | ||||
|                             if (match[1] === '0' || match[1] === '') { | ||||
|                                 //just closing | ||||
|                                 inside = false; | ||||
|                                 prev_color = prev_background = ''; | ||||
|                             } else { | ||||
|                                 // someone forget to close - move to next | ||||
|                                 code[1] = code[1] || prev_color; | ||||
|                                 code[2] = code[2] || prev_background; | ||||
|                                 output.push('[[' + code.join(';') + ']'); | ||||
|                                 // store colors to next use | ||||
|                                 if (code[1]) { | ||||
|                                     prev_color = code[1]; | ||||
|                                 } | ||||
|                                 if (code[2]) { | ||||
|                                     prev_background = code[2]; | ||||
|                                 } | ||||
|                             } | ||||
|                         } else { | ||||
|                             if (match[1] != '0') { | ||||
|                                 inside = true; | ||||
|                                 code[1] = code[1] || prev_color; | ||||
|                                 code[2] = code[2] || prev_background; | ||||
|                                 output.push('[[' + code.join(';') + ']'); | ||||
|                                 // store colors to next use | ||||
|                                 if (code[1]) { | ||||
|                                     prev_color = code[1]; | ||||
|                                 } | ||||
|                                 if (code[2]) { | ||||
|                                     prev_background = code[2]; | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                         break; | ||||
|                     } | ||||
|                 } else { | ||||
|                     output.push(splitted[i]); | ||||
|                 } | ||||
|             } | ||||
|             if (inside) { | ||||
|                 output.push(']'); | ||||
|             } | ||||
|             return output.join(''); //.replace(/\[\[[^\]]+\]\]/g, ''); | ||||
|         }; | ||||
|     })(); | ||||
|     $.terminal.defaults.formatters.push($.terminal.overtyping); | ||||
|     $.terminal.defaults.formatters.push($.terminal.from_ansi); | ||||
| })(jQuery); | ||||
							
								
								
									
										48
									
								
								static/term/pkg/jquery.terminal/js/xml_formatting.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | ||||
| /**@license | ||||
|  *       __ _____                     ________                              __ | ||||
|  *      / // _  /__ __ _____ ___ __ _/__  ___/__ ___ ______ __ __  __ ___  / / | ||||
|  *  __ / // // // // // _  // _// // / / // _  // _//     // //  \/ // _ \/ / | ||||
|  * /  / // // // // // ___// / / // / / // ___// / / / / // // /\  // // / /__ | ||||
|  * \___//____ \\___//____//_/ _\_  / /_//____//_/ /_/ /_//_//_/ /_/ \__\_\___/ | ||||
|  *           \/              /____/ | ||||
|  * http://terminal.jcubic.pl | ||||
|  * | ||||
|  * This is example of how to create custom formatter for jQuery Terminal | ||||
|  * | ||||
|  * Copyright (c) 2014-2016 Jakub Jankiewicz <http://jcubic.pl> | ||||
|  * Released under the MIT license | ||||
|  * | ||||
|  */ | ||||
| (function($) { | ||||
|     if (!$.terminal) { | ||||
|         throw new Error('$.terminal is not defined'); | ||||
|     } | ||||
|     // this formatter allow to echo xml where tags are colors like: | ||||
|     // <red>hello <navy>blue</navy> world</red> | ||||
|     $.terminal.defaults.formatters.push(function(string) { | ||||
|         var stack = []; | ||||
|         var output = []; | ||||
|         var parts = string.split(/(<\/?[a-zA-Z]+>)/); | ||||
|         for (var i=0; i<parts.length; ++i) { | ||||
|             if (parts[i][0] == '<') { | ||||
|                 if (parts[i][1] == '/') { | ||||
|                     if (stack.length) { | ||||
|                         stack.pop(); | ||||
|                     } | ||||
|                 } else { | ||||
|                     stack.push(parts[i].replace(/^<|>$/g, '')); | ||||
|                 } | ||||
|             } else { | ||||
|                 if (stack.length) { | ||||
|                     // top of the stack | ||||
|                     output.push('[[;' + stack[stack.length-1] + ';]'); | ||||
|                 } | ||||
|                 output.push(parts[i]); | ||||
|                 if (stack.length) { | ||||
|                      output.push(']'); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return output.join(''); | ||||
|     }); | ||||
| })(jQuery); | ||||
							
								
								
									
										688
									
								
								static/term/pkg/jquery.terminal/spec/terminalSpec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,688 @@ | ||||
| if (typeof window === 'undefined') { | ||||
|     var node = true; | ||||
|     var jsdom = require("jsdom"); | ||||
|     global.document = jsdom.jsdom(); | ||||
|     global.window = global.document.parentWindow; | ||||
|     var navigator = {userAgent: "node-js", platform: "Linux i686"}; | ||||
|     global.window.navigator = global.navigator = navigator; | ||||
|     global.jQuery = global.$ = require("jquery"); | ||||
|     require('../js/jquery.terminal-src'); | ||||
|     require('../js/unix_formatting'); | ||||
| } | ||||
| describe('Terminal utils', function() { | ||||
|     var command = 'test "foo bar" baz /^asd [x]/ str\\ str 10 1e10'; | ||||
|     var args = '"foo bar" baz /^asd [x]/ str\\ str 10 1e10'; | ||||
|     describe('$.terminal.split_arguments', function() { | ||||
|         it('should create array of arguments', function() { | ||||
|             expect($.terminal.split_arguments(args)).toEqual([ | ||||
|                     'foo bar', | ||||
|                     'baz', | ||||
|                     '/^asd [x]/', | ||||
|                     'str str', | ||||
|                     '10', | ||||
|                     '1e10' | ||||
|             ]); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.parse_arguments', function() { | ||||
|         it('should create array of arguments and convert types', function() { | ||||
|             expect($.terminal.parse_arguments(args)).toEqual([ | ||||
|                     'foo bar', | ||||
|                     'baz', | ||||
|                     /^asd [x]/, | ||||
|                     'str str', | ||||
|                     10, | ||||
|                     1e10 | ||||
|             ]); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.split_command', function() { | ||||
|         it('Should split command', function() { | ||||
|             var cmd = jQuery.terminal.split_command(command); | ||||
|             expect(cmd).toEqual({ | ||||
|                 command: command, | ||||
|                 name: 'test', | ||||
|                 args: [ | ||||
|                     'foo bar', | ||||
|                     'baz', | ||||
|                     '/^asd [x]/', | ||||
|                     'str str', | ||||
|                     '10', | ||||
|                     '1e10' | ||||
|                 ], | ||||
|                 rest: '"foo bar" baz /^asd [x]/ str\\ str 10 1e10' | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.parse_command', function() { | ||||
|         it('should split and parse command', function() { | ||||
|             var cmd = jQuery.terminal.parse_command(command); | ||||
|             expect(cmd).toEqual({ | ||||
|                 command: command, | ||||
|                 name: 'test', | ||||
|                 args: [ | ||||
|                     'foo bar', | ||||
|                     'baz', | ||||
|                     /^asd [x]/, | ||||
|                     'str str', | ||||
|                     10, | ||||
|                     1e10 | ||||
|                 ], | ||||
|                 rest: '"foo bar" baz /^asd [x]/ str\\ str 10 1e10' | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
|     var ansi_string = '\x1b[2;31;46mFoo\x1b[1;3;4;32;45mBar\x1b[0m\x1b[7mBaz'; | ||||
|     describe('$.terminal.from_ansi', function() { | ||||
|         it('should convert ansi to terminal formatting', function() { | ||||
|             var string = $.terminal.from_ansi(ansi_string); | ||||
|             expect(string).toEqual('[[;#640000;#008787]Foo][[biu;#44D544;#F5F]'+ | ||||
|                                    'Bar][[;#000;#AAA]Baz]'); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.overtyping', function() { | ||||
|         var string = 'HELLO TERMINAL'.replace(/./g, function(chr) { | ||||
|             return chr == ' ' ? chr : chr + '\x08' + chr; | ||||
|         }); | ||||
|         var result = '[[b;#fff;]HELLO] [[b;#fff;]TERMINAL]'; | ||||
|         it('should convert to terminal formatting', function() { | ||||
|             expect($.terminal.overtyping(string)).toEqual(result); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.escape_brackets', function() { | ||||
|         var string = '[[jQuery]] [[Terminal]]'; | ||||
|         var result = '[[jQuery]] [[Terminal]]'; | ||||
|         it('should replace [ and ] with html entities', function() { | ||||
|             expect($.terminal.escape_brackets(string)).toEqual(result); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.encode', function() { | ||||
|         var tags = '<hello> </hello>\t<world> </world>'; | ||||
|         var tags_result = '<hello> </hello>   '+ | ||||
|             ' <world> </world>'; | ||||
|         it('should convert < > space and tabs', function() { | ||||
|             expect($.terminal.encode(tags)).toEqual(tags_result); | ||||
|         }); | ||||
|         var entites = '& & & &64; = ['; | ||||
|         //'& & & &64; = [' | ||||
|         var ent_result = '& & & &64; ='+ | ||||
|             ' &#91'; | ||||
|         it('it should convert & but not when used with entities', function() { | ||||
|             expect($.terminal.encode(entites)).toEqual(ent_result); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.format_split', function() { | ||||
|     }); | ||||
|     describe('$.terminal.is_formatting', function() { | ||||
|  | ||||
|         it('should detect terminal formatting', function() { | ||||
|             var formattings = [ | ||||
|                 '[[;;]Te[xt]', | ||||
|                 '[[;;]Te\\]xt]', | ||||
|                 '[[;;]]', | ||||
|                 '[[gui;;;class]Text]', | ||||
|                 '[[b;#fff;]Text]', | ||||
|                 '[[b;red;blue]Text]']; | ||||
|             var not_formattings = [ | ||||
|                 '[[;;]Text[', | ||||
|                 '[[Text]]', | ||||
|                 '[[Text[[', | ||||
|                 '[[;]Text]', | ||||
|                 'Text]', | ||||
|                 '[[Text', | ||||
|                 '[;;]Text]']; | ||||
|             formattings.forEach(function(formatting) { | ||||
|                 expect($.terminal.is_formatting(formatting)).toEqual(true); | ||||
|             }); | ||||
|             not_formattings.forEach(function(formatting) { | ||||
|                 expect($.terminal.is_formatting(formatting)).toEqual(false); | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.escape_regex', function() { | ||||
|         it('should escape regex special characters', function() { | ||||
|             var safe = "\\\\\\^\\*\\+\\?\\.\\$\\[\\]\\{\\}\\(\\)"; | ||||
|             expect($.terminal.escape_regex('\\^*+?.$[]{}()')).toEqual(safe); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.have_formatting', function() { | ||||
|         var formattings = [ | ||||
|             'some text [[;;]Te[xt] and formatting', | ||||
|             'some text [[;;]Te\\]xt] and formatting', | ||||
|             'some text [[;;]] and formatting', | ||||
|             'some text [[gui;;;class]Text] and formatting', | ||||
|             'some text [[b;#fff;]Text] and formatting', | ||||
|             'some text [[b;red;blue]Text] and formatting']; | ||||
|         var not_formattings = [ | ||||
|             'some text [[;;]Text[ and formatting', | ||||
|             'some text [[Text]] and formatting', | ||||
|             'some text [[Text[[ and formatting', | ||||
|             'some text [[;]Text] and formatting', | ||||
|             'some text Text] and formatting', | ||||
|             'some text [[Text and formatting', | ||||
|             'some text [;;]Text] and formatting']; | ||||
|         it('should detect terminal formatting', function() { | ||||
|             formattings.forEach(function(formatting) { | ||||
|                 expect($.terminal.have_formatting(formatting)).toEqual(true); | ||||
|             }); | ||||
|             not_formattings.forEach(function(formatting) { | ||||
|                 expect($.terminal.have_formatting(formatting)).toEqual(false); | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.valid_color', function() { | ||||
|         it('should mark hex color as valid', function() { | ||||
|             var valid_colors = ['#fff', '#fab', '#ffaacc', 'red', 'blue']; | ||||
|             valid_colors.forEach(function(color) { | ||||
|                 expect($.terminal.valid_color(color)).toBe(true); | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.format', function() { | ||||
|         var format = '[[biugs;#fff;#000]Foo][[i;;;foo]Bar][[ous;;]Baz]'; | ||||
|         it('should create html span tags with style and classes', function() { | ||||
|             var string = $.terminal.format(format); | ||||
|             expect(string).toEqual('<span style="font-weight:bold;text-decorat'+ | ||||
|                                    'ion:underline line-through;font-style:ital'+ | ||||
|                                    'ic;color:#fff;text-shadow:0 0 5px #fff;bac'+ | ||||
|                                    'kground-color:#000" data-text="Foo">Foo</s'+ | ||||
|                                    'pan><span style="font-style:italic;" class'+ | ||||
|                                    '="foo" data-text="Bar">Bar</span><span sty'+ | ||||
|                                    'le="text-decoration:underline line-through'+ | ||||
|                                    ' overline;" data-text="Baz">Baz</span>'); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.strip', function() { | ||||
|         var formatting = '-_-[[biugs;#fff;#000]Foo]-_-[[i;;;foo]Bar]-_-[[ous;;'+ | ||||
|             ']Baz]-_-'; | ||||
|         var result = '-_-Foo-_-Bar-_-Baz-_-'; | ||||
|         it('should remove formatting', function() { | ||||
|             expect($.terminal.strip(formatting)).toEqual(result); | ||||
|         }); | ||||
|     }); | ||||
|     describe('$.terminal.split_equal', function() { | ||||
|         var text = ['[[bui;#fff;]Lorem ipsum dolor sit amet, consectetur adipi', | ||||
|             'scing elit. Nulla sed dolor nisl, in suscipit justo. Donec a enim', | ||||
|             ' et est porttitor semper at vitae augue. Proin at nulla at dui ma', | ||||
|             'ttis mattis. Nam a volutpat ante. Aliquam consequat dui eu sem co', | ||||
|             'nvallis ullamcorper. Nulla suscipit, massa vitae suscipit ornare,', | ||||
|             ' tellus] est [[b;;#f00]consequat nunc, quis blandit elit odio eu ', | ||||
|             'arcu. Nam a urna nec nisl varius sodales. Mauris iaculis tincidun', | ||||
|             't orci id commodo. Aliquam] non magna quis [[i;;]tortor malesuada', | ||||
|             ' aliquam] eget ut lacus. Nam ut vestibulum est. Praesent volutpat', | ||||
|             ' tellus in eros dapibus elementum. Nam laoreet risus non nulla mo', | ||||
|             'llis ac luctus [[ub;#fff;]felis dapibus. Pellentesque mattis elem', | ||||
|             'entum augue non sollicitudin. Nullam lobortis fermentum elit ac m', | ||||
|             'ollis. Nam ac varius risus. Cras faucibus euismod nulla, ac aucto', | ||||
|             'r diam rutrum sit amet. Nulla vel odio erat], ac mattis enim.' | ||||
|         ].join(''); | ||||
|         it('should split text that into equal length chunks', function() { | ||||
|             var cols = [10, 40, 60, 400]; | ||||
|             for (var i=cols.length; i--;) { | ||||
|                 var lines = $.terminal.split_equal(text, cols[i]); | ||||
|                 var success = true; | ||||
|                 for (var j=0; j<lines.length; ++j) { | ||||
|                     if ($.terminal.strip(lines[j]).length > cols[i]) { | ||||
|                         success = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 expect(success).toEqual(true); | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
| }); | ||||
| function support_animations() { | ||||
|     var animation = false, | ||||
|     animationstring = 'animation', | ||||
|     keyframeprefix = '', | ||||
|     domPrefixes = 'Webkit Moz O ms Khtml'.split(' '), | ||||
|     pfx  = '', | ||||
|     elm = document.createElement('div'); | ||||
|     if (elm.style.animationName) { animation = true; } | ||||
|     if (animation === false) { | ||||
|         for (var i = 0; i < domPrefixes.length; i++) { | ||||
|             var name = domPrefixes[i] + 'AnimationName'; | ||||
|             if (elm.style[ name ] !== undefined) { | ||||
|                 pfx = domPrefixes[i]; | ||||
|                 animationstring = pfx + 'Animation'; | ||||
|                 keyframeprefix = '-' + pfx.toLowerCase() + '-'; | ||||
|                 animation = true; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     return animation; | ||||
| } | ||||
| function enter_text(text) { | ||||
|     var e; | ||||
|     var $root = $(document.documentElement || window); | ||||
|     for (var i=0; i<text.length; ++i) { | ||||
|         e = $.Event("keypress"); | ||||
|         e.which = e.keyCode = text.charCodeAt(i); | ||||
|         e.ctrlKey = false; | ||||
|         e.altKey = false; | ||||
|         $root.trigger(e); | ||||
|     } | ||||
| } | ||||
| function shortcut(ctrl, alt, shift, which) { | ||||
|     var e = $.Event("keydown"); | ||||
|     e.ctrlKey = ctrl; | ||||
|     e.altKey = alt; | ||||
|     e.shiftKey = shift; | ||||
|     e.which = e.keyCode = which; | ||||
|     $(document.documentElement || window).trigger(e); | ||||
| } | ||||
| function enter_key() { | ||||
|     shortcut(false, false, false, 13); | ||||
| } | ||||
|  | ||||
| function tests_on_ready() { | ||||
|     describe('Terminal plugin', function() { | ||||
|         describe('terminal create / terminal destroy', function() { | ||||
|             var term = $('<div></div>').appendTo('body').terminal(); | ||||
|             it('should create terminal', function() { | ||||
|                 expect(term.length).toBe(1); | ||||
|             }); | ||||
|             it('should have proper elements', function() { | ||||
|                 expect(term.hasClass('terminal')).toBe(true); | ||||
|                 expect(term.find('.terminal-output').length).toBe(1); | ||||
|                 expect(term.find('.cmd').length).toBe(1); | ||||
|                 var prompt = term.find('.prompt'); | ||||
|                 expect(prompt.length).toBe(1); | ||||
|                 expect(prompt.is('span')).toBe(true); | ||||
|                 expect(prompt.children().length).toBe(1); | ||||
|                 var cursor = term.find('.cursor'); | ||||
|                 expect(cursor.length).toBe(1); | ||||
|                 expect(cursor.is('span')).toBe(true); | ||||
|                 expect(cursor.prev().is('span')).toBe(true); | ||||
|                 expect(cursor.next().is('span')).toBe(true); | ||||
|                 term.focus(true); | ||||
|                 if (support_animations()) { | ||||
|                     expect(cursor.hasClass('blink')).toBe(true); | ||||
|                 } | ||||
|                 expect(term.find('.clipboard').length).toBe(1); | ||||
|             }); | ||||
|             it('should have signature', function() { | ||||
|                 var sig = term.find('.terminal-output div div').map(function() { return $(this).text(); }).get().join('\n'); | ||||
|                 expect(term.signature().replace(/ /g, '\xA0')).toEqual(sig); | ||||
|             }); | ||||
|             it('should have default prompt', function() { | ||||
|                 var prompt = term.find('.prompt'); | ||||
|                 expect(prompt.html()).toEqual("<span>> </span>"); | ||||
|                 expect(prompt.text()).toEqual('>\xA0'); | ||||
|             }); | ||||
|             it('should destroy terminal', function() { | ||||
|                 term.destroy(); | ||||
|                 expect(term.children().length).toBe(0); | ||||
|                 term.remove(); | ||||
|             }); | ||||
|         }); | ||||
|         describe('exec', function() { | ||||
|             var interpreter = { | ||||
|                 foo: function() { | ||||
|                 } | ||||
|             }; | ||||
|              | ||||
|             var term = $('<div></div>').appendTo('body').terminal(interpreter); | ||||
|              | ||||
|             it('should execute function', function() { | ||||
|                 var spy = spyOn(interpreter, 'foo'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 term.exec('foo').then(function() { | ||||
|                     expect(interpreter.foo).toHaveBeenCalled(); | ||||
|                     term.destroy().remove(); | ||||
|                 }); | ||||
|             }); | ||||
|         }); | ||||
|         describe('enter text', function() { | ||||
|             var interpreter = { | ||||
|                 foo: function() { | ||||
|                 } | ||||
|             }; | ||||
|             var term = $('<div></div>').appendTo('body').terminal(interpreter); | ||||
|             it('text should appear and interpreter function should be called', function() { | ||||
|                 term.focus(true); | ||||
|                 var spy = spyOn(interpreter, 'foo'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 enter_text('foo'); | ||||
|                 enter_key(); | ||||
|                 expect(interpreter.foo).toHaveBeenCalled(); | ||||
|                 var last_div = term.find('.terminal-output > div:last-child'); | ||||
|                 expect(last_div.hasClass('command')).toBe(true); | ||||
|                 expect(last_div.children().html()).toEqual('<span>> foo</span>'); | ||||
|                 term.destroy().remove(); | ||||
|             }); | ||||
|         }); | ||||
|         describe('prompt', function() { | ||||
|             var term = $('<div></div>').appendTo('body').terminal($.noop, { | ||||
|                 prompt: '>>> ' | ||||
|             }); | ||||
|             it('should return prompt', function() { | ||||
|                 expect(term.get_prompt()).toEqual('>>> '); | ||||
|                 expect(term.find('.prompt').html()).toEqual('<span>>>> </span>'); | ||||
|             }); | ||||
|             it('should set prompt', function() { | ||||
|                 term.set_prompt('||| '); | ||||
|                 expect(term.get_prompt()).toEqual('||| '); | ||||
|                 expect(term.find('.prompt').html()).toEqual('<span>||| </span>'); | ||||
|                 function prompt(callback) { | ||||
|                     callback('>>> '); | ||||
|                 } | ||||
|                 term.set_prompt(prompt); | ||||
|                 expect(term.get_prompt()).toEqual(prompt); | ||||
|                 expect(term.find('.prompt').html()).toEqual('<span>>>> </span>'); | ||||
|             }); | ||||
|             it('should format prompt', function() { | ||||
|                 var prompt = '<span style="font-weight:bold;text-decoration:underline;color:#fff;" data-text=">>>">>>></span><span> </span>'; | ||||
|                 term.set_prompt('[[ub;#fff;]>>>] '); | ||||
|                 expect(term.find('.prompt').html()).toEqual(prompt); | ||||
|                 term.set_prompt(function(callback) { | ||||
|                     callback('[[ub;#fff;]>>>] '); | ||||
|                 }); | ||||
|                 expect(term.find('.prompt').html()).toEqual(prompt); | ||||
|                 term.destroy().remove(); | ||||
|             }); | ||||
|         }); | ||||
|         describe('cmd plugin', function() { | ||||
|             var term = $('<div></div>').appendTo('body').css('overflow-y', 'scroll').terminal($.noop, { | ||||
|                 name: 'cmd', | ||||
|                 numChars: 150, | ||||
|                 numRows: 20 | ||||
|             }); | ||||
|             var string = ''; | ||||
|             for (var i=term.cols(); i--;) { | ||||
|                 term.insert('M'); | ||||
|             } | ||||
|             var cmd = term.cmd(); | ||||
|             var line = cmd.find('.prompt').next(); | ||||
|             it('text should have 2 lines', function() { | ||||
|                 expect(line.is('div')).toBe(true); | ||||
|                 expect(line.text().length).toBe(term.cols()-2); | ||||
|             }); | ||||
|             it('cmd plugin moving cursor', function() { | ||||
|                 cmd.position(-8, true); | ||||
|                 var before = cmd.find('.prompt').next(); | ||||
|                 var cursor = cmd.find('.cursor'); | ||||
|                 var after = cursor.next(); | ||||
|                 expect(before.is('span')).toBe(true); | ||||
|                 expect(before.text().length).toBe(term.cols()-8); | ||||
|                 expect(after.next().text().length).toBe(2); | ||||
|                 expect(after.text().length).toBe(5); | ||||
|                 expect(cursor.text()).toBe('M'); | ||||
|             }); | ||||
|             it('should remove characters', function() { | ||||
|                 cmd['delete'](-10); | ||||
|                 var before = cmd.find('.prompt').next(); | ||||
|                 var cursor = cmd.find('.cursor'); | ||||
|                 var after = cursor.next(); | ||||
|                 expect(before.text().length).toEqual(term.cols()-8-10); | ||||
|                 cmd['delete'](8); | ||||
|                 expect(cursor.text()).toEqual('\xA0'); | ||||
|                 expect(after.text().length).toEqual(0); | ||||
|             }); | ||||
|             var history = cmd.history() | ||||
|             it('should have one entry in history', function() { | ||||
|                 cmd.purge(); | ||||
|                 term.set_command('something').focus(true); | ||||
|                 enter_key(); | ||||
|                 expect(history.data()).toEqual(['something']); | ||||
|             }); | ||||
|             it('should not add item to history if history is disabled', function() { | ||||
|                 history.disable(); | ||||
|                 term.set_command('something else'); | ||||
|                 enter_key(); | ||||
|                 expect(history.data()).toEqual(['something']); | ||||
|             }); | ||||
|             it('should remove commands from history', function() { | ||||
|                 var spy = spyOn(history, 'purge'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 cmd.purge(); | ||||
|                 expect(history.purge).toHaveBeenCalled(); | ||||
|                 expect(history.data()).toEqual([]); | ||||
|             }); | ||||
|             it('should have name', function() { | ||||
|                 expect(cmd.name()).toEqual('cmd_4'); | ||||
|             }); | ||||
|             it('should return command', function() { | ||||
|                 cmd.set('foo'); | ||||
|                 expect(cmd.get()).toEqual('foo'); | ||||
|             }); | ||||
|             it('should not move position', function() { | ||||
|                 var pos = cmd.position(); | ||||
|                 cmd.insert('bar', true); | ||||
|                 expect(cmd.position()).toEqual(pos); | ||||
|             }); | ||||
|             it('should return $.noop for commands', function() { | ||||
|                 expect($.terminal.active().commands()).toEqual($.noop); | ||||
|             }); | ||||
|             it('should set position', function() { | ||||
|                 cmd.position(0); | ||||
|                 expect(cmd.position()).toEqual(0); | ||||
|             }); | ||||
|             it('should set and remove mask', function() { | ||||
|                 cmd.mask('•'); | ||||
|                 cmd.position(6); | ||||
|                 var before = cmd.find('.prompt').next(); | ||||
|                 expect(before.text()).toEqual('••••••'); | ||||
|                 expect(cmd.get()).toEqual('foobar'); | ||||
|                 cmd.mask(false); | ||||
|                 expect(before.text()).toEqual('foobar'); | ||||
|             }); | ||||
|             it('should execute functions on shortcuts', function() { | ||||
|                 var spy; | ||||
|                 spy = spyOn(cmd, 'position'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 shortcut(true, false, false, 65); // CTRL+A | ||||
|                 expect(cmd.position).toHaveBeenCalled(); | ||||
|                 spy = spyOn(cmd, 'delete'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 shortcut(true, false, false, 75); // CTRL+K | ||||
|                 expect(cmd['delete']).toHaveBeenCalled(); | ||||
|                 spy = spyOn(cmd, 'insert'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 shortcut(true, false, false, 89); // CTRL+Y | ||||
|                 expect(cmd.insert).toHaveBeenCalled(); | ||||
|                 shortcut(true, false, false, 85); // CTRL+U | ||||
|                 expect(cmd.kill_text()).toEqual('foobar'); | ||||
|                 shortcut(true, false, true, 13); | ||||
|                 expect(cmd.find('.prompt').next().text()).toEqual('\xA0'); | ||||
|                 expect(cmd.get()).toEqual('\n'); | ||||
|                 cmd.set(''); | ||||
|                 shortcut(false, false, false, 9); // TAB | ||||
|                 expect(cmd.get()).toEqual('\t'); | ||||
|                 history.enable(); | ||||
|                 cmd.set('foo bar'); | ||||
|                 enter_key(); | ||||
|                 shortcut(false, false, false, 38); // UP ARROW | ||||
|                 expect(cmd.get()).toEqual('foo bar'); | ||||
|                 shortcut(false, false, false, 40); // DOWN ARROW | ||||
|                 expect(cmd.get()).toEqual(''); | ||||
|                 cmd.insert('hello'); | ||||
|                 shortcut(false, false, false, 38); | ||||
|                 shortcut(false, false, false, 40); | ||||
|                 expect(cmd.get()).toEqual('hello'); | ||||
|                 shortcut(true, false, false, 80); // CTRL+P | ||||
|                 expect(cmd.get()).toEqual('foo bar'); | ||||
|                 shortcut(true, false, false, 78); // CTRL+N | ||||
|                 expect(cmd.get()).toEqual('hello'); | ||||
|                 cmd.set('foo bar baz'); | ||||
|                 shortcut(false, false, false, 37); // LEFT ARROW | ||||
|                 expect(cmd.position()).toEqual(10); | ||||
|                 shortcut(true, false, false, 37); // moving by words | ||||
|                 expect(cmd.position()).toEqual(8); | ||||
|                 shortcut(true, false, false, 37); | ||||
|                 expect(cmd.position()).toEqual(4); | ||||
|                 shortcut(true, false, false, 37); | ||||
|                 expect(cmd.position()).toEqual(0); | ||||
|                 shortcut(false, false, false, 39); // RIGHT ARROW | ||||
|                 expect(cmd.position()).toEqual(1); | ||||
|                 shortcut(true, false, false, 39); | ||||
|                 expect(cmd.position()).toEqual(3); | ||||
|                 shortcut(true, false, false, 39); | ||||
|                 expect(cmd.position()).toEqual(7); | ||||
|                 shortcut(true, false, false, 39); | ||||
|                 expect(cmd.position()).toEqual(11); | ||||
|                 shortcut(false, false, false, 36); // HOME | ||||
|                 expect(cmd.position()).toEqual(0); | ||||
|                 shortcut(false, false, false, 35); // END | ||||
|                 expect(cmd.position()).toEqual(cmd.get().length); | ||||
|                 shortcut(true, false, false, 82); // CTRL+R | ||||
|                 expect(cmd.prompt()).toEqual("(reverse-i-search)`': "); | ||||
|                 enter_text('foo'); | ||||
|                 expect(cmd.get()).toEqual('foo bar'); | ||||
|                 shortcut(true, false, false, 71); // CTRL+G | ||||
|                 expect(cmd.get()).toEqual('foo bar baz'); | ||||
|                 cmd.purge(); | ||||
|                 term.destroy(); | ||||
|             }); | ||||
|         }); | ||||
|         function JSONRPCMock(url, object) { | ||||
|             var ajax = $.ajax; | ||||
|             var system = { | ||||
|                 'sdversion': '1.0', | ||||
|                 'name': 'DemoService', | ||||
|                 'address': url, | ||||
|                 // md5('JSONRPCMock') | ||||
|                 'id': 'urn:md5:e1a975ac782ce4ed0a504ceb909abf44', | ||||
|                 'procs': [] | ||||
|             }; | ||||
|             for (var key in object) { | ||||
|                 var proc = { | ||||
|                     name: key | ||||
|                 }; | ||||
|                 if ($.isFunction(object[key])) { | ||||
|                     var re = /function[^\(]+\(([^\)]+)\)/; | ||||
|                     var m = object[key].toString().match(re); | ||||
|                     if (m) { | ||||
|                         proc.params = m[1].split(/\s*,\s*/); | ||||
|                     } | ||||
|                 } | ||||
|                 system.procs.push(proc); | ||||
|             } | ||||
|             $.ajax = function(obj) { | ||||
|                 if (obj.url == url) { | ||||
|                     var defer = $.Deferred(); | ||||
|                     try { | ||||
|                         var req = JSON.parse(obj.data); | ||||
|                         var resp; | ||||
|                         if (req.method == 'system.describe') { | ||||
|                             resp = system; | ||||
|                         } else { | ||||
|                             var error = null; | ||||
|                             var ret = null | ||||
|                             try { | ||||
|                                 ret = object[req.method].apply(null, req.params); | ||||
|                             } catch (e) { | ||||
|                                 error = {message: e.message}; | ||||
|                             } | ||||
|                             resp = { | ||||
|                                 id: req.id, | ||||
|                                 jsonrpc: '1.1', | ||||
|                                 result: ret, | ||||
|                                 error: error | ||||
|                             }; | ||||
|                         } | ||||
|                         resp = JSON.stringify(resp); | ||||
|                         if ($.isFunction(obj.success)) { | ||||
|                             obj.success(resp, 'OK', { | ||||
|                                 getResponseHeader: function(header) { | ||||
|                                     if (header == 'Content-Type') { | ||||
|                                         return 'application/json'; | ||||
|                                     } | ||||
|                                 } | ||||
|                             }); | ||||
|                         } | ||||
|                         defer.resolve(resp); | ||||
|                     } catch (e) { | ||||
|                         throw new Error(e.message); | ||||
|                     } | ||||
|                     return defer.promise(); | ||||
|                 } else { | ||||
|                     return ajax.apply($, arguments); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         var object = { | ||||
|             echo: function(token, str) { | ||||
|                 return str; | ||||
|             }, | ||||
|             login: function(user, password) { | ||||
|                 if (user == 'demo' && password == 'demo') { | ||||
|                     return 'TOKEN'; | ||||
|                 } else { | ||||
|                     return null; | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|         JSONRPCMock('/test', object); | ||||
|         describe('JSON-RPC', function() { | ||||
|             var term = $('<div></div>').appendTo('body').terminal('/test', { | ||||
|                 login: true | ||||
|             }); | ||||
|             it('should call login', function() { | ||||
|                 term.focus(); | ||||
|                 var spy = spyOn(object, 'login'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 term.insert('test'); | ||||
|                 enter_key(); | ||||
|                 term.insert('test'); | ||||
|                 enter_key(); | ||||
|                 var last_div = term.find('.terminal-output > div:last-child'); | ||||
|                 expect(last_div.text()).toEqual('Wrong password try again!'); | ||||
|                 expect(object.login).toHaveBeenCalledWith('test', 'test'); | ||||
|                 term.insert('demo'); | ||||
|                 enter_key(); | ||||
|                 term.insert('demo'); | ||||
|                 enter_key(); | ||||
|                 expect(object.login).toHaveBeenCalledWith('demo', 'demo'); | ||||
|             }); | ||||
|             it('should call a function', function() { | ||||
|                 term.focus(); | ||||
|                 var spy = spyOn(object, 'echo'); | ||||
|                 if (spy.andCallThrough) { | ||||
|                     spy.andCallThrough(); | ||||
|                 } else { | ||||
|                     spy.and.callThrough(); | ||||
|                 } | ||||
|                 term.insert('echo hello'); | ||||
|                 enter_key(); | ||||
|                 expect(object.echo).toHaveBeenCalledWith('TOKEN', 'hello'); | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
| if (node) { | ||||
|     tests_on_ready(); | ||||
| } else { | ||||
|     $(tests_on_ready); | ||||
| } | ||||
|  | ||||