package main





import (

"html/template"

"io"

"io/ioutil"

"log"

"net/http"

"os"

"path"

"runtime/debug"

)





const (

ListDir      = 0x0001

UPLOAD_DIR   = "./uploads"

TEMPLATE_DIR = "./views"

)





//cache storage all template

var templates = make(map[string]*template.Template)





func init() {

fileInfoArr, err := ioutil.ReadDir(TEMPLATE_DIR)

check(err)

var templateName, templatePath string

for _, fileInfo := range fileInfoArr {

templateName = fileInfo.Name()

if ext := path.Ext(templateName); ext != ".html" {

continue

}

templatePath = TEMPLATE_DIR + "/" + templateName

//log.Println(templateName)

log.Println("Loading template:", templatePath)

t := template.Must(template.ParseFiles(templatePath))

templates[templateName] = t

}





//if const template

/*for _, tmpl := range []string{"upload", "list"} {

//Must ensure in case can't analytic will do error operate,If the template loading is not successful, the program will exit

t := template.Must(template.ParseFiles(tmpl + ".html"))

templates[tmpl] = t

}*/

}

func check(err error) {

if err != nil {

panic(err)

}

}

func renderHtml(w http.ResponseWriter, tmpl string, locals map[string]interface{}) {

/*tt := templates[tmpl]

log.Println("xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", templates, tmpl, tt)*/

log.Println(locals)

err := templates[tmpl].Execute(w, locals)





check(err)

}

func isExists(path string) bool {

_, err := os.Stat(path)

//log.Println(err)

if err == nil {

return true

}

return os.IsExist(err)

}





//Callback method,

func uploadHandler(w http.ResponseWriter, r *http.Request) {

//log.Println(r.Method)

if r.Method == "GET" {

renderHtml(w, "upload.html", nil)

}

if r.Method == "POST" {

f, h, err := r.FormFile("image")

//log.Println(f, h, err)

check(err)

filename := h.Filename

defer f.Close()

//log.Println(UPLOAD_DIR, filename)

t, err := ioutil.TempFile(UPLOAD_DIR, filename)

//log.Println(t)

check(err)

defer t.Close()

_, err = io.Copy(t, f) //dst src

check(err)

http.Redirect(w, r, "/view?

id="+filename, http.StatusFound)





}

}

func viewHandler(w http.ResponseWriter, r *http.Request) {

imageId := r.FormValue("id")

imagePath := UPLOAD_DIR + "/" + imageId

log.Println(imageId, imagePath)

if exists := isExists(imagePath); !exists {

http.NotFound(w, r)

return

}

//log.Println("222222222222")

w.Header().Set("Content-Type", "image")

http.ServeFile(w, r, imagePath)

}

func listHandler(w http.ResponseWriter, r *http.Request) {

fileInfoArr, err := ioutil.ReadDir("./uploads")

//log.Println(fileInfoArr)

check(err)

if err != nil {

http.Error(w, err.Error(), http.StatusInternalServerError)

return

}

locals := make(map[string]interface{})

images := []string{}

for _, fileInfo := range fileInfoArr {

//log.Println(fileInfo)

images = append(images, fileInfo.Name())

}

locals["images"] = images //这里map的key是images,相应list.html里面的$.images

renderHtml(w, "list.html", locals)





}



//巧用闭包避免执行时错误崩溃

func safeHandler(fn http.HandlerFunc) http.HandlerFunc {





return func(w http.ResponseWriter, r *http.Request) {

defer func() {

log.Println("can't past execute or finished execute")

if e, ok := recover().(error); ok {

log.Println("50x error")

//50x error

http.Error(w, e.Error(), http.StatusInternalServerError)

log.Println("Warn : panic in %v. - %v", fn, e)

log.Println(string(debug.Stack()))

}

}()





log.Println("if no panic then first execute")

fn(w, r)

}

}

func staticDirHandler(mux *http.ServeMux, prefix string, staticDir string, flags int) {

mux.HandleFunc(prefix, func(w http.ResponseWriter, r *http.Request) {

file := staticDir + r.URL.Path[len(prefix)-1:]

if (flags & ListDir) == 0 {

if exists := isExists(file); !exists {

http.NotFound(w, r)

return

}

}

http.ServeFile(w, r, file)

})





}

func main() {

mux := http.NewServeMux()

staticDirHandler(mux, "/assets/", "./public", 0)

mux.HandleFunc("/", safeHandler(listHandler))

mux.HandleFunc("/view", safeHandler(viewHandler))

mux.HandleFunc("/upload", safeHandler(uploadHandler))

err := http.ListenAndServe(":8080", mux)

if err != nil {

log.Fatal("ListenAndServe:", err.Error())

}

}

uoload.html

<html>

<head>

<meta charset="utf-8">

<title>upload</title>

</head>

<body>

<form method="POST" action="/upload" enctype="multipart/form-data">

choosing an image to upload :<br>

<input name="image" type="file">

<input type="submit" value="Upload">

</form>

</body>

</html>

list.html

<html>

<head>

<meta charset="utf-8">

<title>list</title>

</head>

<body>

<ol>

{{range $.images}}

<li><a href="/view?id={{.|urlquery}}">{{.|html}}</a></li>

{{end}}

</ol>

</body>

</html>

注意文件结构

photoweb

-photoweb.go

-public

-js

-css

-images

-uploads

-views

-upload.html

-list.html

解决调用七牛音频问题

演示样例仅仅提供代码方式调用api,不可以像图片处理直接在url中加入參数进行改动

若强用url会提示要预处理音频转码。一般转码採用异步方式

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbmVlZGthbmU=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

OK,在go的SDK代码中进行,结果发现api给的參数仅仅是个大概,go的api里面的相应參数。不全然和官网贴出来的一样

我们搜需PutPolicy

在rs/token.go 中,看到相应deadline。grep 'xx' *  -R查看文件中内容

package main





import (

"fmt"

. "github.com/qiniu/api/conf"

"github.com/qiniu/api/fop"

"github.com/qiniu/api/io"

"github.com/qiniu/api/rs"

"log"

)





func init() {





ACCESS_KEY = "自己的ak"

SECRET_KEY = "自己sk"

}





//GET upload access token

func uptoken(bucketName string) string {

putPolicy := rs.PutPolicy{

Scope: bucketName,

//CallbackUrl: callbackUrl,

//CallbackBody:callbackBody,

//ReturnUrl: returnUrl,

//ReturnBody: returnBody,

//AsyncOps: asyncOps,

//EndUser: endUser,

//Expires: expires,

Expires:             1406555272, //截止时间戳

PersistentOps:       "avthumb/mp3",

PersistentNotifyUrl: "http://fake.com/qiniu/notify",

}

return putPolicy.Token(nil)

}





func main() {

//上传本地文件

upload("a")





//5.1 获取文件信息

//getFileInfo()





//6.1.1 查看图像属性

//imageAttr()





//5.2 删除文件

//delFile()





}





//6.1.1 查看图像属性

func imageAttr() {

var imageUrl = "http://needkane.qiniudn.com/kane2.jpg"

ii := fop.ImageInfo{}

infoRet, err := ii.Call(nil, imageUrl)

if err != nil {

// 产生错误

log.Println("fop getImageInfo failed:", err)

return

}





log.Println(infoRet.Height, infoRet.Width, infoRet.ColorModel, infoRet.Format)

}





func makeImageInfoUrl(imageUrl string) string {

ii := fop.ImageInfo{}

return ii.MakeRequest(imageUrl)

}





//5.2 删除文件

func delFile() {

bucket := "needkane"

key := "goupload.jpg"

var rsCli = rs.New(nil)





err := rsCli.Delete(nil, bucket, key)

if err != nil {

// 产生错误

log.Println("rs.Copy failed:", err)

return

}

}





//5.1 获取文件信息

func getFileInfo() {

var ret rs.Entry

bucket := "needkane"

key := "kane3.jpg"

var rsCli = rs.New(nil)

var err error

ret, err = rsCli.Stat(nil, bucket, key)





if err != nil {

// 产生错误

log.Println("rs.Stat failed:", err)

return

}





// 处理返回值

log.Println(ret)

}





//上传本地文件

func upload(key string) {

uptoken := uptoken("needkane")

fmt.Printf("uptoken:%s\n", uptoken)





var err error

var ret io.PutRet

var extra = &io.PutExtra{

//Params: params,

//MimeType: mieType,

//Crc32: crc32,

//CheckCrc: CheckCrc,

}





var localFile = "/home/qboxtest/Downloads/a.wav"





// ret 变量用于存取返回的信息,详情见 io.PutRet

// uptoken 为业务server生成的上传口令

// key 为文件存储的标识

// localFile 为本地文件名称

// extra 为上传文件的额外信息。详情见 io.PutExtra,可选

err = io.PutFile(nil, &ret, uptoken, key, localFile, extra)





if err != nil {

//上传产生错误

log.Print("io.PutFile failed:", err)

return

}





//上传成功,处理返回值

log.Print(ret.Hash, ret.Key)





}

用 file 命令查看文件

第一个go的web程序;调用七牛云存储的音频api问题解决;条件搜寻文件中的内容,字符串拼接+在上一行的更多相关文章

  1. ueditor上传图片到七牛云存储(form api,java)

    转:http://my.oschina.net/duoduo3369/blog/174655 ueditor上传图片到七牛云存储 ueditor结合七牛传图片 七牛的试炼 开发前的准备与注意事项说明 ...

  2. 七牛云存储的 Javascript Web 前端文件上传

    因为我的个人网站 restran.net 已经启用,博客园的内容已经不再更新.请访问我的个人网站获取这篇文章的最新内容,七牛云存储的 Web 前端文件上传 七牛是不错的云存储产品,特别是有免费的配额可 ...

  3. 代码实现:定义一个文件输入流,调用read(byte[] b)方法,将a.txt文件中的内容打印出来(byte数组大小限制为5)

    package com.loaderman.test; import java.io.ByteArrayOutputStream; import java.io.FileInputStream; im ...

  4. CodeIgniter - 集成七牛云存储

    最近有一个项目需要集成七牛云存储的图片存储和调用功能,程序是基于CodeIgniter2.1.3的PHP框架.刚拿到手完全无从下手的感觉,因为像框架这种东西,想从官方的PHPSDK集成进去,需要改动很 ...

  5. Go语言实战 - 网站性能优化第一弹“七牛云存储”

    由于用户纷纷反应山坡网的打开速度比较慢,所以两天前我们决定把服务器从linode迁移到阿里云. 整个迁移过程非常平滑,基本上一个小时就完成了.而且阿里云的配套设施提供的也很不错,运行状态监控什么的都有 ...

  6. JavaWeb结合七牛云存储搭建个人相册服务

    JavaWeb结合七牛云存储搭建个人相册服务 一.引言1. 课程概述 相信很多人都知道网站一般会有很多图片,对于小型网站来说,图片放在网站服务器上不算什么,但当图片数量很大时,会造成服务器很臃肿,相应 ...

  7. 《介绍一款开源的类Excel电子表格软件》续:七牛云存储实战(C#)

    两个月前的发布的博客<介绍一款开源的类Excel电子表格软件>引起了热议:在博客园有近2000个View.超过20个评论. 同时有热心读者电话咨询如何能够在SpreadDesing中实现存 ...

  8. PHP使用七牛云存储之图片的上传、下载、303重定向教程,CI框架实例

    网上关于七牛云存储的教程除了官网上的API文档,其他的资料太少了.研究了下API之后,现在已经能实现图片的上传和下载及上传之后的重定向. http://blog.csdn.net/cqcre/arti ...

  9. wordpress使用video.js与七牛云存储实现无广告视频分享应用

    video.js是一款极受欢迎的基于HTML5的开源WEB视频播放器,其充分利用了HTML5的视频支持特性,可以实现全平台的无视频插件播放功能,对于现在流行的手机.PAD等移动智能终端有极佳的应用体验 ...

随机推荐

  1. Limu:有关JavaScript的那些值得一看的书

    来源&作者:Limu 又好久没写东西了 ,写上一篇的时候还以为接下来的工作会轻松一些 ,结果未从我所愿呐 ,又是一阵忙碌.而这段时间穿插着做了很多12年淘宝校园招聘的前端面试 ,很多同学都有问 ...

  2. 由易信界面——谈谈fragment 状态的保存

    看看我要实现的效果: 其实,这种左右界面切换保存布局方式,不只是易信界面这么用罢了.这更是大多数app布局的主流,而在android平台上面,随着谷歌大力推荐fragment的使用,用fragment ...

  3. Android -- DecorView

    DecorView 开发中,通常都是在onCreate()中调用setContentView(R.layout.custom_layout)来实现想要的页面布局.页面都是依附在窗口之上的,而Decor ...

  4. 【Spark】SparkStreaming-Kafka-Redis-集成-基础参考资料

    SparkStreaming-Kafka-Redis-集成-基础参考资料 Overview - Spark 2.2.0 Documentation Spark Streaming + Kafka In ...

  5. SciPy 安装不上?

    参考:链接:https://www.zhihu.com/question/30188492/answer/150928275来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处 ...

  6. Linux下逻辑地址-线性地址-物理地址图解(转)

    一.逻辑地址转线性地址 机器语言指令中出现的内存地址,都是逻辑地址,需要转换成线性地址,再经过MMU(CPU中的内存管理单元)转换成物理地址才能够被访问到. 我们写个最简单的hello world程序 ...

  7. GPUImage简单滤镜使用(一)

    今天来学习一下一个简单滤镜使用的流程,通过调节亮度滤镜来了解.先将GPUImage库导入到项目中,引入头文件"GPUImage.h"   一.创建亮度滤镜对象    GPUImag ...

  8. Python 使用pymongo操作mongodb库

    Python 使用pymongo操作mongodb库 2016-12-31 21:55 1115人阅读 评论(0) 收藏 举报  分类: - - - Python(10)  版权声明:本文为博主原创文 ...

  9. maven 配置篇 之pom.xml

    http://www.blogjava.net/zyl/archive/2006/12/30/91055.html http://maven.apache.org/pom.html的翻译.     m ...

  10. 使用Json.Net解决MVC中各种json操作

    最近收集了几篇文章,用于替换MVC中各种json操作,微软mvc当然用自家的序列化,速度慢不说,还容易出问题,自定义性也太差,比如得特意解决循环引用的问题,比如datetime的序列化格式,比如性能. ...