the contents of the messages are not completely specified in the paper we assume that the user sends his request and his itinerary in the clear
we will adopt the following format for messages: \<lbrace>A,r,I,L\<rbrace> A: originator (agent) r: request (number) I: next shops (agent list) L: collected offers (offer list)
in the paper, the authors use nonces r_i to add redundancy in the offer in order to make it safer against dictionary attacks it is not necessary in our modelization since crypto is assumed to be strong (Crypt in injective)
******************************************************************************)
subsubsection\<open>offer chaining:
B chains his offer for A with the head offer of L for sending it to C\<close>
definition chain :: "agent => nat => agent => msg => agent => msg"where "chain B ofr A L C == let m1= Crypt (pubK A) (Nonce ofr) in let m2= Hash \<lbrace>head L, Agent C\<rbrace> in
sign B \<lbrace>m1,m2\<rbrace>"
declare Let_def [simp]
lemma chain_inj [iff]: "(chain B ofr A L C = chain B' ofr' A' L' C')
= (B=B' & ofr=ofr' & A=A' & head L = head L' & C=C')" by (auto simp: chain_def Let_def)
lemma Nonce_in_chain [iff]: "Nonce ofr \ parts {chain B ofr A L C}" by (auto simp: chain_def sign_def)
subsubsection\<open>agent whose key is used to sign an offer\<close>
fun shop :: "msg => msg"where "shop \B,X,Crypt K H\ = Agent (agt K)"
lemma shop_chain [simp]: "shop (chain B ofr A L C) = Agent B" by (simp add: chain_def sign_def)
subsubsection\<open>nonce used in an offer\<close>
fun nonce :: "msg => msg"where "nonce \B,\Crypt K ofr,m2\,CryptH\ = ofr"
lemma nonce_chain [simp]: "nonce (chain B ofr A L C) = Nonce ofr" by (simp add: chain_def sign_def)
lemma next_shop_chain [iff]: "next_shop (chain B ofr A L C) = C" by (simp add: chain_def sign_def)
subsubsection\<open>anchor of the offer list\<close>
definition anchor :: "agent => nat => agent => msg"where "anchor A n B == chain A n A (cons nil nil) B"
lemma anchor_inj [iff]: "(anchor A n B = anchor A' n' B')
= (A=A' & n=n' & B=B')" by (auto simp: anchor_def)
lemma Nonce_in_anchor [iff]: "Nonce n \ parts {anchor A n B}" by (auto simp: anchor_def)
lemma shop_anchor [simp]: "shop (anchor A n B) = Agent A" by (simp add: anchor_def)
lemma nonce_anchor [simp]: "nonce (anchor A n B) = Nonce n" by (simp add: anchor_def)
lemma next_shop_anchor [iff]: "next_shop (anchor A n B) = B" by (simp add: anchor_def)
subsubsection\<open>request event\<close>
definition reqm :: "agent => nat => nat => msg => agent => msg"where "reqm A r n I B == \Agent A, Number r, cons (Agent A) (cons (Agent B) I),
cons (anchor A n B) nil\<rbrace>"
lemma reqm_inj [iff]: "(reqm A r n I B = reqm A' r' n' I' B')
= (A=A' & r=r' & n=n' & I=I' & B=B')" by (auto simp: reqm_def)
lemma Nonce_in_reqm [iff]: "Nonce n \ parts {reqm A r n I B}" by (auto simp: reqm_def)
definition req :: "agent => nat => nat => msg => agent => event"where "req A r n I B == Says A B (reqm A r n I B)"
lemma req_inj [iff]: "(req A r n I B = req A' r' n' I' B')
= (A=A' & r=r' & n=n' & I=I' & B=B')" by (auto simp: req_def)
subsubsection\<open>propose event\<close>
definition prom :: "agent => nat => agent => nat => msg => msg =>
msg => agent => msg" where "prom B ofr A r I L J C == \Agent A, Number r,
app (J, del (Agent B, I)), cons (chain B ofr A L C) L\<rbrace>"
lemma prom_inj [dest]: "prom B ofr A r I L J C
= prom B' ofr' A' r' I' L' J' C' \<Longrightarrow> B=B' & ofr=ofr' & A=A' & r=r' & L=L' & C=C'" by (auto simp: prom_def)
lemma Nonce_in_prom [iff]: "Nonce ofr \ parts {prom B ofr A r I L J C}" by (auto simp: prom_def)
definition pro :: "agent => nat => agent => nat => msg => msg =>
msg => agent => event" where "pro B ofr A r I L J C == Says B C (prom B ofr A r I L J C)"
lemma pro_inj [dest]: "pro B ofr A r I L J C = pro B' ofr' A' r' I' L' J' C' \<Longrightarrow> B=B' & ofr=ofr' & A=A' & r=r' & L=L' & C=C'" by (auto simp: pro_def dest: prom_inj)
subsubsection\<open>protocol\<close>
inductive_set p1 :: "event list set" where
Nil: "[] \ p1"
| Fake: "\evsf \ p1; X \ synth (analz (spies evsf))\ \ Says Spy B X # evsf \ p1"
| Request: "\evsr \ p1; Nonce n \ used evsr; I \ agl\ \ req A r n I B # evsr \ p1"
| Propose: "\evsp \ p1; Says A' B \Agent A,Number r,I,cons M L\ \ set evsp;
I \<in> agl; J \<in> agl; isin (Agent C, app (J, del (Agent B, I)));
Nonce ofr \<notin> used evsp\<rbrakk> \<Longrightarrow> pro B ofr A r I (cons M L) J C # evsp \<in> p1"
inductive_set
valid :: "agent \ nat \ agent \ msg set" for A :: agent and n :: nat and B :: agent where
Request [intro]: "cons (anchor A n B) nil \ valid A n B"
| Propose [intro]: "L \ valid A n B \<Longrightarrow> cons (chain (next_shop (head L)) ofr A L C) L \<in> valid A n B"
subsubsection\<open>basic properties of valid\<close>
lemma valid_not_empty: "L \ valid A n B \ \M L'. L = cons M L'" by (erule valid.cases, auto)
lemma valid_pos_len: "L \ valid A n B \ 0 < len L" by (erule valid.induct, auto)
subsubsection\<open>offers of an offer list\<close>
definition offer_nonces :: "msg \ msg set" where "offer_nonces L \ {X. X \ parts {L} \ (\n. X = Nonce n)}"
subsubsection\<open>the originator can get the offers\<close>
lemma"L \ valid A n B \ offer_nonces L \ analz (insert L (initState A))" by (erule valid.induct, auto simp: anchor_def chain_def sign_def
offer_nonces_def initState.simps)
subsubsection\<open>list of offers\<close>
fun offers :: "msg => msg"where "offers (cons M L) = cons \shop M, nonce M\ (offers L)" | "offers other = nil"
subsubsection\<open>list of agents whose keys are used to sign a list of offers\<close>
fun shops :: "msg => msg"where "shops (cons M L) = cons (shop M) (shops L)" | "shops other = other"
lemma shops_in_agl: "L \ valid A n B \ shops L \ agl" by (erule valid.induct, auto simp: anchor_def chain_def sign_def)
subsubsection\<open>builds a trace from an itinerary\<close>
fun offer_list :: "agent \ nat \ agent \ msg \ nat \ msg" where "offer_list (A,n,B,nil,ofr) = cons (anchor A n B) nil" | "offer_list (A,n,B,cons (Agent C) I,ofr) = ( let L = offer_list (A,n,B,I,Suc ofr) in
cons (chain (next_shop (head L)) ofr A L C) L)"
lemma"I \ agl \ \ofr. offer_list (A,n,B,I,ofr) \ valid A n B" by (erule agl.induct, auto)
fun trace :: "agent \ nat \ agent \ nat \ msg \ msg \ msg \<Rightarrow> event list" where "trace (B,ofr,A,r,I,L,nil) = []" | "trace (B,ofr,A,r,I,L,cons (Agent D) K) = ( let C = (if K=nil then B else agt_nb (head K)) in let I' = (if K=nil then cons (Agent A) (cons (Agent B) I)
else cons (Agent A) (app (I, cons (head K) nil))) in let I'' = app (I, cons (head K) nil) in
pro C (Suc ofr) A r I' L nil D
# trace (B,Suc ofr,A,r,I'',tail L,K))"
definition trace' :: "agent \ nat \ nat \ msg \ agent \ nat \ event list" where "trace' A r n I B ofr \ ( let AI = cons (Agent A) I in let L = offer_list (A,n,B,AI,ofr) in
trace (B,ofr,A,r,nil,L,AI))"
declare trace'_def [simp]
subsubsection\<open>there is a trace in which the originator receives a valid answer\<close>
lemma p1_not_empty: "evs \ p1 \ req A r n I B \ set evs \
(\<exists>evs'. evs' @ evs \<in> p1 \<and> pro B' ofr A r I' L J A \<in> set evs' \<and> L \<in> valid A n B)" oops
subsection\<open>properties of protocol P1\<close>
text\<open>publicly verifiable forward integrity:
anyone can verify the validity of an offer list\<close>
subsubsection\<open>strong forward integrity:
except the last one, no offer can be modified\<close>
lemma strong_forward_integrity: "\L. Suc i < len L \<longrightarrow> L \<in> valid A n B \<and> repl (L,Suc i,M) \<in> valid A n B \<longrightarrow> M = ith (L,Suc i)" apply (induct i) (* i = 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,xa,l'a\ \ valid A n B" for x xa l'a) apply (ind_cases "\x,M,l'a\ \ valid A n B" for x l'a) apply (simp add: chain_def) (* i > 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,repl(l',Suc na,M)\ \ valid A n B" for x l' na) apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,l'\ \ valid A n B" for x l') by (drule_tac x=l' in spec, simp, blast)
subsubsection\<open>insertion resilience:
except at the beginning, no offer can be inserted\<close>
lemma chain_isnt_head [simp]: "L \ valid A n B \
head L \<noteq> chain (next_shop (head L)) ofr A L C" by (erule valid.induct, auto simp: chain_def sign_def anchor_def)
lemma insertion_resilience: "\L. L \ valid A n B \ Suc i < len L \<longrightarrow> ins (L,Suc i,M) \<notin> valid A n B"
supply [[simproc del: defined_all]] apply (induct i) (* i = 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,l'\ \ valid A n B" for x l', simp) apply (ind_cases "\x,M,l'\ \ valid A n B" for x l', clarsimp) apply (ind_cases "\head l',l'\ \ valid A n B" for l', simp, simp) (* i > 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,l'\ \ valid A n B" for x l') apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,ins(l',Suc na,M)\ \ valid A n B" for x l' na) apply (frule len_not_empty, clarsimp) by (drule_tac x=l' in spec, clarsimp)
subsubsection\<open>truncation resilience:
only shop i can truncate at offer i\<close>
lemma truncation_resilience: "\L. L \ valid A n B \ Suc i < len L \<longrightarrow> cons M (trunc (L,Suc i)) \<in> valid A n B \<longrightarrow> shop M = shop (ith (L,i))" apply (induct i) (* i = 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,l'\ \ valid A n B" for x l') apply (frule len_not_empty, clarsimp) apply (ind_cases "\M,l'\ \ valid A n B" for l') apply (frule len_not_empty, clarsimp, simp) (* i > 0 *) apply clarify apply (frule len_not_empty, clarsimp) apply (ind_cases "\x,l'\ \ valid A n B" for x l') apply (frule len_not_empty, clarsimp) by (drule_tac x=l' in spec, clarsimp)
subsubsection\<open>declarations for tactics\<close>
lemma p1_is_one_step [iff]: "one_step p1" unfolding one_step_def by (clarify, ind_cases "ev#evs \ p1" for ev evs, auto)
lemma p1_has_only_Says' [rule_format]: "evs \ p1 \
ev \<in> set evs \<longrightarrow> (\<exists>A B X. ev=Says A B X)" by (erule p1.induct, auto simp: req_def pro_def)
lemma priK_analz_Friend_imp_bad [rule_format,dest]: "\evs \ p1; Friend B \ A\ \<Longrightarrow> (Key (priK A) \<in> analz (knows (Friend B) evs)) \<longrightarrow> (A \<in> bad)" by auto
lemma priK_notin_knows_max_Friend: "\evs \ p1; A \ bad; A \ Friend C\ \<Longrightarrow> Key (priK A) \<notin> analz (knows_max (Friend C) evs)" apply (rule not_parts_not_analz, simp add: knows_max_def, safe) apply (drule_tac H="spies' evs"in parts_sub) apply (rule_tac p=p1 in knows_max'_sub_spies', simp+) apply (drule_tac H="spies evs"in parts_sub) by (auto dest: knows'_sub_knows [THEN subsetD] priK_notin_initState_Friend)
lemma agl_guard [intro]: "I \ agl \ I \ guard n Ks" by (erule agl.induct, auto)
lemma Says_to_knows_max'_guard: "\Says A' C \A'',r,I,L\ \ set evs;
Guard n Ks (knows_max' C evs)\ \ L \ guard n Ks" by (auto dest: Says_to_knows_max')
lemma Says_from_knows_max'_guard: "\Says C A' \A'',r,I,L\ \ set evs;
Guard n Ks (knows_max' C evs)\ \ L \ guard n Ks" by (auto dest: Says_from_knows_max')
lemma Says_Nonce_not_used_guard: "\Says A' B \A'',r,I,L\ \ set evs;
Nonce n \<notin> used evs\<rbrakk> \<Longrightarrow> L \<in> guard n Ks" by (drule not_used_not_parts, auto)
subsubsection\<open>guardedness of messages\<close>
lemma chain_guard [iff]: "chain B ofr A L C \ guard n {priK A}" by (case_tac "ofr=n", auto simp: chain_def sign_def)
lemma chain_guard_Nonce_neq [intro]: "n \ ofr \<Longrightarrow> chain B ofr A' L C \<in> guard n {priK A}" by (auto simp: chain_def sign_def)
lemma anchor_guard [iff]: "anchor A n' B \ guard n {priK A}" by (case_tac "n'=n", auto simp: anchor_def)
lemma anchor_guard_Nonce_neq [intro]: "n \ n' \<Longrightarrow> anchor A' n' B \<in> guard n {priK A}" by (auto simp: anchor_def)
lemma reqm_guard [intro]: "I \ agl \ reqm A r n' I B \ guard n {priK A}" by (case_tac "n'=n", auto simp: reqm_def)
lemma reqm_guard_Nonce_neq [intro]: "\n \ n'; I \ agl\ \<Longrightarrow> reqm A' r n' I B \<in> guard n {priK A}" by (auto simp: reqm_def)
lemma prom_guard [intro]: "\I \ agl; J \ agl; L \ guard n {priK A}\ \<Longrightarrow> prom B ofr A r I L J C \<in> guard n {priK A}" by (auto simp: prom_def)
lemma prom_guard_Nonce_neq [intro]: "\n \ ofr; I \ agl; J \ agl;
L \<in> guard n {priK A}\<rbrakk> \<Longrightarrow> prom B ofr A' r I L J C \<in> guard n {priK A}" by (auto simp: prom_def)
subsubsection\<open>Nonce uniqueness\<close>
lemma uniq_Nonce_in_chain [dest]: "Nonce k \ parts {chain B ofr A L C} \ k=ofr" by (auto simp: chain_def sign_def)
lemma uniq_Nonce_in_anchor [dest]: "Nonce k \ parts {anchor A n B} \ k=n" by (auto simp: anchor_def chain_def sign_def)
lemma uniq_Nonce_in_reqm [dest]: "\Nonce k \ parts {reqm A r n I B};
I \<in> agl\<rbrakk> \<Longrightarrow> k=n" by (auto simp: reqm_def dest: no_Nonce_in_agl)
lemma uniq_Nonce_in_prom [dest]: "\Nonce k \ parts {prom B ofr A r I L J C};
I \<in> agl; J \<in> agl; Nonce k \<notin> parts {L}\<rbrakk> \<Longrightarrow> k=ofr" by (auto simp: prom_def dest: no_Nonce_in_agl no_Nonce_in_appdel)
subsubsection\<open>requests are guarded\<close>
lemma req_imp_Guard [rule_format]: "\evs \ p1; A \ bad\ \
req A r n I B \<in> set evs \<longrightarrow> Guard n {priK A} (spies evs)" apply (erule p1.induct, simp) apply (simp add: req_def knows.simps, safe) apply (erule in_synth_Guard, erule Guard_analz, simp) by (auto simp: req_def pro_def dest: Says_imp_knows_Spy)
lemma req_imp_Guard_Friend: "\evs \ p1; A \ bad; req A r n I B \ set evs\ \<Longrightarrow> Guard n {priK A} (knows_max (Friend C) evs)" apply (rule Guard_knows_max') apply (rule_tac H="spies evs"in Guard_mono) apply (rule req_imp_Guard, simp+) apply (rule_tac B="spies' evs"in subset_trans) apply (rule_tac p=p1 in knows_max'_sub_spies', simp+) by (rule knows'_sub_knows)
subsubsection\<open>propositions are guarded\<close>
lemma pro_imp_Guard_Friend: "\evs \ p1; B \ bad; A \ bad;
pro B ofr A r I (cons M L) J C \<in> set evs\<rbrakk> \<Longrightarrow> Guard ofr {priK A} (knows_max (Friend D) evs)" apply (rule Guard_knows_max') apply (rule_tac H="spies evs"in Guard_mono) apply (rule pro_imp_Guard, simp+) apply (rule_tac B="spies' evs"in subset_trans) apply (rule_tac p=p1 in knows_max'_sub_spies', simp+) by (rule knows'_sub_knows)
subsubsection\<open>data confidentiality:
no one other than the originator can decrypt the offers\<close>
lemma Nonce_req_notin_spies: "\evs \ p1; req A r n I B \ set evs; A \ bad\ \<Longrightarrow> Nonce n \<notin> analz (spies evs)" by (frule req_imp_Guard, simp+, erule Guard_Nonce_analz, simp+)
lemma Nonce_req_notin_knows_max_Friend: "\evs \ p1; req A r n I B \ set evs;
A \<notin> bad; A \<noteq> Friend C\<rbrakk> \<Longrightarrow> Nonce n \<notin> analz (knows_max (Friend C) evs)" apply (clarify, frule_tac C=C in req_imp_Guard_Friend, simp+) apply (simp add: knows_max_def, drule Guard_invKey_keyset, simp+) by (drule priK_notin_knows_max_Friend, auto simp: knows_max_def)
lemma Nonce_pro_notin_spies: "\evs \ p1; B \ bad; A \ bad;
pro B ofr A r I (cons M L) J C \<in> set evs\<rbrakk> \<Longrightarrow> Nonce ofr \<notin> analz (spies evs)" by (frule pro_imp_Guard, simp+, erule Guard_Nonce_analz, simp+)
lemma Nonce_pro_notin_knows_max_Friend: "\evs \ p1; B \ bad; A \ bad;
A \<noteq> Friend D; pro B ofr A r I (cons M L) J C \<in> set evs\<rbrakk> \<Longrightarrow> Nonce ofr \<notin> analz (knows_max (Friend D) evs)" apply (clarify, frule_tac A=A in pro_imp_Guard_Friend, simp+) apply (simp add: knows_max_def, drule Guard_invKey_keyset, simp+) by (drule priK_notin_knows_max_Friend, auto simp: knows_max_def)
subsubsection\<open>non repudiability:
an offer signed by B has been sent by B\<close>
lemma Crypt_reqm: "\Crypt (priK A) X \ parts {reqm A' r n I B}; I \ agl\ \ A=A'" by (auto simp: reqm_def anchor_def chain_def sign_def dest: no_Crypt_in_agl)
lemma Crypt_prom: "\Crypt (priK A) X \ parts {prom B ofr A' r I L J C};
I \<in> agl; J \<in> agl\<rbrakk> \<Longrightarrow> A=B \<or> Crypt (priK A) X \<in> parts {L}" apply (simp add: prom_def anchor_def chain_def sign_def) by (blast dest: no_Crypt_in_agl no_Crypt_in_appdel)
lemma Crypt_safeness: "\evs \ p1; A \ bad\ \ Crypt (priK A) X \ parts (spies evs) \<longrightarrow> (\<exists>B Y. Says A B Y \<in> set evs \<and> Crypt (priK A) X \<in> parts {Y})" apply (erule p1.induct) (* Nil *) apply simp (* Fake *) apply clarsimp apply (drule_tac P="\G. Crypt (priK A) X \ G" in parts_insert_substD, simp) apply (erule disjE) apply (drule_tac K="priK A"in Crypt_synth, simp+, blast, blast) (* Request *) apply (simp add: req_def, clarify) apply (drule_tac P="\G. Crypt (priK A) X \ G" in parts_insert_substD, simp) apply (erule disjE) apply (frule Crypt_reqm, simp, clarify) apply (rule_tac x=B in exI, rule_tac x="reqm A r n I B"in exI, simp, blast) (* Propose *) apply (simp add: pro_def, clarify) apply (drule_tac P="\G. Crypt (priK A) X \ G" in parts_insert_substD, simp) apply (rotate_tac -1, erule disjE) apply (frule Crypt_prom, simp, simp) apply (rotate_tac -1, erule disjE) apply (rule_tac x=C in exI) apply (rule_tac x="prom B ofr Aa r I (cons M L) J C"in exI, blast) apply (subgoal_tac "cons M L \ parts (spies evsp)") apply (drule_tac G="{cons M L}"and H="spies evsp"in parts_trans, blast, blast) apply (drule Says_imp_spies, rotate_tac -1, drule parts.Inj) apply (drule parts.Snd, drule parts.Snd, drule parts.Snd) by auto
lemma Crypt_Hash_imp_sign: "\evs \ p1; A \ bad\ \
Crypt (priK A) (Hash X) \<in> parts (spies evs) \<longrightarrow> (\<exists>B Y. Says A B Y \<in> set evs \<and> sign A X \<in> parts {Y})" apply (erule p1.induct) (* Nil *) apply simp (* Fake *) apply clarsimp apply (drule_tac P="\G. Crypt (priK A) (Hash X) \ G" in parts_insert_substD) apply simp apply (erule disjE) apply (drule_tac K="priK A"in Crypt_synth, simp+, blast, blast) (* Request *) apply (simp add: req_def, clarify) apply (drule_tac P="\G. Crypt (priK A) (Hash X) \ G" in parts_insert_substD) apply simp apply (erule disjE) apply (frule Crypt_reqm, simp+) apply (rule_tac x=B in exI, rule_tac x="reqm Aa r n I B"in exI) apply (simp add: reqm_def sign_def anchor_def no_Crypt_in_agl) apply (simp add: chain_def sign_def, blast) (* Propose *) apply (simp add: pro_def, clarify) apply (drule_tac P="\G. Crypt (priK A) (Hash X) \ G" in parts_insert_substD) apply simp apply (rotate_tac -1, erule disjE) apply (simp add: prom_def sign_def no_Crypt_in_agl no_Crypt_in_appdel) apply (simp add: chain_def sign_def) apply (rotate_tac -1, erule disjE) apply (rule_tac x=C in exI) apply (rule_tac x="prom B ofr Aa r I (cons M L) J C"in exI) apply (simp add: prom_def chain_def sign_def) apply (erule impE) apply (blast dest: get_ML parts_sub) apply (blast del: MPair_parts)+ done
lemma sign_safeness: "\evs \ p1; A \ bad\ \ sign A X \ parts (spies evs) \<longrightarrow> (\<exists>B Y. Says A B Y \<in> set evs \<and> sign A X \<in> parts {Y})" apply (clarify, simp add: sign_def, frule parts.Snd) apply (blast dest: Crypt_Hash_imp_sign [unfolded sign_def]) done
end
¤ Dauer der Verarbeitung: 0.17 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.