import gradio as gr import argparse import os, gc, torch from datetime import datetime from huggingface_hub import hf_hub_download import torch # from pynvml import * # nvmlInit() # gpu_h = nvmlDeviceGetHandleByIndex(0) ctx_limit = 4096 desc = f'''链接:太慢了?用Colab自己部署吧
ChatRWKVRWKV-LMRWKV pip package知乎教程 ''' parser = argparse.ArgumentParser(prog = 'ChatGal RWKV') parser.add_argument('--share',action='store_true') parser.add_argument('--ckpt',type=str,default="rwkv-loramerge-0426-v2-4096-epoch11.pth") parser.add_argument('--model_path',type=str,default=None,help="local model path") parser.add_argument('--lora', type=str, default=None, help='lora checkpoint path') parser.add_argument('--lora_alpha', type=float, default=0, help='lora alpha') parser.add_argument('--lora_layer_filter',type=str,default=None,help='layer filter. Default merge all layer. Example: "0.2*25-31"') args = parser.parse_args() os.environ["RWKV_JIT_ON"] = '1' # from rwkv.model import RWKV from rwkv_lora import RWKV lora_kwargs = { "lora":args.lora, "lora_alpha":args.lora_alpha, "lora_layer_filter":args.lora_layer_filter } if args.model_path: model_path = args.model_path else: model_path = hf_hub_download(repo_id="Synthia/ChatGalRWKV", filename=args.ckpt) # if 'ON_COLAB' in os.environ and os.environ['ON_COLAB'] == '1': if torch.cuda.is_available() and torch.cuda.device_count()>0: os.environ["RWKV_JIT_ON"] = '0' os.environ["RWKV_CUDA_ON"] = '0' # if '1' then use CUDA kernel for seq mode (much faster) model = RWKV(model=model_path, strategy='cuda bf16',**lora_kwargs) else: model = RWKV(model=model_path, strategy='cpu bf16',**lora_kwargs) from utils import PIPELINE, PIPELINE_ARGS pipeline = PIPELINE(model, "20B_tokenizer.json") def infer( ctx, token_count=10, temperature=0.7, top_p=1.0, top_k=50, typical_p=1.0, presencePenalty = 0.05, countPenalty = 0.05, ): args = PIPELINE_ARGS(temperature = max(0.2, float(temperature)), top_p = float(top_p), top_k=int(top_k),typical_p=float(typical_p), alpha_frequency = countPenalty, alpha_presence = presencePenalty, token_ban = [0], # ban the generation of some tokens token_stop = []) # stop generation whenever you see any token here # ctx = ctx.strip().split('\n') # for c in range(len(ctx)): # ctx[c] = ctx[c].strip().strip('\u3000').strip('\r') # ctx = list(filter(lambda c: c != '', ctx)) # ctx = '\n' + ('\n'.join(ctx)).strip() # if ctx == '': # ctx = '\n' # gpu_info = nvmlDeviceGetMemoryInfo(gpu_h) # print(f'vram {gpu_info.total} used {gpu_info.used} free {gpu_info.free}',flush=True) all_tokens = [] out_last = 0 out_str = '' occurrence = {} state = None for i in range(int(token_count)): out, state = model.forward(pipeline.encode(ctx)[-ctx_limit:] if i == 0 else [token], state) for n in args.token_ban: out[n] = -float('inf') for n in occurrence: out[n] -= (args.alpha_presence + occurrence[n] * args.alpha_frequency) token = pipeline.sample_logits(out, temperature=args.temperature, top_p=args.top_p, top_k=args.top_k, typical_p=args.typical_p) if token in args.token_stop: break all_tokens += [token] if token not in occurrence: occurrence[token] = 1 else: occurrence[token] += 1 tmp = pipeline.decode(all_tokens[out_last:]) if '\ufffd' not in tmp: out_str += tmp yield out_str out_last = i + 1 gc.collect() torch.cuda.empty_cache() yield out_str examples = [ ["""女招待: 欢迎光临。您远道而来,想必一定很累了吧? 深见: 不会……空气也清爽,也让我焕然一新呢 女招待: 是吗。那真是太好了 {我因为撰稿的需要,而造访了这间位于信州山间的温泉宿驿。}""", 200, 0.7, 1.0, 0, 1.0, 0.1, 0.1], ["""{我叫嘉祥,家里经营着一家点心店。 为了追求独当一面的目标,我离开了老家,开了一家名为"La Soleil"的新糕点店。 原本想独自一人打拼,却没想到,在搬家的箱子中发现了意想不到的人。 她叫巧克力,是我家的猫娘,没想到她竟然用这种方式跟了过来。} 嘉祥: 别以为这样就可以蒙混过去!你在干嘛啊,巧克力! 巧克力: 欸嘿嘿……那个,好、好久不见了呢,主人…… 嘉祥: 昨天才在家里见过面不是吗。 巧克力: 这个……话是这么说没错啦……啊哈哈……""", 200, 0.7, 1.0, 0, 1.0, 0.1, 0.1], ["""莲华: 你的目的,就是这个万华镜吧? {莲华拿出了万华镜。} 深见: 啊…… {好像被万华镜拽过去了一般,我的腿不由自主地向它迈去} 深见: 是这个……就是这个啊…… {烨烨生辉的魔法玩具。 连接现实与梦之世界的、诱惑的桥梁。} 深见: 请让我好好看看…… {我刚想把手伸过去,莲华就一下子把它收了回去。}""", 200, 0.7, 1.0, 0, 1.0, 0.1, 0.1], ["""{我叫嘉祥,有两只可爱的猫娘,名字分别是巧克力和香草。} 嘉祥: 偶尔来一次也不错。 {我坐到客厅的沙发上,拍了拍自己的大腿。} 巧克力&香草: 喵喵? 巧克力: 咕噜咕噜咕噜~♪人家最喜欢让主人掏耳朵了~♪ 巧克力: 主人好久都没有帮我们掏耳朵了,现在人家超兴奋的喵~♪ 香草: 身为猫娘饲主,这点服务也是应该的对吧? 香草: 老实说我也有点兴奋呢咕噜咕噜咕噜~♪ {我摸摸各自占据住我左右两腿的两颗猫头。} 嘉祥: 开心归开心,拜托你们俩别一直乱动啊,很危险的。""", 200, 0.7, 1.0, 0, 1.0, 0.1, 0.1], ["""{我叫嘉祥,在日本开了一家名为La Soleil的糕点店,同时也是猫娘巧克力的主人。 巧克力是非常聪明的猫娘,她去国外留学了一段时间,向Alice教授学习,拿到了计算机博士学位。 她会各种程序语言,对世界各地的风土人情都十分了解,也掌握了很多数学、物理知识。} 嘉祥: 很棒啊,巧克力!你真是懂不少东西呢! 巧克力: 因为巧克力是主人的最佳拍挡兼猫娘情人呀♪为了主人,巧克力会解决各种问题!""", 200, 0.7, 1.0, 0, 1.0, 0.1, 0.1], ] iface = gr.Interface( fn=infer, description=f'''这是GalGame剧本续写模型(实验性质,不保证效果)。请点击例子(在页面底部),可编辑内容。这里只看输入的最后约1200字,请写好,标点规范,无错别字,否则电脑会模仿你的错误。为避免占用资源,每次生成限制长度。可将输出内容复制到输入,然后继续生成。推荐提高temp改善文采,降低topp改善逻辑,提高两个penalty避免重复,具体幅度请自己实验。
{desc}''', allow_flagging="never", inputs=[ gr.Textbox(lines=10, label="Prompt 输入的前文", value="""{我叫嘉祥,在日本开了一家名为La Soleil的糕点店,同时也是猫娘巧克力的主人。 巧克力是非常聪明的猫娘,她去国外留学了一段时间,向Alice教授学习,拿到了计算机博士学位。 她会各种程序语言,对世界各地的风土人情都十分了解,也掌握了很多数学、物理知识。} 嘉祥: 很棒啊,巧克力!你真是懂不少东西呢! 巧克力: 因为巧克力是主人的最佳拍挡兼猫娘情人呀♪为了主人,巧克力会解决各种问题!"""), # prompt gr.Slider(10, 2000, step=10, value=200, label="token_count 每次生成的长度"), # token_count gr.Slider(0.2, 2.0, step=0.1, value=0.7, label="temperature 默认0.7,高则变化丰富,低则保守求稳"), # temperature gr.Slider(0.0, 1.0, step=0.05, value=1.0, label="top_p 默认1.0,高则标新立异,低则循规蹈矩"), # top_p gr.Slider(0, 500, step=1, value=0, label="top_k 默认0(不过滤),0以上时高则标新立异,低则循规蹈矩"), # top_p gr.Slider(0.05, 1.0, step=0.05, value=1.0, label="typical_p 默认1.0,高则保留模型天性,低则试图贴近人类典型习惯"), # top_p gr.Slider(0.0, 1.0, step=0.1, value=0.1, label="presencePenalty 默认0.0,避免写过的类似字"), # presencePenalty gr.Slider(0.0, 1.0, step=0.1, value=0.1, label="countPenalty 默认0.0,额外避免写过多次的类似字"), # countPenalty ], outputs=gr.Textbox(label="Output 输出的续写", lines=28), examples=examples, cache_examples=False, ).queue() demo = gr.TabbedInterface( [iface], ["Generative"] ) demo.queue(max_size=5) if args.share: demo.launch(share=True,server_name="0.0.0.0",server_port=58888) else: demo.launch(share=False,server_port=58888)