smarter delete
This commit is contained in:
		| @@ -135,6 +135,9 @@ type Database interface { | |||||||
| 	// get all attachments for this message | 	// get all attachments for this message | ||||||
| 	GetPostAttachments(message_id string) []string | 	GetPostAttachments(message_id string) []string | ||||||
|  |  | ||||||
|  | 	// get all attachments in a thread | ||||||
|  | 	GetThreadAttachments(rootmsgid string) ([]string, error) | ||||||
|  |  | ||||||
| 	// get all attachments for this message | 	// get all attachments for this message | ||||||
| 	GetPostAttachmentModels(prefix, message_id string) []AttachmentModel | 	GetPostAttachmentModels(prefix, message_id string) []AttachmentModel | ||||||
|  |  | ||||||
|   | |||||||
| @@ -46,9 +46,8 @@ type expire struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (self expire) ExpirePost(messageID string) { | func (self expire) ExpirePost(messageID string) { | ||||||
| 	self.handleEvent(deleteEvent(self.store.GetFilename(messageID))) |  | ||||||
| 	// get article headers |  | ||||||
| 	headers := self.store.GetHeaders(messageID) | 	headers := self.store.GetHeaders(messageID) | ||||||
|  | 	// get article headers | ||||||
| 	if headers != nil { | 	if headers != nil { | ||||||
| 		group := headers.Get("Newsgroups", "") | 		group := headers.Get("Newsgroups", "") | ||||||
| 		// is this a root post ? | 		// is this a root post ? | ||||||
| @@ -57,6 +56,7 @@ func (self expire) ExpirePost(messageID string) { | |||||||
| 			// ya, expire the entire thread | 			// ya, expire the entire thread | ||||||
| 			self.ExpireThread(group, messageID) | 			self.ExpireThread(group, messageID) | ||||||
| 		} else { | 		} else { | ||||||
|  | 			self.handleEvent(deleteEvent(self.store.GetFilename(messageID))) | ||||||
| 			self.expireCache(group, messageID, ref) | 			self.expireCache(group, messageID, ref) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -70,14 +70,26 @@ func (self expire) ExpireGroup(newsgroup string, keep int) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (self expire) ExpireThread(group, rootMsgid string) { | func (self expire) ExpireThread(group, rootMsgid string) { | ||||||
| 	replies, err := self.database.GetMessageIDByHeader("References", rootMsgid) | 	files, err := self.database.GetThreadAttachments(rootMsgid) | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		for _, reply := range replies { | 		for _, file := range files { | ||||||
| 			self.handleEvent(deleteEvent(self.store.GetFilename(reply))) | 			img := self.store.AttachmentFilepath(file) | ||||||
|  | 			os.Remove(img) | ||||||
|  | 			thm := self.store.ThumbnailFilepath(file) | ||||||
|  | 			os.Remove(thm) | ||||||
| 		} | 		} | ||||||
|  | 	} else { | ||||||
|  | 		log.Println("expirethread::GetThreadAttachments:", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	replies := self.database.GetThreadReplies(rootMsgid, 0, 0) | ||||||
|  |  | ||||||
|  | 	for _, msgid := range replies { | ||||||
|  | 		self.store.Remove(msgid) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	self.store.Remove(rootMsgid) | ||||||
| 	self.database.DeleteThread(rootMsgid) | 	self.database.DeleteThread(rootMsgid) | ||||||
| 	self.database.DeleteArticle(rootMsgid) |  | ||||||
| 	self.expireCache(group, rootMsgid, rootMsgid) | 	self.expireCache(group, rootMsgid, rootMsgid) | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -137,11 +149,14 @@ func (self expire) handleEvent(ev deleteEvent) { | |||||||
| 			os.Remove(thm) | 			os.Remove(thm) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	banned := self.database.ArticleBanned(ev.MessageID()) | ||||||
|  | 	if !banned { | ||||||
| 		err := self.database.BanArticle(ev.MessageID(), "expired") | 		err := self.database.BanArticle(ev.MessageID(), "expired") | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println("failed to ban for expiration", err) | 			log.Println("failed to ban for expiration", err) | ||||||
| 		} | 		} | ||||||
| 	err = self.database.DeleteArticle(ev.MessageID()) | 	} | ||||||
|  | 	err := self.database.DeleteArticle(ev.MessageID()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("failed to delete article", err) | 		log.Println("failed to delete article", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -97,7 +97,9 @@ const DeleteArticle_2 = "DeleteArticle_2" | |||||||
| const DeleteArticle_3 = "DeleteArticle_3" | const DeleteArticle_3 = "DeleteArticle_3" | ||||||
| const DeleteArticle_4 = "DeleteArticle_4" | const DeleteArticle_4 = "DeleteArticle_4" | ||||||
| const DeleteArticle_5 = "DeleteArticle_5" | const DeleteArticle_5 = "DeleteArticle_5" | ||||||
|  | const DeleteArticleV8 = "DeleteArticleV8" | ||||||
| const DeleteThread = "DeleteThread" | const DeleteThread = "DeleteThread" | ||||||
|  | const DeleteThreadV8 = "DeleteThreadV8" | ||||||
| const GetThreadReplyPostModels_1 = "GetThreadReplyPostModels_1" | const GetThreadReplyPostModels_1 = "GetThreadReplyPostModels_1" | ||||||
| const GetThreadReplyPostModels_2 = "GetThreadReplyPostModels_2" | const GetThreadReplyPostModels_2 = "GetThreadReplyPostModels_2" | ||||||
| const GetThreadReplies_1 = "GetThreadReplies_1" | const GetThreadReplies_1 = "GetThreadReplies_1" | ||||||
| @@ -109,6 +111,7 @@ const HasNewsgroup = "HasNewsgroup" | |||||||
| const HasArticle = "HasArticle" | const HasArticle = "HasArticle" | ||||||
| const HasArticleLocal = "HasArticleLocal" | const HasArticleLocal = "HasArticleLocal" | ||||||
| const GetPostAttachments = "GetPostAttachments" | const GetPostAttachments = "GetPostAttachments" | ||||||
|  | const GetThreadAttachments = "GetThreadAttachments" | ||||||
| const GetPostAttachmentModels = "GetPostAttachmentModels" | const GetPostAttachmentModels = "GetPostAttachmentModels" | ||||||
| const RegisterArticle_1 = "RegisterArticle_1" | const RegisterArticle_1 = "RegisterArticle_1" | ||||||
| const RegisterArticle_2 = "RegisterArticle_2" | const RegisterArticle_2 = "RegisterArticle_2" | ||||||
| @@ -165,6 +168,8 @@ func (self *PostgresDatabase) prepareStatements() { | |||||||
| 		DeleteArticle_4:                 "DELETE FROM ArticleKeys WHERE message_id = $1", | 		DeleteArticle_4:                 "DELETE FROM ArticleKeys WHERE message_id = $1", | ||||||
| 		DeleteArticle_5:                 "DELETE FROM ArticleAttachments WHERE message_id = $1", | 		DeleteArticle_5:                 "DELETE FROM ArticleAttachments WHERE message_id = $1", | ||||||
| 		DeleteThread:                    "DELETE FROM ArticleThreads WHERE root_message_id = $1", | 		DeleteThread:                    "DELETE FROM ArticleThreads WHERE root_message_id = $1", | ||||||
|  | 		DeleteArticleV8:                 "DELETE FROM ArticlePosts WHERE message_id = $1", | ||||||
|  | 		DeleteThreadV8:                  "DELETE FROM ArticlePosts WHERE root_message_id = $1 OR message_id = $1", | ||||||
| 		GetThreadReplyPostModels_1:      "SELECT newsgroup, message_id, ref_id, name, subject, path, time_posted, message, addr FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ORDER BY time_posted DESC LIMIT $2 ) ORDER BY time_posted ASC", | 		GetThreadReplyPostModels_1:      "SELECT newsgroup, message_id, ref_id, name, subject, path, time_posted, message, addr FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ORDER BY time_posted DESC LIMIT $2 ) ORDER BY time_posted ASC", | ||||||
| 		GetThreadReplyPostModels_2:      "SELECT newsgroup, message_id, ref_id, name, subject, path, time_posted, message, addr FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ) ORDER BY time_posted ASC", | 		GetThreadReplyPostModels_2:      "SELECT newsgroup, message_id, ref_id, name, subject, path, time_posted, message, addr FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ) ORDER BY time_posted ASC", | ||||||
| 		GetThreadReplies_1:              "SELECT message_id FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ORDER BY time_posted DESC LIMIT $2 ) ORDER BY time_posted ASC", | 		GetThreadReplies_1:              "SELECT message_id FROM ArticlePosts WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE ref_id = $1 ORDER BY time_posted DESC LIMIT $2 ) ORDER BY time_posted ASC", | ||||||
| @@ -176,6 +181,7 @@ func (self *PostgresDatabase) prepareStatements() { | |||||||
| 		HasArticle:                      "SELECT 1 FROM Articles WHERE message_id = $1", | 		HasArticle:                      "SELECT 1 FROM Articles WHERE message_id = $1", | ||||||
| 		HasArticleLocal:                 "SELECT 1 FROM ArticlePosts WHERE message_id = $1", | 		HasArticleLocal:                 "SELECT 1 FROM ArticlePosts WHERE message_id = $1", | ||||||
| 		GetPostAttachments:              "SELECT filepath FROM ArticleAttachments WHERE message_id = $1", | 		GetPostAttachments:              "SELECT filepath FROM ArticleAttachments WHERE message_id = $1", | ||||||
|  | 		GetThreadAttachments:            "SELECT filepath FROM ArticleAttachments WHERE message_id IN ( SELECT message_id FROM ArticlePosts WHERE root_message_id = $1 OR message_id = $1)", | ||||||
| 		GetPostAttachmentModels:         "SELECT filepath, filename FROM ArticleAttachments WHERE message_id = $1", | 		GetPostAttachmentModels:         "SELECT filepath, filename FROM ArticleAttachments WHERE message_id = $1", | ||||||
| 		RegisterArticle_1:               "INSERT INTO Articles (message_id, message_id_hash, message_newsgroup, time_obtained, message_ref_id) VALUES($1, $2, $3, $4, $5)", | 		RegisterArticle_1:               "INSERT INTO Articles (message_id, message_id_hash, message_newsgroup, time_obtained, message_ref_id) VALUES($1, $2, $3, $4, $5)", | ||||||
| 		RegisterArticle_2:               "UPDATE Newsgroups SET last_post = $1 WHERE name = $2", | 		RegisterArticle_2:               "UPDATE Newsgroups SET last_post = $1 WHERE name = $2", | ||||||
| @@ -247,6 +253,8 @@ func (self *PostgresDatabase) CreateTables() { | |||||||
| 			// upgrade to version 7 | 			// upgrade to version 7 | ||||||
| 			self.upgrade6to7() | 			self.upgrade6to7() | ||||||
| 		} else if version == 7 { | 		} else if version == 7 { | ||||||
|  | 			self.upgrade7to8() | ||||||
|  | 		} else if version == 8 { | ||||||
| 			// we are up to date | 			// we are up to date | ||||||
| 			log.Println("we are up to date at version", version) | 			log.Println("we are up to date at version", version) | ||||||
| 			break | 			break | ||||||
| @@ -589,6 +597,24 @@ func (self *PostgresDatabase) upgrade6to7() { | |||||||
| 	self.setDBVersion(7) | 	self.setDBVersion(7) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (self *PostgresDatabase) upgrade7to8() { | ||||||
|  | 	log.Println("migrating 7 -> 8") | ||||||
|  | 	cmds := []string{ | ||||||
|  | 		"ALTER TABLE ArticleNumbers DROP CONSTRAINT FOREIGN KEY (message_id) REFERENCES ArticlePosts(message_id)", | ||||||
|  | 		"ALTER TABLE ArticleNumbers ADD CONSTRAINT FOREIGN KEY (message_id) REFERENCES ArticlePosts(message_id) ON DELETE CASCADE", | ||||||
|  | 		"ALTER TABLE NNTPHeaders DROP CONSTRAINT FOREIGN KEY (header_article_message_id) REFERENCES ArticlePosts(message_id)", | ||||||
|  | 		"ALTER TABLE NNTPHeaders ADD CONSTRAINT FOREIGN KEY (header_article_message_id) REFERENCES ArticlePosts(message_id) ON DELETE CASCADE", | ||||||
|  | 	} | ||||||
|  | 	for _, cmd := range cmds { | ||||||
|  | 		log.Println("exec", cmd) | ||||||
|  | 		_, err := self.conn.Exec(cmd) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Fatalf("%s: %s", cmd, err.Error()) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	self.setDBVersion(8) | ||||||
|  | } | ||||||
|  |  | ||||||
| // create all tables for database version 0 | // create all tables for database version 0 | ||||||
| func (self *PostgresDatabase) createTablesV0() { | func (self *PostgresDatabase) createTablesV0() { | ||||||
| 	tables := make(map[string]string) | 	tables := make(map[string]string) | ||||||
| @@ -1182,17 +1208,20 @@ func (self *PostgresDatabase) GetThreadModel(prefix, msgid string) (th ThreadMod | |||||||
| } | } | ||||||
|  |  | ||||||
| func (self *PostgresDatabase) DeleteThread(msgid string) (err error) { | func (self *PostgresDatabase) DeleteThread(msgid string) (err error) { | ||||||
| 	_, err = self.conn.Exec(self.stmt[DeleteThread], msgid) | 	_, err = self.conn.Exec(self.stmt[DeleteThreadV8], msgid) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *PostgresDatabase) DeleteArticle(msgid string) (err error) { | func (self *PostgresDatabase) DeleteArticle(msgid string) (err error) { | ||||||
|  | 	/* | ||||||
| 		for _, q := range []string{DeleteArticle_1, DeleteArticle_2, DeleteArticle_3, DeleteArticle_4, DeleteArticle_5} { | 		for _, q := range []string{DeleteArticle_1, DeleteArticle_2, DeleteArticle_3, DeleteArticle_4, DeleteArticle_5} { | ||||||
| 			_, err = self.conn.Exec(self.stmt[q], msgid) | 			_, err = self.conn.Exec(self.stmt[q], msgid) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 	*/ | ||||||
|  | 	_, err = self.conn.Exec(self.stmt[DeleteArticleV8], msgid) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -1367,6 +1396,22 @@ func (self *PostgresDatabase) RegisterNewsgroup(group string) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (self *PostgresDatabase) GetThreadAttachments(rootmsgid string) (atts []string, err error) { | ||||||
|  | 	var rows *sql.Rows | ||||||
|  | 	rows, err = self.conn.Query(self.stmt[GetThreadAttachments], rootmsgid) | ||||||
|  | 	if err == nil { | ||||||
|  | 		for rows.Next() { | ||||||
|  | 			var msgid string | ||||||
|  | 			rows.Scan(&msgid) | ||||||
|  | 			atts = append(atts, msgid) | ||||||
|  | 		} | ||||||
|  | 		rows.Close() | ||||||
|  | 	} else if err == sql.ErrNoRows { | ||||||
|  | 		err = nil | ||||||
|  | 	} | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  |  | ||||||
| func (self *PostgresDatabase) GetPostAttachments(messageID string) (atts []string) { | func (self *PostgresDatabase) GetPostAttachments(messageID string) (atts []string) { | ||||||
| 	rows, err := self.conn.Query(self.stmt[GetPostAttachments], messageID) | 	rows, err := self.conn.Query(self.stmt[GetPostAttachments], messageID) | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user