From aca49481104ee9cbd3de61adcc459beac2afe1ad Mon Sep 17 00:00:00 2001 From: Tdxdxoz Date: Sat, 15 Aug 2020 13:05:56 +0800 Subject: [PATCH] =?UTF-8?q?=E5=AE=8C=E6=88=90=E4=B8=BB=E9=A1=B5=E9=A1=B5?= =?UTF-8?q?=E9=9D=A2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 3 + ask.py | 109 ++++++++++++++++++++++++++++++ static/ask.html | 141 +++++++++++++++++++++++++++++++++++++++ static/img/logo-text.svg | 43 ++++++++++++ static/img/logo.svg | 1 + static/js/notify.min.js | 1 + 6 files changed, 298 insertions(+) create mode 100644 ask.py create mode 100644 static/ask.html create mode 100644 static/img/logo-text.svg create mode 100644 static/img/logo.svg create mode 100644 static/js/notify.min.js diff --git a/.gitignore b/.gitignore index b6e4761..38153ec 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,6 @@ +*.secret +*.db + # Byte-compiled / optimized / DLL files __pycache__/ *.py[cod] diff --git a/ask.py b/ask.py new file mode 100644 index 0000000..dca70f8 --- /dev/null +++ b/ask.py @@ -0,0 +1,109 @@ +from flask import Flask, request, render_template, send_from_directory +from flask_sqlalchemy import SQLAlchemy +from mastodon import Mastodon +import re +#import html2text + +BOT_NAME = '@ask_me_bot' +DOMAIN = 'thu.closed.social' + +token = open('token.secret','r').read().strip('\n') +th = Mastodon( + access_token = token, + api_base_url = 'https://' + DOMAIN +) + + +app = Flask(__name__) +app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///ask.db' + + +#h2t = html2text.HTML2Text() +#h2t.ignore_links = True + +def PM(msg, name): + th.status_post(msg + '\n@' + name, visibility='direct') + +db = SQLAlchemy(app) + +''' +class Record(db.Model): + id = db.Column(db.Integer, primary_key=True) + s = db.Column(db.String(64)) + name_hash = db.Column(db.String(64)) + full_hash = db.Column(db.String(64)) + ip = db.Column(db.String(32)) + cs_username = db.Column(db.String(32)) + + def __init__(self, s, name_hash, full_hash, ip): + self.s = s + self.name_hash = name_hash + self.full_hash = full_hash + self.ip = ip + self.cs_username = '' + + def __repr__(self): + return '%s[%s]<%s>'%(self.s, self.cs_username, self.ip) +''' + +@app.route('/js/') +def send_js(path): + return send_from_directory('static/js', path) +@app.route('/img/') +def send_img(path): + return send_from_directory('static/img', path) + +@app.route('/askMe/') +def root(): + return app.send_static_file('ask.html') + +@app.route('/askMe/inbox/', methods=['POST']) +def api(): + username = request.form.get('username') + if not re.match('[a-z0-9_]{1,30}(@[a-z\.-_]+)?', username): + return '闭社id格式错误', 422 + + return 'okkk' + + ha = request.form.get('hash') + if( not ha or len(ha) != 64 * 2): + return '哈希格式不正确', 422 + + ip = request.remote_addr + if ip in ip_count: + ip_count[ip] += 1 + if ip_count[ip] > 50: + return '该ip告白次数太多', 403 + else: + ip_count[ip] = 1 + + if Record.query.filter_by(s=s).count(): + return '暗号重复', 422 + if Record.query.filter_by(name_hash=ha[64:]).count(): + return '一个名字只能告白一次,\n重名/哈希冲突请联系主办方', 422 + + ta = Record.query.filter_by(full_hash=ha[:64]).first() + + rec = Record(s, ha[64:], ha[:64], ip) + rec = Record(s, ha[64:], ha[:64], ip) + db.session.add(rec) + db.session.commit() + + if not ta: + return '' + else: + if ta.cs_username: + PM('叮~ TA也给你表白啦! https://closed.social/meetLove/result/', ta.cs_username) + return 'y' if ta.cs_username else 'n' + +@app.route('/meetLove/result/') +def result(): + rs = Record.query.all() + rs.sort(key=lambda r:r.full_hash) + + lovers = [(rs[i].s[:-4]+'****', rs[i+1].s[:-4]+'****') for i in range(len(rs)-1) if rs[i].full_hash == rs[i+1].full_hash] + + return render_template('result.html', lovers=lovers) + +if __name__ == '__main__': + app.run() diff --git a/static/ask.html b/static/ask.html new file mode 100644 index 0000000..4db3643 --- /dev/null +++ b/static/ask.html @@ -0,0 +1,141 @@ + + + + + + + + + + + 匿名提问箱 + + + + +
+ + +
+ +

匿名提问箱

+
+ +

提问无需登陆,真正匿名
+ 链接路径加密,避免乱入

+ +
+ +
+ +
+ +

创建提问箱

+ +

分享加密链接

+ +

好友匿名提问

+ +

bot在闭社私信提醒

+ +

回复私信进行答复或删除

+
+ + +
+ +
+

输入闭社id并私信 @ask_me_bot “新建”或“重置”,然后点击按钮,即可新建提问箱或重置提问箱链接。

+
+
+
+
@
+
+ +
+ +
+


+ * 可通过重置链接并不再分享新链接来实现关闭提问箱。
+ ** 非清华站用户请使用完整用户名而非本地用户名,例如 @somebody@tha.closed.social。
+ ** 高级用法:私信“重置[路径]”以指定加密路径,而非默认的随机路径。限长度不超过32的小写字母,示例:
“@ask_me_bot 重置[hhhhhhh]”。 +

+
+ +
+ +
+ + + + + + diff --git a/static/img/logo-text.svg b/static/img/logo-text.svg new file mode 100644 index 0000000..ddb88e7 --- /dev/null +++ b/static/img/logo-text.svg @@ -0,0 +1,43 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/static/img/logo.svg b/static/img/logo.svg new file mode 100644 index 0000000..75a0658 --- /dev/null +++ b/static/img/logo.svg @@ -0,0 +1 @@ + diff --git a/static/js/notify.min.js b/static/js/notify.min.js new file mode 100644 index 0000000..6766fd1 --- /dev/null +++ b/static/js/notify.min.js @@ -0,0 +1 @@ +(function(e){typeof define=="function"&&define.amd?define(["jquery"],e):typeof module=="object"&&module.exports?module.exports=function(t,n){return n===undefined&&(typeof window!="undefined"?n=require("jquery"):n=require("jquery")(t)),e(n),n}:e(jQuery)})(function(e){function A(t,n,i){typeof i=="string"&&(i={className:i}),this.options=E(w,e.isPlainObject(i)?i:{}),this.loadHTML(),this.wrapper=e(h.html),this.options.clickToHide&&this.wrapper.addClass(r+"-hidable"),this.wrapper.data(r,this),this.arrow=this.wrapper.find("."+r+"-arrow"),this.container=this.wrapper.find("."+r+"-container"),this.container.append(this.userContainer),t&&t.length&&(this.elementType=t.attr("type"),this.originalElement=t,this.elem=N(t),this.elem.data(r,this),this.elem.before(this.wrapper)),this.container.hide(),this.run(n)}var t=[].indexOf||function(e){for(var t=0,n=this.length;t\n
\n
\n',css:"."+r+"-corner {\n position: fixed;\n margin: 5px;\n z-index: 1050;\n}\n\n."+r+"-corner ."+r+"-wrapper,\n."+r+"-corner ."+r+"-container {\n position: relative;\n display: block;\n height: inherit;\n width: inherit;\n margin: 3px;\n}\n\n."+r+"-wrapper {\n z-index: 1;\n position: absolute;\n display: inline-block;\n height: 0;\n width: 0;\n}\n\n."+r+"-container {\n display: none;\n z-index: 1;\n position: absolute;\n}\n\n."+r+"-hidable {\n cursor: pointer;\n}\n\n[data-notify-text],[data-notify-html] {\n position: relative;\n}\n\n."+r+"-arrow {\n position: absolute;\n z-index: 2;\n width: 0;\n height: 0;\n}"},p={"border-radius":["-webkit-","-moz-"]},d=function(e){return c[e]},v=function(e){if(!e)throw"Missing Style name";c[e]&&delete c[e]},m=function(t,i){if(!t)throw"Missing Style name";if(!i)throw"Missing Style definition";if(!i.html)throw"Missing Style HTML";var s=c[t];s&&s.cssElem&&(window.console&&console.warn(n+": overwriting style '"+t+"'"),c[t].cssElem.remove()),i.name=t,c[t]=i;var o="";i.classes&&e.each(i.classes,function(t,n){return o+="."+r+"-"+i.name+"-"+t+" {\n",e.each(n,function(t,n){return p[t]&&e.each(p[t],function(e,r){return o+=" "+r+t+": "+n+";\n"}),o+=" "+t+": "+n+";\n"}),o+="}\n"}),i.css&&(o+="/* styles for "+i.name+" */\n"+i.css),o&&(i.cssElem=g(o),i.cssElem.attr("id","notify-"+i.name));var u={},a=e(i.html);y("html",a,u),y("text",a,u),i.fields=u},g=function(t){var n,r,i;r=x("style"),r.attr("type","text/css"),e("head").append(r);try{r.html(t)}catch(s){r[0].styleSheet.cssText=t}return r},y=function(t,n,r){var s;return t!=="html"&&(t="text"),s="data-notify-"+t,b(n,"["+s+"]").each(function(){var n;n=e(this).attr(s),n||(n=i),r[n]=t})},b=function(e,t){return e.is(t)?e:e.find(t)},w={clickToHide:!0,autoHide:!0,autoHideDelay:5e3,arrowShow:!0,arrowSize:5,breakNewLines:!0,elementPosition:"bottom",globalPosition:"top right",style:"bootstrap",className:"error",showAnimation:"slideDown",showDuration:400,hideAnimation:"slideUp",hideDuration:200,gap:5},E=function(t,n){var r;return r=function(){},r.prototype=t,e.extend(!0,new r,n)},S=function(t){return e.extend(w,t)},x=function(t){return e("<"+t+">")},T={},N=function(t){var n;return t.is("[type=radio]")&&(n=t.parents("form:first").find("[type=radio]").filter(function(n,r){return e(r).attr("name")===t.attr("name")}),t=n.first()),t},C=function(e,t,n){var r,i;if(typeof n=="string")n=parseInt(n,10);else if(typeof n!="number")return;if(isNaN(n))return;return r=s[f[t.charAt(0)]],i=t,e[r]!==undefined&&(t=s[r.charAt(0)],n=-n),e[t]===undefined?e[t]=n:e[t]+=n,null},k=function(e,t,n){if(e==="l"||e==="t")return 0;if(e==="c"||e==="m")return n/2-t/2;if(e==="r"||e==="b")return n-t;throw"Invalid alignment"},L=function(e){return L.e=L.e||x("div"),L.e.text(e).html()};A.prototype.loadHTML=function(){var t;t=this.getStyle(),this.userContainer=e(t.html),this.userFields=t.fields},A.prototype.show=function(e,t){var n,r,i,s,o;r=function(n){return function(){!e&&!n.elem&&n.destroy();if(t)return t()}}(this),o=this.container.parent().parents(":hidden").length>0,i=this.container.add(this.arrow),n=[];if(o&&e)s="show";else if(o&&!e)s="hide";else if(!o&&e)s=this.options.showAnimation,n.push(this.options.showDuration);else{if(!!o||!!e)return r();s=this.options.hideAnimation,n.push(this.options.hideDuration)}return n.push(r),i[s].apply(i,n)},A.prototype.setGlobalPosition=function(){var t=this.getPosition(),n=t[0],i=t[1],o=s[n],u=s[i],a=n+"|"+i,f=T[a];if(!f||!document.body.contains(f[0])){f=T[a]=x("div");var l={};l[o]=0,u==="middle"?l.top="45%":u==="center"?l.left="45%":l[u]=0,f.css(l).addClass(r+"-corner"),e("body").append(f)}return f.prepend(this.wrapper)},A.prototype.setElementPosition=function(){var n,r,i,l,c,h,p,d,v,m,g,y,b,w,E,S,x,T,N,L,A,O,M,_,D,P,H,B,j;H=this.getPosition(),_=H[0],O=H[1],M=H[2],g=this.elem.position(),d=this.elem.outerHeight(),y=this.elem.outerWidth(),v=this.elem.innerHeight(),m=this.elem.innerWidth(),j=this.wrapper.position(),c=this.container.height(),h=this.container.width(),T=s[_],L=f[_],A=s[L],p={},p[A]=_==="b"?d:_==="r"?y:0,C(p,"top",g.top-j.top),C(p,"left",g.left-j.left),B=["top","left"];for(w=0,S=B.length;w=0&&C(r,s[O],i*2)}t.call(u,_)>=0?(C(p,"left",k(O,h,y)),r&&C(r,"left",k(O,i,m))):t.call(o,_)>=0&&(C(p,"top",k(O,c,d)),r&&C(r,"top",k(O,i,v))),this.container.is(":visible")&&(p.display="block"),this.container.removeAttr("style").css(p);if(r)return this.arrow.removeAttr("style").css(r)},A.prototype.getPosition=function(){var e,n,r,i,s,f,c,h;h=this.options.position||(this.elem?this.options.elementPosition:this.options.globalPosition),e=l(h),e.length===0&&(e[0]="b");if(n=e[0],t.call(a,n)<0)throw"Must be one of ["+a+"]";if(e.length===1||(r=e[0],t.call(u,r)>=0)&&(i=e[1],t.call(o,i)<0)||(s=e[0],t.call(o,s)>=0)&&(f=e[1],t.call(u,f)<0))e[1]=(c=e[0],t.call(o,c)>=0)?"m":"l";return e.length===2&&(e[2]=e[1]),e},A.prototype.getStyle=function(e){var t;e||(e=this.options.style),e||(e="default"),t=c[e];if(!t)throw"Missing style: "+e;return t},A.prototype.updateClasses=function(){var t,n;return t=["base"],e.isArray(this.options.className)?t=t.concat(this.options.className):this.options.className&&t.push(this.options.className),n=this.getStyle(),t=e.map(t,function(e){return r+"-"+n.name+"-"+e}).join(" "),this.userContainer.attr("class",t)},A.prototype.run=function(t,n){var r,s,o,u,a;e.isPlainObject(n)?e.extend(this.options,n):e.type(n)==="string"&&(this.options.className=n);if(this.container&&!t){this.show(!1);return}if(!this.container&&!t)return;s={},e.isPlainObject(t)?s=t:s[i]=t;for(o in s){r=s[o],u=this.userFields[o];if(!u)continue;u==="text"&&(r=L(r),this.options.breakNewLines&&(r=r.replace(/\n/g,"
"))),a=o===i?"":"="+o,b(this.userContainer,"[data-notify-"+u+a+"]").html(r)}this.updateClasses(),this.elem?this.setElementPosition():this.setGlobalPosition(),this.show(!0),this.options.autoHide&&(clearTimeout(this.autohideTimer),this.autohideTimer=setTimeout(this.show.bind(this,!1),this.options.autoHideDelay))},A.prototype.destroy=function(){this.wrapper.data(r,null),this.wrapper.remove()},e[n]=function(t,r,i){return t&&t.nodeName||t.jquery?e(t)[n](r,i):(i=r,r=t,new A(null,r,i)),t},e.fn[n]=function(t,n){return e(this).each(function(){var i=N(e(this)).data(r);i&&i.destroy();var s=new A(e(this),t,n)}),this},e.extend(e[n],{defaults:S,addStyle:m,removeStyle:v,pluginOptions:w,getStyle:d,insertCSS:g}),m("bootstrap",{html:"
\n\n
",classes:{base:{"font-weight":"bold",padding:"8px 15px 8px 14px","text-shadow":"0 1px 0 rgba(255, 255, 255, 0.5)","background-color":"#fcf8e3",border:"1px solid #fbeed5","border-radius":"4px","white-space":"nowrap","padding-left":"25px","background-repeat":"no-repeat","background-position":"3px 7px"},error:{color:"#B94A48","background-color":"#F2DEDE","border-color":"#EED3D7","background-image":"url()"},success:{color:"#468847","background-color":"#DFF0D8","border-color":"#D6E9C6","background-image":"url()"},info:{color:"#3A87AD","background-color":"#D9EDF7","border-color":"#BCE8F1","background-image":"url()"},warn:{color:"#C09853","background-color":"#FCF8E3","border-color":"#FBEED5","background-image":"url()"}}}),e(function(){g(h.css).attr("id","core-notify"),e(document).on("click","."+r+"-hidable",function(t){e(this).trigger("notify-hide")}),e(document).on("notify-hide","."+r+"-wrapper",function(t){var n=e(this).data(r);n&&n.show(!1)})})}) \ No newline at end of file